关于.h .cpp和inline的讨论

简介:

很多书讲得不是很详细的。 
简单说:

.h里面是声明,编译器不会为其建立实体,仅仅是告知所有模块,这个世界上有这么个东东,函数,定义什么的,你们要用,可以,编译器会帮你连接到实体。 
.cpp里面就是实例了,编译器会产生真实的代码段,全局变量也会分配具体的单元(如果能分配的话)

如果在.h文件中强行定义实体,连接的时候,会报告错误,说有个命名拥有多个实体。 
比如你定义char* szData="aaa"; 
这个如果写到.h文件里面,则每个引用该.h的cpp产生的obj里面都有一个szD 
ata的实体,则link程序就晕菜了,不知道哪个算,就报错。 
因此,如果要定义这种具体需要分配单元的字符串变量,需要在.cpp里面定义,然后在.h中声明:char* szData;,看见没,这没有分配单元,只是提醒大家,符号表中有这个szData,大家可以用。 
可见,同样的char*szData,写在.h里面是声明,而写在.cpp里面,是定义。大家别弄混了。

反过来,.h里面声明了,所有的.cpp里面没有定义,则大家不用这个命名也就罢了,如果一用,link又晕菜了,找不到这个命名具体对应哪个地址单元,也会报错。

简单说 
对于变量,.h中可以声明,但是不要有赋初值动作,赋值放到.cpp里面去,就都ok了。 
对于函数,大家其实都知道.h里面声明,在.cpp里面定义,就好了,不过,按照c的习惯,在.h里面的声明前面最好加上extern修饰,这样,即使以后你的函数放在哪个lib或者dll、so里面,也是可以被正确link的。

这里又说一下inline。 
C和C++的实现,我的理解其实是不一样的。 
C里面,要求显式声明inline,而C++中,直接写到.h文件中,类的声明中就好了。 
不过,从编译器角度理解,二者差不多。 
在编译阶段,就是cc阶段,内联函数都是产生代码,但是这些代码是虚代码,本身不会被显式定义地址段,也不会正式进入符号表被调用,而是类似宏一样,被直接展开到每个调用的地方。 
因此到了link阶段,是根本看不到这些函数的,符号表中没有,就好像没有它一样,它已经在前一阶段被自动展开,作为每个调用函数的一部分,编译成原始代码了。 
而普通的函数,link的过程,基本上就是查符号表,安排基栈地址什么的,然后在调用处写上一个call。 
这种先展开和后定址的差异,应该是inline函数和普通函数最大的差异性了。 
嗯,还有一种方法我比较喜欢用。 
不管是inline函数还是普通函数,还是什么变量。 
我在某个cpp里面实现了,但是不在.h里面声明。 
我一直等待用它的时候,在用的函数的cpp实现前面临时声明。这么做的好处是减少内部定义暴露在外的可能性,减少耦合性。

比如我在a.cpp里面定义了函数Func1和char* szData。 
void Func1(void) 

    //... 

char* szData="aaa";

但是我不在a.h里面声明。

b.cpp里面,有个函数Func2要用的时候,我再来声明

void Func1(void); 
char* szData;  //看见没,临时声明 
void Func2(void) 

    //... 
    strcpy(szBuffer,szData); 
    Func1(); 

反正谁用谁声明,不用的别看,这样写,程序bug少很多的。

嗯,如果是inline函数,由于函数实体就是声明,因此,这种用法的话,需要把inline函数全部拷贝一遍过来,而不能简单写个声明了事。

不过,拷贝应该不难吧?呵呵

总的来说,写.h文件有时候要慎重一点,懒一点,能不声明就不声明,因为以后用的人,可能会乱想乱用,造成不必要的bug。

这也说明,其实C和C++一样,都可以做到模块化的内聚编程的,C++呢,通过private,强行拒绝外部的无关使用,告诉用户,许看不许摸。C呢,通过严格控制声明,连看都不准看,呵呵,都是为了保护程序安全。 
内联函数,很多编译器会刻意忽视inline这个声明,如果程序需要用inline优化,需要手动打开,这个要查一下具体的语言编译器的说明。 
gcc下的说明请参考这里 
http://qzone.qq.com/blog/115547829-1219133815 
和这里 
http://blog.csdn.net/hngsc_0/archive/2008/12/13/3509952.aspx

VC基本上是无视inline声明的,它把inline作为自己的手段来控制,比如选择尺寸最大,速度最快时,inline就有效,选择尺寸最小,inline就无效了。可以看看这里。 
http://hi.baidu.com/joyjjjz/blog/item/e9d2347ab62671e92f73b305.html

inline的使用,应该是优化到了极致时采用的一种方法,其提升性能的能力比较有限,很多时候,远远不及一个算法优化来得明显,其实如果不是做高性能程序的话,就没有必要考虑了。

嗯,VC我一般用来做验证,高性能优化主要是针对gcc的,所以对inline在VC下的表现一般不太关心。这是因为Windows本身就是个很重的操作系统,没有什么实时性,也不太适合做高性能应用。如果真的一个应用已经到了需要用inline做优化的地步,一般首选平台都是Linux了,小机器一般就选VxWorks了。 
呵呵,一家之言,欢迎拍砖哈。

本文转自 tonyxiaohome  51CTO博客,原文链接:http://blog.51cto.com/tonyxiaohome/198747 ,如需转载请自行联系原作者

相关文章
|
机器学习/深度学习 人工智能 网络协议
CPP2022-23-函数进阶-函数指针
CPP2022-23-函数进阶-函数指针
260 0
|
7月前
|
存储 C语言
C语言学习记录——复习数据存储类别(auto、register、static、extern)
C语言学习记录——复习数据存储类别(auto、register、static、extern)
60 1
|
8月前
|
编译器 C++ 开发者
[C++从入门到精通] 2.inline内联函数、const的相关用法
[C++从入门到精通] 2.inline内联函数、const的相关用法
75 0
|
前端开发 容器
【前端】一文读懂display: inline-block
【前端】一文读懂display: inline-block
488 2
|
安全 编译器 C语言
【C++】初阶 --- 内联函数(inline)
【C++】初阶 --- 内联函数(inline)
91 0
|
存储 安全 编译器
[C++] C++入门第二篇 -- 引用& -- 内联函数inline -- auto+for(上)
[C++] C++入门第二篇 -- 引用& -- 内联函数inline -- auto+for(上)
|
存储 编译器 C语言
[C++] C++入门第二篇 -- 引用& -- 内联函数inline -- auto+for(下)
[C++] C++入门第二篇 -- 引用& -- 内联函数inline -- auto+for(下)
|
编译器 C++
【C++知识点】内联函数inline
【C++知识点】内联函数inline
80 0
|
算法 编译器 C语言
C++__万能头文件bits/stdc++.h的优缺点
C++__万能头文件bits/stdc++.h的优缺点
409 0