C++编译与链接(0)-.h与.cpp中的定义与声明

C++中有的东西需要放在可以在.h文件中定义,有的东西则必须放在.cpp文件中定义,有的东西在不同的cpp文件中的名字可以一样,而有的则不能一样

那么究竟哪些东西可在头文件中定义,声明,哪些东西又必须在.cpp中定义,声明呢?

*以下所有的讨论都是在全局命名空间中(即不定义自己的namespace)下进行的

函数

1、在.h中只能声明函数,在.cpp中可以声明与定义函数

如果在.h中声明并定义一个函数,则该函数只能被#include一次,否则则会出现重定义错误

比如

1.h

#pragma once

void show()
{

}

a.cpp

#include "1.h"

b.cpp

#include "1.h"

error LNK2005: "void __cdecl show(void)" ([email protected]@YAXXZ) 已经在 a.obj
中定义

所以要避免在头文件中定义函数

2、在不同.cpp中定义的函数原型(函数返回值,函数名称,函数参数)不能完全一样,

比如如果有在两个.cpp文件中均存在

void show(){};

会出现重定义错误

内联函数

为了确保所有调用该inline函数的文件中的定义一样,所以需要是在.h文件中定义

注意这里的inline对于编译器来说只是建议性的,关于该内联函数被拒绝会在下一篇文章中介绍

typedef

在不同的cpp中可以一样

变量

1、在.h中只能声明,在.cpp中可以声明与定义一个变量

如果在.h中的定义一个变量,则该变量被include两次以上时则会出现重定义错误

2、在不同.cpp中定义的变量的名字与类型不同一样

常量

1、如果const常量是用常量表达式进行初始化的,则可以在.h中声明与定义

2、如果const变量是用非常量表达式进行初始化的,那么该变量应该在cpp文件中定义,而在.h文件中进行声明。

3、不同cpp中以定义名字与类型一样的变量

static变量

1、在不同的cpp中可以定义名字与类型一样的变量

2、如果在.h中定义一个static成员,则所有include该文件的文件均拥有一份独立的该static成员,一个文件对其的修改不会影响到另一个文件

所以static变量一般是放在.cpp出现并定义.

例如

1.h

#pragma once

static int a = 5;

a.cpp


#include "1.h"
#include <iostream>
using namespace std;

void showstatic()
{
cout << "In a.cpp:" << a << endl;
a = 1;
cout << "In a.cpp:" << a << endl;
}

b.cpp


#include "1.h"
#include <iostream>
using namespace std;

void showstatic();
int main()
{
showstatic();
cout << "In b.cpp:" << a << endl;
system("pause");
}

static函数

在不同的cpp中可以定义函数原型一样的函数

不同的cpp中类的名字可以一样

类成员与函数

在.h中定义,所有成员必须在类中声明,在cpp中实现

非静态的常量整形数据成员不能就地初始化(*C++11中,标准允许使用等号=或者花括号{}进行就地的非静态成员变量初始化)

在类内部定义的成员函数将自动作为inline处理

在.h外部定义的函数需要加上inline说明

否则在被include多次时会出现重定义错误

1.h


#pragma once
#include <iostream>

class A
{
public:
void show();
};
void A::show()//无inline
{
std::cout << "hello" << std::endl;
}

a.cpp

#include "1.h"
#include <iostream>
using namespace std;

b.cpp

#include "1.h"
#include <iostream>
using namespace std;

error LNK2005: "public: void __thiscall A::show(void)" ([email protected]@@QAEXXZ) 已经在
a.obj 中定义

类的const成员

在类中声明变量为const类型的成员不可以就地初始化

const常量的初始化必须在构造函数初始化列表中初始化,而不可以在构造函数函数体内初始化(*C++11中,标准允许使用等号=或者花括号{}进行就地的非静态成员变量初始化)

#pragma once
class A
{
public:
const int i=50;
};

error C2864: “A::i”: 只有静态常量整型数据成员才可以在类中初始化   
d:\我的资料库\documents\visual studio 2010\projects\fasd\fasd\1.h   
5    1    fasd

