C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」C语言中,数组初始化的方式主要有三种:1、声明时,使用{0}初始化;2、使用memset;3、用for循环赋值。那么,这三种方法的原理以及效率如何呢?请看下面的测试代码:[cpp]viewplaincopy#defineARRAY_SIZE_MAX(1*1024*1024)voidfunction1(){chararray[ARRAY_SIZE_…

大家好,又见面了,我是你们的朋友全栈君。

C语言中,数组初始化的方式主要有三种:

1、声明时,使用 {0} 初始化;

2、使用memset;

3、用for循环赋值。

那么,这三种方法的原理以及效率如何呢? 请看下面的测试代码:

[cpp] 
view plain 
copy

  1. #define ARRAY_SIZE_MAX  (1*1024*1024)  
  2.   
  3. void function1()  
  4. {  
  5.     char array[ARRAY_SIZE_MAX] = {0};  //声明时使用{0}初始化为全0  
  6. }  
  7.   
  8. void function2()  
  9. {  
  10.     char array[ARRAY_SIZE_MAX];  
  11.     memset(array, 0, ARRAY_SIZE_MAX);  //使用memset方法  
  12. }  
  13.   
  14. void function3()  
  15. {  
  16.     int i = 0;  
  17.     char array[ARRAY_SIZE_MAX];  
  18.     for (i = 0; i < ARRAY_SIZE_MAX; i++)  //for循环赋值  
  19.     {  
  20.         array[i] = 0;  
  21.     }  
  22. }  

效率:

分别执行上面三种方法,统计下平均时间可以得出:  for循环浪费的时间最多,{0} 与memset 耗时差不多。

原理:

1、for循环,就是循环赋值,不解释了

2、memset,很容易找到memset内部实现代码,这里也不解释了

3、{0} 内部是怎么实现的呢?

将上述代码编译成汇编格式如下:

function1如下:

[cpp] 
view plain 
copy

  1. pushl   %ebp  
  2. movl    %esp, %ebp  
  3. subl    $1048600, %esp  
  4. leal    -1048584(%ebp), %eax  
  5. movl    $1048576, %edx  
  6. movl    %edx, 8(%esp)  
  7. movl    $0, 4(%esp)  
  8. movl    %eax, (%esp)  
  9. call    memset  
  10. leave  
  11. ret  

function2如下:

[cpp] 
view plain 
copy

  1. pushl   %ebp  
  2. movl    %esp, %ebp  
  3. subl    $1048600, %esp  
  4. movl    $1048576, 8(%esp)  
  5. movl    $0, 4(%esp)  
  6. leal    -1048584(%ebp), %eax  
  7. movl    %eax, (%esp)  
  8. call    memset  
  9. leave  
  10. ret  

通过汇编代码可以看出,{0}初始化方式,调用了memset函数!

对三种方法的选取:

1、for 最浪费时间,不建议(其实memset内部也是用循环实现的,只不过memset经过了严格优化,所以性能更高);

2、{0} 可能有移植性问题,虽然绝大多数编译器看到{0} 都是将数组全部初始化为0, 但是不保证所有编译器都是这样实现的;

3、综合1、2, 推荐使用memset方法。

附录:对于{0}初始化的测试

这是很基础的东西,但基础的重要性不言而喻,我敢肯定这个知识点我肯定曾经了解过,但现在,我不敢确定,由此可见纪录的重要性,这世界没有什么捷径,找对方向,然后不停重复.所以从今天开始,我会比较详细的纪录这些比较小的知识点,其实还是有不少有意思的地方的.

    写这篇文章的起因在于<<COM技术内幕>>第七章新东西太多,看的我目不暇接,所以在网上找了些例子看,其中就有一个例子中出现了这样的语句: 

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」



wchar_t wname[128]={0};

char cname[256]={0};

我感兴趣的是:
1.这种赋值的结果.
2.这种形式是否符合标准编码规则?

我找到了如下资料,可能有助于对这个知识点的掌握.

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」

