头文件中只有声明,没有定义

2023-05-16

前言:

头文件中只有声明,而没有定义。这是为什么呢?刚看到这个问题我也比较纳闷。因为我学C++之前一直是这样的,直到学习了C++中的内联函数,内联函数的声明和定义分别在不同的源文件中,出现了链接错误。这个时候又接触到了这个问题:头文件中只有声明而没有定义。在本篇博客中,我会分析这方面的内容,同时也会讲到内联函数。 


目录

前言:

头文件声明和定义

《高质量C/C++编程指南》说明

内联函数

内联函数练习题


头文件声明和定义

虽然我们常说定义不能放在头文件中,但是也有例外:

  1. 头文件中可以定义普通变量或函数,但是前提条件是只有一个.c或者.cpp文件包含了这个头文件,否则会反生链接错误;

  2. 头文件中可以定义const或者static修饰的变量或者函数;

  3. 类的定义放在头文件中;

  4. inline函数定义在头文件中。

简单测试用例:在一个工程中创建三个文件:

test.h

#ifndef __TEST_H__
#define __TEST_H__

#include<iostream>
#include<stdio.h>

int x;

#endif//__TEST_H__

test.cpp

#include"test.h"
int main()
{
	x = 9;
	std::cout << x << std::endl;
	system("pause");
	return 0;
}

test2.cpp

#include"test.h"

 运行程序,按照原来的思路,应该程序输出9。运行程序我们会发现以下的链接错误:

此处两个.cpp文件都包含了.h文件,也就是说x变量会在这两个.cpp文件中存在一个副本。在链接阶段,链接器就会发现存在多个相同变量名的全局变量,导致链接出错。所以.h文件中一般只能包含全局变量的声明,函数声明,宏定义一类的,在.h文件中定义变量是不被推荐的。

但是如果只有test.cpp文件包含了头文件test.h,而test2.cpp没有包含test.h文件,此时的程序是正常的。函数跟变量是一样的效果,此时就是我上面所说的第一种例外情况。

那么如何在头文件中声明一个变量呢?我们可以使用extern关键字

extern int x;//声明一个变量,并没有分配实际地址。记住不可以赋值,如果赋值了就成了定义了

此时我们可以在任何引用此文件的地方进行赋值,例如可以在主文件中进行赋值。

test.h

#ifndef __TEST_H__
#define __TEST_H__

#include<iostream>
#include<stdio.h>

extern int x;
extern int func(int y);

#endif//__TEST_H__

test2.cpp

#include"test.h"
int func(int y)
{
	return ++y;
}

test.cpp

#include"test.h"
int main()
{
	int x;
	x = 9;
	std::cout << x << std::endl;
	std::cout << func(x) << std::endl;
	system("pause");
	return 0;
}

此时运行程序会输出9和10。结果也是我们预期的。在test.h文件中变量(函数)声明前都加了extern变量,表示此处声明的这个变量(函数)在本项目的其他文件中定义了,此处仅是声明一下,表示其他文件只要包含了我这个test.h文件,就可以使用这个变量(函数)。


《高质量C/C++编程指南》说明

在《高质量C/C++编程指南》一书中,对此也有说明:

【建议1-2-1】头文件中只存放“声明”,而不存放“定义”。

在C++语法中,类的成员函数可以在声明的同时被定义,并且自动成为内联函数。这虽然会带来书写上的方便,但却造成了风格不一致,弊大于利。建议将成员函数的定义分开,不论该函数体有多么小。

【建定1-2-2】不提倡使用全局变量,尽量不要在头文件中出现现象 extern int value 这类声明。

书中还给出了C++/C头文件的结构:

#ifndef GRAPHICS_H         //防止graphic.h被重复引用
#define GRAPHICS_H

#include<math.h>           //引用标准库的头文件(编译器将从标准库目录中开始搜索)
...
#include"myheader.h"       //引用非标准库的头文件(编译器将从用户的工作目录开始搜索)
...
void Function(..);         //全局函数声明

class Box                  //类结构体声明
{
...
};

#endif

在写这篇博客的时候,在网上看到一篇很好的博客,在此也把链接展示给各位道友:

http://blog.sina.com.cn/s/blog_6af956630100voy9.html


 

内联函数

内联函数是C++的范畴,以inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开,没有函数压栈的开销,
内联函数提升程序运行的效率。