类的静态的数据成员

不可以就地初始化,需要到.cpp中进行定义

(对于非常量的静态成员变量,C++11与C++98保持了一致。需要到头文件以外去定义它)

类的静态的常量整形数据成员

可以就地初始化

class A
{
private:
const static int i = 5;
};

模板(不考虑export)

模板函数与模板类的声明与实现必须放在一个文件中

总结











































  是否可以在.h中定义 在不同.cpp中是否可以重名 特殊说明
函数 不可以,会出现重定义错误 不可以  
内联函数 可以 可以 为了确保所有调用该inline函数的文件中的定义一样,所以需要是在.h文件中定义
typedef ---------------------- 可以  
常量 可以 可以

1、常量表达式进行初始化的,则可以在.h中声明与定义

2、非常量表达式进行初始化的,那么该变量应该在cpp文件中定义,而在.h文件中进行声明。

变量 不可以,会出现重定义错误 不可以(类型与名字)  
static变量 可以 可以

在.h中定义一个static成员,则所有include该文件的文件均拥有一份独立的该static成员,一个文件对其的修改不会影响到另一个文件

所以static变量一般是放在.cpp出现并定义.

static函数 可以 可以  






































  是否可以在.h中定义 是否可以就地初始化 特殊说明
可以    
类数据成员 ------------------ 不可以 (*C++11中,标准允许使用等号=或者花括号{}进行就地的非静态成员变量初始化)
类成员函数 ------------------ ----------------

在.h外部定义的函数需要加上inline说明

否则在被include多次时会出现重定义错误

类const数据 ------------------ 不可以

1、在类中声明变量为const类型的成员不可以就地初始化

const常量的初始化必须在构造函数初始化列表中初始化,而不可以在构造函数函数体内初始化

2、同类数据成员中的特殊说明

类的静态的数据成员 ------------------- 不可以

不可以就地初始化,需要到.cpp中进行定义

(对于非常量的静态成员变量,C++11与C++98保持了一致。需要到头文件以外去定义它)

类的静态的常量整形数据成员

------------------ 可以  








  特殊说明
模板

模板函数与模板类的声明与实现必须放在一个文件中

至于为什么会这样,与C++的编译和链接,和编译产生的目标文件(.obj),内部链接,外部链接有关,

我会在下一篇文章中向大家介绍

C++编译与链接(0)-.h与.cpp中的定义与声明,码迷,mamicode.com

时间: 2024-10-21 20:59:57

C++编译与链接(0)-.h与.cpp中的定义与声明的相关文章

C++编译与链接(2)-浅谈内部链接与外部链接

发现每次写技术博客时,都会在文章开头处花费一番功夫 ...从前,有一个程序员....他的名字叫magicsoar 为什么有时会出现aaa已在bbb中重定义的错误? 为什么有时会出现无法解析的外部符号? 为什么有的内联函数的定义需要写在头文件中? 为什么对于模板,声明和定义都在写在一起? 读完这篇博客,相信你会有一个初步的认识 注,我们现在谈的编译其实可以认为由4个环节组成,其中有编译环节,链接环节, 我会尽量在上下文中指明说的总体的编译,还是具体的编译环节,望读者周知 关于编译过程详解说明,可以

读书笔记:程序员的自我修养-----第二章(编译和链接)

自己之前一直以为目标文件是经过汇编生成.s之后编译生成的,好菜. 源程序到可执行程序过程:预处理  编译 汇编 链接 预处理: 展开宏定义,处理条件预编译指令,插入头文件,删除注释,添加行号和文件名标示,保留#pragma编译器指令. 编译: 词法分析:词法扫描器按照词法规则产生记号,根据记号放入到相应表中.比如讲标示符放入符号表,将数字和字符串常量防盗文字表等. 语法分析:产生以表达式为节点的语法树 语义分析:语义分析器所能分析的只是静态语义(声明,类型的匹配,类型的转换),语义分析之后,每个