/*

初始化值的个数可少于数组元素个数.当初始化值的个数少于数组元素个数时,前面的按序初始化相应值, 后面的初始化为0(全局或静态数组)或为不确定值(局部数组).

*/

我相信上面的资料是C和C++语言的标准规范,但实际编译器处理时,可能会和规范有所不同.因为编译器原则上要遵从语言规范,但对于局部数组的不确定值到底是多少,怎么处理,编译器就可以灵活处理.我测试了三种编译器,其实编译器赋予的值是固定的,都是0.

在这篇blog中 如流,新一代智能工作平台 谈论了相同的话题,现对其摘录如下: 

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」

/*

一直以为 int a[256]={0};是把a的所有元素初始化为0,int a[256]={1};是把a所有的元素初始化为1.

调试的时查看内存发现不是那么一回事,翻了一下《The C++ Programming Language》总算有定论。PDF的竟然不然复制,就把它这章翻译了,如下

5.2.1   数组初始化 

数组可以用一个列值来初始化,例如

         int v1[] ={1,2,3,4};

         char v2[]={‘a’,’b’,’c’,0};

当数组定义时没有指定大小,当初始化采用列表初始化了,那么数组的大小由初始化时列表元素个数决定。所以v1和v2分别为 int[4] 和char[4]类型。如果明确指定了数组大小,当在初始化时指定的元素个数超过这个大小就会产生错误。例如:

         char   v3[2] ={‘a’,’b’,0};   //错误:太多的初始化值了

         char   v3[3] ={‘a’,’b’,0};   //正确

如果初始化时指定的的元素个数比数组大小少,剩下的元素都回被初始化为   0。例如

         int   v5[8]={1,2,3,4};

等价于

          int   v5[8]={1,2,3,4,0,0,0,0};

注意没有如下形式的数组赋值:

         void f()

         {

             v4={‘c’,’d’,0};   //错误:不是数组赋值

         }

如果你想这样的复制的话,请使用 vector(16章第三节) 或者 valarray(22章第四节)。

        字符数组可以方便地采用字符串直接初始化(参考第五章 2.2小节)

         译注: 就是 这样啦   char   alpha []=”abcdefghijklmn”;

*/

下面来看一个例子:

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」

#include <iostream.h>

int array1[5]={1,2,3};

static int array2[5]={1};

void main()

{

    int arr1[5]={2};

    static int arr2[5]={1,2};

    

    int n;

    cout <<“global: “;

    for(n=0; n<5; n++)

        cout <<” ” <<array1[n];

    

    cout <<” global static: “;

    for(n=0; n<5; n++)

        cout <<” ” <<array2[n];

    

    cout <<” local: “;

    for(n=0; n<5; n++)

        cout <<” ” <<arr1[n];

    

    cout <<” local static: “;

    for(n=0; n<5; n++)

        cout <<” ” <<arr2[n];

    cout <<endl;

}

 

在这个例子中,全局和静态数组都按语言规范要求被初始化为0,但是局部数组并没有向前面所说的为不确定值,下面是用gcc,VC6.0,tuborC++分别编译的结果(注意gcc用g++编译c++文件,gcc不会链接库的):

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」

/*

GCC 可同时用来编译 C 程序和 C++ 程序。一般来说,C 编译器通过源文件的后缀名来判断是 C 程序还是 C++ 程序。在 Linux 中,C 源文件的后缀名为 .c,而 C++ 源文件的后缀名为 .C 或 .cpp。

    但是,gcc 命令只能编译 C++ 源文件,而不能自动和 C++ 程序使用的库连接。因此,通常使用 g++ 命令来完成 C++ 程序的编译和连接,该程序会自动调用 gcc 实现编译。

*/

GCC:

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」


VC6.0:

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」


TurboC++