内联也有三种情况:

1.在类中加inline声明定义且在类内定义,编译通过,其实在类中定义的函数都是默认成内联的。

2.在类中加inline声明,在类外的同一头文件中定义(加inline或者不加inline均可),编译通过。

3.在类中声明,在类.cpp文件中定义(不管加不加inline),编译不通过。因为inline被展开,就没有函数地址了,链接就会找不到。

此处我只给出第三种情况的测试用例:

test.h

#ifndef __TEST_H__
#define __TEST_H__

#include<iostream>
#include<stdio.h>
using namespace std;
class AA {
public:
	AA(int y = 0)
		:x(y)
	{
		cout << "AA" << endl;
	}
	~AA()
	{
		cout << "~AA" << endl;
	}
	inline int add(int y);
private:
	int x;
};

#endif//__TEST_H__

test2.cpp

#include"test.h"
inline int AA::add(int y)
{
	return x+y;
}

test.cpp

#include"test.h"

int main()
{
	AA a(10);
	cout<<a.add(10)<<endl;
	
	system("pause");
	return 0;
}

内联函数练习题

关于c++的inline关键字,以下说法正确的是()

  • A.使用inline关键字的函数会被编译器在调用处展开
  • B.头文件中可以包含inline函数的声明
  • C.可以在同一个项目的不同源文件内定义函数名相同但实现不同的inline函数
  • D.定义在Class声明内的成员函数默认是inline函数
  • E.优先使用Class声明内定义的inline函数
  • F.优先使用Class实现的内inline函数的实现
  • A 项错误,因为使用 inline 关键字的函数只是用户希望它成为内联函数,但编译器有权忽略这个请求,比如:若此函数体太大,则不会把它作为内联函数展开的。
  • B 项错误,头文件中不仅要包含 inline 函数的声明,而且必须包含定义,且在定义时必须加上 inline 。【关键字 inline 必须与函数定义体放在一起才能使函数成为内联,仅将 inline 放在函数声明前面不起任何作用】
  • C 项错误, inline 函数可以定义在源文件中,但多个源文件中的同名 inline 函数的实现必须相同。一般把 inline 函数的定义放在头文件中更加合适。
  • D 项正确,类内的成员函数,默认都是 inline 的。【定义在类声明之中的成员函数将自动地成为内联函数】
  • EF 项无意思,不管是 class 声明中定义的 inline 函数,还是 class 实现中定义的 inline 函数,不存在优先不优先的问题,因为 class 的成员函数都是 inline 的,加了关键字 inline 也没什么特殊的。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

头文件中只有声明,没有定义 的相关文章

