现在的位置: 首页 > 综合 > 正文

C++主题年技巧积累#2——我被static撞了一下腰

2013年10月21日 ⁄ 综合 ⁄ 共 4603字 ⁄ 字号 评论关闭
 

C++主题年技巧积累#2——我被static撞了一下腰

CSDN旗下水之真谛http://blog.csdn.net/FantasiaX)出品

 

前传:

         刚刚参加博文视点出版社三周年庆典回来,兴奋之余想到今天还没有更新Blog,于是跑上来更新一下——我尽量“好好学习,天天上博”。哎呀,今天见到好多名人啊!先是在金戈老师旁边坐下,然后又去问候了久仰大名的孟岩老师,在孟老师的帮助下又找到《Beginning C# Object》的译者,也是本次晚会的摄像师——韩磊老师,你问韩磊唱歌没?没有!歌被欧阳璟(《程序员》的老编,大帅哥!还是本次年会的摄影师)给唱了。晚会的精彩那没的说,大家等着看视频吧,估计CSDN上不久就会放出来。会间在杨福川的带领下,先是见到了梁晶编辑,然后又见到了亲自为我斧正译文的方舟老师。方舟老师人真不错,几分钟里还抓紧时间教我应该怎么译技术文章,怎么理顺段落、句型,如何重组定语、去掉不必要的连词……在这里,我深深地向您鞠上一躬,道一声:谢谢!同时也感谢梁编辑的不懈努力!接下来,我有幸见到了博文视点的周筠老师和CSDN的总裁蒋涛先生。过程见还见到了很多平时“只见MSN不见人”的朋友,其中包括帅哥佘广。最后还有机会见到技术专家金旭亮老师。

         庆典参加后,感觉博文视点出版社和CSDN(包括《程序员》杂志)真是两个充满激情、活力四射的团队!衷心地祝福博文视点出版社出版更多优秀的技术书籍——就像席间韩磊老师说的:出一本是一本,不糟踏书,人家买书,是真想学东西啊!也祝CSDN和《程序员》杂志越办越红火!

打油诗一首,赠予辛勤工作在出版一线的朋友们,祝你们的2007年万事如意!诗名《日出》,意思是希望博文视点出版社能“天天出书,天天出好书”——夫,日出,日日出,又日出!

日出

    学海茫茫寻师苦,

    博文乐把众人渡。

    书香满载齐挥桨,

    破浪乘风向日出!

 

小序:

         咳咳,收心啦收心啦!热闹是人家的,知识是自己的。还是收回心来写自己的技术心得吧。话说“由俭入奢易,由奢入俭难”啊……这话一点不假,连写程序都是这样。像我,以前用惯了语法舒服流畅的C#,再回过头来学C++就深刻地体验到了这一点:C#之所以用起来简洁舒服,是因为它把很多应该由程序员自己做的事情(比如分配和释放内存)都替程序员做了。换句话说,C#C/C++的设计理念是不一样的,C/C++假设程序做的一切事情(包括某些事情不去做)都是正确的,相信程序员是聪明人,决不为程序员多做任何事情;C#正好反过来,认为程序员应该不去关心底层这些东西、由程序员关注底层是容易出错的、程序员总会忘记在该分配内存的地方分配内存或者在该释放内存的地方释放内存(微软是怀疑程序员都是傻瓜还是打算把程序员都培养成傻瓜?),总之,程序员应该更多地去关心架构和实现,而不是这些细枝末节的东西……

         于是,我被培养成了一个不折不扣的“傻瓜型”程序员。所以,今天也就理所当然地被static关键字撞了一下腰、绊了一个跤。

 

正文:

         一切都源于我在写练习程序时那一瞬间的妄想……

         static关键字?小菜,C#C++里都有,原理是一样的,会了C#还写不出C++的来?看着!咱这就一样写一个出来!

         先来C#的!

//----------------水之真谛----------------//
//  http://blog.csdn.net/FantasiaX
//-----------------------------------------------//

using System;
class
Student
{
    public static void Report() // C#
中成员方法的声明和定义合二为一
    {
        Console.WriteLine("I am a C# student.");
    }
}

class Program
{
    static void Main(string[] args)
    {
        Student.Report();
// OK
    }
}

         再来C++的!

#include

class Student
{
public:
         static void Report()    // C++
成员函数的声明和定义亦可合二为一,但决不是上策!
         {
                   std::cout"I am a C++ student."
         }
};