C语言 数组初始化的三种常用方法({0}, memset, for循环赋值)以及原理「建议收藏」


    这说明了对局部数组没有初始化的元素的值,这几种编译器都将其设置为0.但是,如果如果不对数组进行初始化,即在定义的同时没有用列表初始化,那么局部数组的值就取决于编译器而对程序员来说就是不可预料的了.有时间可以测试一下各个编译器,不过在vc中是0xcc.所以对局部数组的初始化要特别小心.但是全局的数组和静态数组还是会被正确的赋于0值的.

    最后要重申下对变量初始化的重要性, http://blog.vckbase.com/smileonce/archive/2005/06/18/6777.html  这里列举了没有初始化造成的事故.

    此外,这个blog地址值得收藏,在http://blog.vckbase.com/ 排行榜的blog都值得仔细看.

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/161038.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • idea 添加Tomcat_懂车帝怎么添加

    idea 添加Tomcat_懂车帝怎么添加使用IDEA编辑器开发项目十分便捷,这里介绍使用IDEA编辑器添加Tomcat1、新建web工程这里有一个已经创建好的web项目2、配置tomcat配置tomcat前,先确保本地已经下载并安装完成了tomcat如果不清楚如何安装tomcat,请参考:安装tomcat点击Run,EditConfigurations…点击+号,添加服务配置找到TomcatServer,选择Local自定义Name,这里是T…

    2022年10月17日
    3
  • 详细教你如何部署ICE服务(一)

    详细教你如何部署ICE服务(一)这系列文章将会一步步教你如何部署一个ICE服务,如果你正在读这篇博客,我想你已经了解了什么是ICE(InternetCommunicationsEngine),以及如何去实现ICE服务,并且了解什么是ICE对象、ICE对象标识符、ICE对象适配器、ICE服务实现servant、ICE通信器等概念,当然如果你连什么是ICE都不知道,我不建议你继续读下去。先说一下ICE的基本组件:(1)

    2022年5月30日
    32
  • mysql左连接查询举例_mysql左右连接查询(有示例图)

    mysql左连接查询举例_mysql左右连接查询(有示例图)小小的连接查询,其实里面有很多学问,今天我就来简单剖析一下。左连接,右连接,内连接的本质:将两个表的数据依据一定条件横向连接起来。给出建表语句:————————————————–createtabletest1(idint,namevarchar(10))createtabletest2(idint,namevarc…

    2022年5月8日
    46
  • JAVA毕业设计_毕业设计外文翻译范例

    JAVA毕业设计_毕业设计外文翻译范例计算机专业毕业设计论文外文文献中英文翻译——java对象1.IntroductionToObjects1.1TheprogressofabstractionAllprogramminglanguagesprovideabstractions.Itcanbearguedthatthecomplexityoftheproblemsyou’reable…

    2022年9月1日
    5
  • 21计算机保研经验分享

    21计算机保研经验分享保研最终去向:哈工大威海-计算机个人情况:学校是211计算机弱校,rank7%;个人有数学建模,小程序,网安的省级奖,几个小科研项目,一段工作室经历,擅长后端搬砖。无论文;自我感觉算是保研er水平一般的,我这个去向怎么样啊,欢迎留言面经:吉林大学软件工程+哈工大威海计算机面试经验分享马上写好一、夏令营夏令营经历:北理工网安入营+时间冲突放弃,只能说非常可惜;吉林大学软件入营+优秀营员;哈工大威海计算机入营+面试合格(共投递11所学校学院,只有两个真正参加,但万幸都有收获)吉林大学软件工程

    2022年5月9日
    58
  • EasyPlayer视频源切换

    EasyPlayer视频源切换EasyPlayer现在支持多视频源快速切换了,我们介绍一下是如何实现的.这个需求通常应用在一个客户端需要查看多个视频源的情况,比如多个监控场景轮播.由于EasyPlayer的播放端已经放在Fragment了,这使得SDK层可以对好多应用层的代码做以封装,这样尽可能给开发者带来方便.比如,两个视频源切换的时候,只需要创建两个PlayFragment,然后对这两个Fragment进行显示切换,如代

    2022年6月17日
    35

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号