随机推荐

  • ROS下使用串口发送数据

    ROS下使用串口发送数据 span class token macro property span class token directive keyword include span span class token string lt
  • 新手如何使用postman(新手使用,简单明了)

    如何使用postman 一 了解postman 1 什么是postman xff1f 软件测试用来做接口测试的工具 2 如何下载postman https www getpostman com xff08 官方下载 xff09 链接 xff
  • 字符串的截取、分割,截取指定字符前面(后面)所有字符

    关于字符串截取问题 xff0c 从网上搜到总结一下 xff1a 已知一个字符串 xff0c 截取第一个指定字符后面所有字符 首先得知道indexof 34 34 的用法 xff0c 例如String i 61 abcdefg xff0c 那
  • [资料分享] 好赢60A无刷电调设置说明书【详细】

    完全针对车模而设计的全新程序算法 xff0c 具有优异的启动效果 加速性能 刹车性能及线性度 xff1b 支持所有无感 xff08 即无霍尔传感器 xff09 无刷电机 xff1b 高品质用料 xff0c 具有强大的耐电流能力 xff1b
  • 单片机学习笔记 —— 串口通信原理

    一 串口通信电路 电路图 xff1a 说明 xff1a 当RXD TXD为低电平时 xff0c 对应的led灯会亮起 二 串口通信控制寄存器 下图为80C51串行口的结构 xff1a SCON serial Control Register
  • 四种方法计算字符串的长度

    在这里我提供四种方法计算字符串的长度 1 使用递归函数 2 数数 xff0c 从第一个字符开始数数 xff0c 没遇到一个字符 xff0c 长度加一 xff0c 直到遇到 34 0 34 停止数数 3 使用strlen函数 xff0c 使用
  • 汉诺塔问题—C语言实现

    一 题目描述 相传在古印度圣庙中 xff0c 有一种被称为汉诺塔 Hanoi 的游戏 该游戏是在一块铜板装置上 xff0c 有三根杆 编号A B C xff0c 在A杆自下而上 由大到小按顺序放置64个金盘 如下图 游戏的目标 把A杆上的金
  • linux三大剑客

    awk是一种很棒的语言 xff0c 适合文本处理和报表生成 使用方法 awk pattern 43 action filenames 尽管操作可能会很复杂 xff0c 但是语法总是这样 xff0c 其中pattern表示AWK再数据中查找的
  • 数据结构与算法之栈

    目录 顺序栈 xff1a 链式栈 xff1a 栈的使用 xff1a 首先 xff1a 栈是一个特殊的线性表 xff0c 只允许在一端进行插入 xff08 压栈 xff09 和删除元素 xff08 进栈 xff09 xff0c 这一端称为栈顶
  • 二叉树的典型习题总结

    二叉树的三种遍历方式 xff1a 1 给定一个二叉树 xff0c 返回它的前序遍历 root left right 递归实现 xff1a public List lt Integer gt preorderTraversal TreeNod
  • javascript简介及基本语法

    这两天了解到一门新的脚本语言 javascript xff0c 貌似能干的事情好多呀哈哈哈哈 xff0c 言归正传啧 目录 javascript简介 js的简介 js的特点 javascript和java的区别 javascript的组成
  • Postman~做接口测试

    在工作中 xff0c 接口测试势必是最有效的测试途径 因此 xff0c 学习接口测试的基础和工具是很有必要的 xff0c 从Postman开始吧 xff5e 目录 1 接口测试简介 2 接口测试流程及用例设计 3 使用Postman执行接口
  • pytest接口测试自动化框架

    目录 pytest简介及安装 pytest的使用规则 pytest运行方式 主函数方式 命令行方式 跳过 标记及预期失败特殊场景处理 pytest前后置 夹具 pytest高级用法fixture pytest接口断言 pytest结合all
  • 测试的阶段性小小总结

    转眼入职2年之余 xff0c 毕业后就投入测试行业 在日常的工作中也有自己的一些思考和总结 2021到2023是多变的两年 xff0c 加入教培行业 xff0c 受双减政策影响 xff0c 注定艰难 参与了各种类型的测试项目 xff0c 不
  • 关于Charles抓包

    目录 抓包的原理 抓包的步骤 1 下载Charles 2 PC抓HTTPS协议的包 3 移动端抓包步骤 证书的原理 抓包的原理 抓包的软件非常多 xff0c 其实底层逻辑充当了一个中间人代理的角色来对HTTPS进行抓包 xff0c 结合日常
  • Ubuntu下使用CMakeLists.txt管理C/C++代码

    Ubuntu下使用CMakeLists txt管理C C 43 43 代码 一 CMakeLists txt入门知识1 CMakeLists txt的编译方法2 CMakeLists txt的文件内容3 编译的C文件 二 CMakeList
  • C语言中的__FILE__、__LINE__和__func__等预定义宏和注意事项

    C语言预处理要求定义某些对象宏 xff0c 运用这些预定义宏能使调试变得更简单 xff0c 每个预定义宏的名称一两个下划线字符开头和结尾 xff0c 这些预定义宏不能被取消定义 xff08 undef xff09 或由编程人员重新定义 常用
  • 树莓派4B ubuntu20.04 安装ROS noetic和opencv记录

    文章目录 准备换源安装ROS一些遇到的坑连接wifi添加秘钥时出错 E could not get lock var lib apt list catkin make时出现错误 Could not find the required com
  • c语言变量和常量

    c语言的常量和变量及其类型 常量以及常量的定义 变量 xff1a 在c语言中变量就是能够改变的量 常量 xff1a 就是在定义后不能去改变的值就是常量 int a 61 10 a 61 20 此时的a就是一个变量 const int a 6
  • 头文件中只有声明,没有定义

    前言 xff1a 头文件中只有声明 xff0c 而没有定义 这是为什么呢 xff1f 刚看到这个问题我也比较纳闷 因为我学C 43 43 之前一直是这样的 xff0c 直到学习了C 43 43 中的内联函数 xff0c 内联函数的声明和定义