int main(int argc, char *argv[])
{
         Student::Report();
// OK
         return 0;
}


         哈哈!一次编译通过!我说吗,会C#就会C++,没问题的,只是语法结构上稍有不同。

         接下来再来两个静态成员变量的例子!还是先写C#

//----------------水之真谛----------------//
//  http://blog.csdn.net/FantasiaX
//-----------------------------------------------//

using System;
class
Student
{
    public static string Name; //
猜猜在这里C#都做了些什么?
}

class Program
{
    static void Main(string[] args)
    {
        Student.Name = "Tim"; 
// OK
        Console.WriteLine(Student.Name);
    }
}

         乘胜追击,再来C++的!

#include
#include

class Student
{
public:
         static std::string Name;  //
猜猜C++在这里做什么了?
};

int main(int argc, char *argv[])
{
         Student::Name = "Tim"; 
//Error!!!   狂汗~~~ >_         std::cout         return 0;
}
         ~~~竟然有错误!这是怎么回事呢?不应该啊~~~C#里明明能行的~~~

         百思不得其解后,拿出《C++ Primer 第四版(中文)》来查阅……果然找到很多有用信息。

l         P.399 Row4-5.         ……static数据成员独立于该类的任意对象而存在……

l         P.399 Row5-6.         ……每个static数据成员是与类关联的对象,并不与该类的对象关联……

l         P.401 Row1.             ……static数据成员必须在类定义体的外部定义(正好一次)。

注解:数据成员(Data Member)其实就是成员变量。

         上面最有用的信息就是第三条了:原来还需要在类的外部定义一次!于是修改代码为:

#include
#include

class Student
{
public:
         static std::string Name;  //
猜猜C++在这里做什么了?
};

std::string Student::Name = std::string();  // 类外定义,这句是新添加的。

int main(int argc, char *argv[])
{
         Student::Name = "Tim"; 
// OK
         std::cout         return 0;
}


         问题虽然解决了,但决不能轻易放过这个问题——为什么会这样呢?

 

深挖:

         我把目光移到在《C++ Primer》里找到的前两条消息上。第二条与C#一致,说明不了什么问题,第一条倒是让人眼前一亮——static数据成员独立于该类的任意对象而存在。果真是这样吗?我得自己动手验证一下:

         先写一个类:

class Student
{
public:
         int Age;
};

        
然后执行 std::coutsizeof(Student)得到的结果是4

         把类改写成这样:

class Student
{
public:
         int Age;
         unsigned int StudentID;
};
        
结果是8——预料之中。

         再改写成:

class Student
{
public:
         int Age;
         unsigned int StudentID;
         double Score;
};

        
结果是16——可以理解。

         然后加一个static成员试试——

class Student
{
public:
         int Age;
         unsigned int StudentID;
         double Score;
         static int Amount;
};

        
结果还是16,没有变!原来static的成员数据真的是独立于类空间之外的

         问题解决了吗?No!不但没有解决,反而增多了,一个问题变成了两个问题——

1.         类所占的空间本质是什么?

2.         类的static数据成员究竟在哪里?

第一个问题似乎比较好回答:说“类所占的空间”其实欠妥,应该是“类的实例占内存的大小”。

拿上面这个类来说,它的每一个实例中都将包含三个子数据成员(int Age, double Score, double Score),所以会占去16个字节。而这16个字节将会由new操作符在内存中分配出来,并可以在Student类的构造函数去初始化它们——我们没有显式地初始化它们,所以它们的值是一个由上帝掷骰子得出来的值(使用VC执行的时候如果弹出错误警告,请按Ignore,我得到的Age值是-858993460,阴寿乎?)。

问题回答完了吗?没有!

提问:分配了16个字节的内存说明了什么问题?

回答:说明每个非static成员变量都有自己的内存,叠加起来占16字节。

提问:每个非static成员变量都有自己的内存说明了什么问题?

回答:说明为每个变量都分配了内存(哪儿来的砖??!!)

提问:为变量分配了内存说明了什么?

回答:说明每个非static变量不但已经被声明,而且已经被定义了!因为声明变量并不分配内存,只有定义变量的时候才分配内存!

         这才是第一个问题的本质的答案!

 

         第二个问题在《C++ Primer》这本书里就找不到答案了——毕竟是Primer,不是Advanced。于是又祭出宝卷《深入探索C++对象模型》。一番查阅后,在这里找到半个答案:

l         P. 95 Static Data Members        Static data members按其字面意义,被编译器提出于class之外……并被视为一个global变量……每一个static data member只有一个实体,存放在程序的data segment之中

抱歉!评论已关闭.