c++ *.h和*.cpp在编译中的作用

首先,我们可以将所有东西都放在一个.cpp文件内.然后编译器就将这个.cpp编译成.obj,obj是什么东西?就是编译单元了. 一个程序,可以由一个编译单元组成,也可以有多个编译单元组成. 如果你不想让你的源代码变得很难阅读的话,就请使用多个编译单元吧.(一个函数不能放到两个编译单元里面,但两个以上就可以分别放在一个单元,也就是cpp里面)那么就是一个.cpp对应一个.obj,然后将所有的obj链接起来(通过一个叫链接器的程序),组成一个.exe,也就是程序了. 如果一个.cpp要用到另一个.c

编译原理 (预处理&gt;编译&gt;汇编&gt;链接)(转)

一般高级语言程序编译的过程:预处理.编译.汇编.链接.gcc在后台实际上也经历了这几个过程,我们可以通过-v参数查看它的编译细节,如果想看某个具体的编译过程,则可以分别使用-E,-S,-c和 -O,对应的后台工具则分别为cpp,cc1,as,ld.下面我们将逐步分析这几个过程以及相关的内容,诸如语法检查.代码调试.汇编语言等. 1.预处理 预处理是C语言程序从源代码变成可执行程序的第一步,主要是C语言编译器对各种预处理命令进行处理,包括头文件的包含.宏定义的扩展.条件编译的选择等.打印出预处理之

linux下编译qt5.6.0静态库——configure配置

 随笔 - 116  文章 - 4  评论 - 7 linux下编译qt5.6.0静态库--configure配置 linux下编译qt5.6.0静态库 linux下编译qt5.6.0静态库 configure生成makefile 安装选项 Configure选项 第三方库: 附加选项: QNX/Blackberry选项: Android 选项: 生成makefile 遇到链接检查失败的情况 生成makefile后进行编译 编译时的错误 多重定义'QT_MODBUS()'和'QT_MODBU

《程序员的自我修养》 第二章——编译和链接

摘自http://blog.chinaunix.net/uid-26548237-id-3839979.html <程序员的自我修养>第二章——编译和链接 2.1 被隐藏了的过程    C语句的经典,“Hello World”程序几乎是每个程序员闭着眼睛都能写出的,编译运行一气呵成,基本成了程序入门和开发环境测试的默认标准. #include <stdio.h> int main() { printf("Hello World\n"); return 0; 在L

C++编译与链接(1)-编译与链接过程

大家知道计算机使用的一系列的1和0 那个一个C++语言程序又是如何从一个个.h和.cpp文件变成包含1和0的可执行文件呢? 可以认为有以下的几个阶段 源程序->预处理->编译和优化->生成目标文件->链接->可执行文件 1.预处理 C++的预处理是指在C++程序源代码被编译之前,由预处理器对C++程序源代码进行的处理.这个过程并不对程序的源代码进行解析. 这里的预处理器(preprocessor)是指真正的编译开始之前由编译器调用的一个独立程序. 预处理器主要负责以下的几处

详解编译、链接

被隐藏了的过程    现如今在流行的集成开发环境下我们很少需要关注编译和链接的过程,而隐藏在程序运行期间的过程可不简单,即使使用命令行来编译一个源代码文件,简单的一句"gcc hello.c"命令就包含了非常复杂的过程. 1 #include<stdio.h> 2 3 int main() 4 { 5 printf("Hello word\n"); 6 return 0; 7 } 在Linux系统下使用gcc编译程序时只须简单的命令: $gcc hell

include .h 以及.cpp的记录

VC include 路径解析要了解vc中使用#include命令包含头文件所搜寻的路径,必须先了解vc中的几种路径:1. 系统路径 系统路径在vc中是"Tools->Options->Directories"中"Include files"指定的路径. 打开此对话框会发现默认有这几条路径:C:\Program Files\Microsoft Visual Studio\VC98\INCLUDE C:\Program Files\Microsoft Vi