三、基础语法1(30小时精通C++和外挂实战)

news2025/1/23 13:49:28

三、基础语法1(30小时精通C++和外挂实战)

  • 1,开发环境的搭建
  • 2,cin和cout
  • 3、4,函数重载
  • 5,使用IDA分析exe
  • 6.1,默认参数
  • 6.2,默认参数的本质汇编
  • 7,externC1作用
  • 8,externC-C与C++混合开发
  • 10,externC,定义宏防止头文件被重复包含

1,开发环境的搭建

前面三节课必须看

在这里插入图片描述

课程一共15天,讲完C++语法就会讲外挂,破解最后两天讲解

Exe为windows的可执行文件
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

这是空项目里面什么也没有,源文件放的是我们源代码文件.cpp,右击源文件添加c++文件

Main函数,C++一启动就从main函数这个入口进入

VS很多设置比较反人类
注释快捷键

https://blog.csdn.net/Niteip/article/details/14697677
看《Ogre3d beginner Guide》时其实好多东西我没写blog,现在又想干脆把工程发一下,当个保存

一看工程,哇,500多M,好吧研究哪些不需要。
ipch的文件夹,和一个与工程同名的.sdf文件,而且ipch下面的文件和.sdf文件都很大,这些文件是Visual Studio用来保存预编译的头文件和Intellisense用的,删除这些文件对于工程的开发完全没有影响。那如果我既想使用预编译的头文件和Intellisense,又不想看到这些无聊的文件该怎么办呢?

工具->选项->文本编辑器->C/C++ ->高级->回退位置 那里,两个都设成true

如果你设置了回退位置,那么IPCH等就会到那里去,否则会在系统TEMP里,有个VC++文件夹下。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

上面的常用快捷键要牢记
当我们输入一个字母要另其自动补全,默认框内使用tab。
我们将其折叠

在这里方法和函数是一个概念
//方法=函数

我们怎么打开之前创建的程序,.sln为项目文件,可双击。
或者打开开发工具,打开等
在这里插入图片描述

后期开发外挂VS需要安装一些组件
打开工具-获取工具和功能,不要随便勾选,只勾选使用C++的桌面开发,将来开发的外挂就是跑在windows系统上的桌面程序,所以勾选此

下面一张图,只要保证右边的程序勾选即可
在这里插入图片描述

当打开旧版的程序需要重新生成解决方案

在这里插入图片描述

我们新建出来的C++文件希望以utf-8编码(中文编码)形式存储文件内容
但VS默认编码不是utf-8,如果用VS开发的代码,直接使用别的工具打开的话可能发现中文是乱码的
中文编码UTF-8,GB2312,GB18030

如何更改编码格式,工具扩展和更新,联机,forceutf8(with FOM),安装完后新建的C++文件默认就是utf8编码了

2,cin和cout

入口就是程序一启动会执行的函数,main函数入口,其他编程语言是必须先有类再有函数,如java语言必须先有类,再有方法(函数),而C++直接就能定义函数

C++在此的基础上有了面向对象的东西(C with classes)
在这里插入图片描述
在这里插入图片描述

#include <iostream>		//C++头文件不用写.h
using namespace std;		
//上面两句暂时认为固定写法,写完后,就能使用cout了

int main(){
	cout << "hello world!" << endl;
	
	cout << "hello";		//输出hello
	//cout << "\n"; 		//此处为换行
	cout << endl;			//endline结束一行,也可以认为是换行
	cout << "world";		//输出world

//面向对象、运算符(操作符)重载
	//上面三句等价于下面一句
	cout << "hello" << endl << "world";


	cout << "please type a number:" << endl;
	int age;
	cin >> age;//cin从键盘接收东西,并赋值给age这个变量(我们输入一个数字敲回车,就将数字赋值给age)

	cout << "age is" << age << endl;

getchar();//如果一个getchar,这个getchar也会识别前面输入数字的回车,此函数执行完成后执行下一步,页面会一闪而过此时需要两个getchar
	getchar();//此函数是等待键盘输入的意思(如果敲回车,就会读取键盘输入),键盘不输入任何东西就一直等待,就不会退出
	return 0;
}

下面这个程序执行,首先打印,打印后就return 0 结束了,所有窗口一闪而过程序退出了,
注意:只要main函数退出就意味着程序结束
那怎么证明刚刚打印存在呢?我们右击项目,在文件资源管理器中打开资文件,打开debug文件,里面有个exe文件,我们下面写的代码就转换成了exe可执行文件

int main(){
	cout << "hello world!" << endl;

	return 0;
}

3、4,函数重载

两个函数的函数名完全一样,但是后面的参数的个数,参数的类型或者参数的顺序不同称为参数重载

c语言不支持函数重载
C++支持函数重载(很多编程语言都支持如java)

在这里插入图片描述

当编译时,尽管源文件中有多个C++文件,一些C++文件中午main函数,当编译时,在这些文件中找那个文件中有main函数,然后执行,多个文件都编译,找他们中的main函数再执行

C++为什么存在重载,编译器会将三个函数名改编,最终形成新的函数名(新函数名包含参数的信息名),这些函数就能同时存在,可以调用这些函数;C语言没有这种技术namemangling,不支持重载,若相同编译后的名字也相同,会冲突

//display_int(编译后的不同名字)
void display(int a){
	cout << "display(int)" << a << endl;
}
//display_long 
void display(long a){
	cout << "display(long )" << a << endl;
}
//display_double 
void display(double a){
	cout << "display(double)" << a << endl;
}

不同的编译器有不同的编译规则(编译后的命名也不相同)

那么我们怎么证明编译后的函数名是不一样的呢?
反汇编
如何做,先按F9打断点,此时就不能通过ctr+F5(直接执行程序,不能调试,就算打了断点也不理会直接执行),按F5此为调试,一旦进入断点模式,就可以右击转到反汇编(只有在调试状态才能转到反汇编,快捷键ALT+G)

在调试模式下

 57: 	dispaly();		//此为C++代码
001A5ADE E8 D2 B9 FF FF       call        dispaly (01A14B5h)  //加粗为汇编代码
//此句代码在内存中只占5个字节

    58: 	display(10);			//此为C++代码,转完汇编后是下面三句加深的代码
001A5AE3 6A 0A                push        0Ah  
001A5AE5 E8 DA B9 FF FF       call        display (01A14C4h)  	//调用
001A5AEA 83 C4 04             add         esp,4  
    59: 	display(10L);//long类型
001A5AED 6A 0A                push        0Ah  
001A5AEF E8 D5 B9 FF FF       call        display (01A14C9h)  
001A5AF4 83 C4 04             add         esp,4  
    60: 	display(10.5);//double类型
001A5AF7 83 EC 08             sub         esp,8  
    60: 	display(10.5);//double类型
001A5AFA F2 0F 10 05 58 CD 1A 00 movsd       xmm0,mmword ptr ds:[1ACD58h]  
001A5B02 F2 0F 11 04 24       movsd       mmword ptr [esp],xmm0  
001A5B07 E8 B3 B9 FF FF       call        display (01A14BFh)  
001A5B0C 83 C4 08             add         esp,8  
    61: 
    62: 	getchar();
001A5B0F 8B F4                mov         esi,esp  
001A5B11 FF 15 D0 01 1B 00    call        dword ptr ds:[1B01D0h]  
001A5B17 3B F4                cmp         esi,esp  
001A5B19 E8 16 B8 FF FF       call        __RTC_CheckEsp (01A1334h)  
    63: 	return 0;
001A5B1E 33 C0                xor         eax,eax  

那么左侧第二列的是什么,为机器码(采用16进制形式显示0和1)

1010 1010 1111 0001 1010 1011  ---转成16进制为-- AA F1 AB	(此处占3个字节)
两个16进制位代表8个二进制位,两个16进制也就是1个字节
001A5ADE E8 D2 B9 FF FF       call        dispaly (01A14B5h)		//此句代码在内存中只占5个字节,当然想要读懂意思看右边汇编
    58: 	display(10);			//此句代码在内存中只占10个字节
001A5AE3 6A 0A                push        0Ah  
001A5AE5 E8 DA B9 FF FF       call        display (01A14C4h)  	//调用
001A5AEA 83 C4 04             add         esp,4 

机器码与汇编是1对1的关系,有了机器码就能得到汇编,反之亦然
我们平时看机器码比较困难,就看右边的汇编

最左侧的是001A5B1E是内存地址,程序一启动,代码就放在内存中,每一句代码就有字节内存地址,要执行某一句代码就要找到代码字节地址从这个字节地址开始执行。每一句机器码都有自己的内存地址

平时写的代码按顺序向下执行的,机器码也是如此,机器指令的内存地址是连续的,机器码是连续存储的一个字节都不差,挨的紧紧密密,不会留空隙,空内存

我们平时编写代码如java之类在内存中都是机器码E8 D2 B9 FF FF 的二进制,右侧的汇编我们能知道其做了什么

dispaly (01A14B5h)  
display (01A14C4h)
display (01A14C9h)  

好像上面调用的三个都是display,我们的结论没有错,真实是编译后的每个函数名不同,只是VS这个开发工具为了方便我们开发者看东西与源代码对的上,展示的是display,但还是有所不同(01A14C9h) 不同,此为函数内存地址不一样,函数内存地址不一样最终执行代码也是不一样的

我们可以点击调试中的逐语句F11,可以一条一条执行汇编语言
从函数地址我们可以看到他们确实是不同函数,但是我们还是无法看到它将我们函数名改掉了,怎么办,借助工具IDA PRO 可以进行一些逆向,对我们最终生成的可执行文件进行逆向操作

我们现在ctr+F5启动此程序会生成一个可执行文件exe,在debug中,我们刚刚编写的这些C++代码最终变成0和1都放在debug中的exe中了,但是当我们双击exe时,exe又会将0和1载进执行代码

刚刚我们按F5时能看到机器码,机器码是存放在exe文件内的,只是双击时,会将机器码载进内存

void dispaly(){
	cout << "display()" << endl;
}
//display_int
void display(int a){
	cout << "display(int)" << a << endl;
}
//display_long 
void display(long a){
	cout << "display(long )" << a << endl;
}
//display_double 
void display(double a){
	cout << "display(double)" << a << endl;
}

这些函数都在exe文件中,我们能否窥探一下这个exe,看看这些函数的真实名字,使用IDA

5,使用IDA分析exe

我们现在ctr+F5启动此程序会生成一个可执行文件exe,在debug中,我们刚刚编写的这些C++代码最终变成0和1都放在debug中的exe中了,但是当我们双击exe时,exe又会将0和1载进执行代码

刚刚我们按F5时能看到机器码,机器码是存放在exe文件内的,只是双击时,会将机器码载进内存

void dispaly(){
	cout << "display()" << endl;
}
//display_int
void display(int a){
	cout << "display(int)" << a << endl;
}
//display_long 
void display(long a){
	cout << "display(long )" << a << endl;
}
//display_double 
void display(double a){
	cout << "display(double)" << a << endl;
}

这些函数都在exe文件中,我们能否窥探一下这个exe,看看这些函数的真实名字,使用IDA

IDA的头像是个女的,据说是世界上第一个女程序员,我们点击new,点击取消,将刚刚的exe拖进去,它能识别出这是个PE文件(potable excutable),像我们windows的可执行文件都是PE格式的文件,此时我们点击OK

我们发现左侧有个function window,在此会展示出我们所有函数的名字,项目中所有函数名称都会展示出来,但是并没有找到想要的display函数开头的,有很多sub这些都是debug模式下生成的东西

我们的项目中有两个模式debug和release。
如果在debug模式下启动此程序会有很多调试信息,生成的可执行文件比较臃肿
而release模式运行此程序会去除调试信息,生成的可执行文件比较精简高效,速度比较快

刚刚我们是用debug模式生成的exe,有很多调试信息就看不到我们display,我们选择release模式下再运行一下,一般我们要发布我们的项目,我们肯定选择release模式,我们的项目想要上线不能太多,要高效,而且项目发布根本不需要调试信息和断点信息

我们在VS选择release模式运行一下,然后选择项目右击打开文件夹,打开release下的exe

在IDA会在debug下生成ID0\ID1等文件,所以点击IDAfile点击CLOSE时选择不保存(don’t pack database)、回收垃圾(collect)、不保存数据库(don’t save),最后OK,此时debug中由IDA生成的文件就清楚了

我们将release文件夹下的exe拖入,点击OK,此时有些不同了,最起码能看到main了,如果是release模式的话,我们的main函数进行了优化,在右面的框中,看到直接打印display(),display(long)等等,在此模式下优化保证代码高效如果调用display发现display无非就是打印,默认调用display函数,会给display开辟一个栈空间给函数,函数执行完里面这个代码后,再回收这个栈空间,这四个函数都有开辟四次栈空间回收四次栈空间,编译器会觉的重复,就直接main函数执行下面四句函数,不再调用

int main(){
//dispaly();
	//display(10);
	//display(10L);//long类型
	//display(10.5);//double类型
上面四句全部替换为下面四句,不再调用
cout << "display()" << endl;
cout << "display(int)" << a << endl;
cout << "display(long )" << a << endl;
cout << "display(double)" << a << endl;

	getchar();
	return 0;
}

最终执行效果是一样的,上面直接执行四句的效率最高,这就是优化,所以在左边看不到display,被优化掉了,在release模式下,打断点,调试,右击反汇编,可以看到是有优化的,没有调用,直接打印如下

  57: 	dispaly();
00C812A6 68 30 1A C8 00       push        0C81A30h  
00C812AB 51                   push        ecx  
00C812AC 8B 0D 5C 30 C8 00    mov         ecx,dword ptr ds:[0C8305Ch]  
00C812B2 BA BC 31 C8 00       mov         edx,0C831BCh  
00C812B7 E8 34 05 00 00       call        std::operator<<<std::char_traits<char> > (0C817F0h)  
00C812BC 83 C4 04             add         esp,4  
00C812BF 8B C8                mov         ecx,eax  
00C812C1 FF 15 40 30 C8 00    call        dword ptr ds:[0C83040h]  
    58: 	display(10);
00C812C7 8B 0D 5C 30 C8 00    mov         ecx,dword ptr ds:[0C8305Ch]  
00C812CD BA C8 31 C8 00       mov         edx,0C831C8h  
00C812D2 68 30 1A C8 00       push        0C81A30h  
00C812D7 6A 0A                push        0Ah  
00C812D9 E8 12 05 00 00       call        std::operator<<<std::char_traits<char> > (0C817F0h)  
00C812DE 8B C8                mov         ecx,eax  
00C812E0 FF 15 64 30 C8 00    call        dword ptr ds:[0C83064h]  
00C812E6 8B C8                mov         ecx,eax  
00C812E8 FF 15 40 30 C8 00    call        dword ptr ds:[0C83040h]  

如果是debug模式

 57: 	dispaly();		//此为C++代码
001A5ADE E8 D2 B9 FF FF       call        dispaly (01A14B5h)  //调用display

    58: 	display(10);			//此为C++代码,转完汇编后是下面三句加深的代码
001A5AE3 6A 0A                push        0Ah  					//传参10
001A5AE5 E8 DA B9 FF FF       call        display (01A14C4h)  	//调用
001A5AEA 83 C4 04             add         esp,4  

所以release模式是编译器帮忙优化代码,它觉得能优化就优化,能提速就提速,但还是没有证明刚刚四个函数最终函数名不一样,此时有个坑,但debug模式看不到,左边会生成一大堆debug模式下的函数名,找不到我们的函数,但是release确实左边少掉了很多东西,看到main函数,有些优化掉了,怎么办

我们可以选择release模式,禁止优化即可,右击项目选择属性,c++有个优化,禁用掉。此时在release模式下再F5开始执行不调试,此时右击反汇编(Alt+g),可以看到有call不是直接打印了

此时将生成的release的exe没有优化过的,也就是存在上面四个函数的,看看函数的名字,此时IDA打开的是这个文件,路径一样需要重新加载,不用关掉,点击file下的路径0,此时看main函数,点击左方main,右侧代码有calldisplay

不同的编译器有不同的行为,如果是G++编译器编译生成的函数名会是令一种名称
最终生成什么函数名不影响我们开发,函数名无论多么古怪,都不用我们管,最终都会调用,而且最终汇编是那个名字,还是要根据内存地址去找,什么名字不重要

我们通过这些看到了本质,在网上搜的只有结论,我们不知道对不对,也不知道怎么验证,但现在可以验证了,通过汇编看本质

6.1,默认参数

在这里插入图片描述
在这里插入图片描述

今天的最后一个知识点,默认参数

#include <iostream>
using namespace std;

//下面定义括号中的是有默认参数的
int sum(int v1=5, int v2=6){
	return v1 + v2;
}

int main(){

	cout << sum() << endl;//11		这个不传参也正确,里面的默认参数相加
	cout << sum(10) << endl;//16	这个传了第一个参数10将第一个默认参数5覆盖掉,与第二个默认参数6相加,结果16
	cout << sum(10, 20) << endl;//30

	getchar();
	return 0;
}

注意默认参数必须从右到左,下面的代码是错误的

int sum(int v1=5, int v2){
	return v1 + v2;
}

int main(){
	cout << sum() << endl;//此处无论上面还是int sum(int v1, int v2=6)都报错,必须传递一个参数覆盖第一个值	
cout << sum(10) << endl;//上面第二个默认值没有,此时传的参数10只覆盖了第一个,第二个必须传参才行此时报错,可将其改为int sum(int v1, int v2=6)
	getchar();
	return 0;
}

如果函数同时有声明、实现,默认参数只能放在函数声明中(如果又有声明又有实现只能在声明中方默认参数)
很多时候main函数写在最前面,而其他函数写main函数在后面,但有些编译器函数放在main函数后面会不认识main函数中的函数的,从上往下走,这时需要将其进行提前的声明

#include <iostream>
using namespace std;

int sum(int v1 = 5, int v2 = 6);		//此处为函数声明
int main(){

	cout << sum() << endl;//11
	cout << sum(10) << endl;//16
	cout << sum(10, 20) << endl;//30

	getchar();
	return 0;
}

int sum(int v1, int v2){		//此处为函数实现
	return v1 + v2;
}

默认参数的值可以是常量、全局符号(全局变量、函数名)
全局变量,写在函数外面的为全局变量

int age = 20;					//全局变量age
int sum(int v1 = 5, int v2 = age);

int main(){

	int age = 20;				//局部变量

	cout << sum() << endl;//11
	cout << sum(10) << endl;//16
	cout << sum(10, 20) << endl;//30

	getchar();
	return 0;
}

int sum(int v1, int v2){
	return v1 + v2;
}

那什么是函数名呢,回顾一下知识点,指针

void test(){
	cout << "test()" << endl;
}
int main(){
	//test();//平时调用直接这样
	//我们也可以通过下面这种方式,带*的p是指针,指向了test这个函数,将函数名赋值给P说明P指向这个函数,接着我们可以直接p();调用这个test函数
	void(*p)() = test;//void代表P指向的函数无返回值,右边()代表指向的函数是没有参数的
	p();
	getchar();
	return 0;
}


void test(int a){
	cout << "test(int)" << endl;
}
int main(){
	
	void(*p)(int) = test;//void代表P指向的函数无返回值,右边(int)代表指向的函数需要传递int类型的参数的
	p(10);			//将10传递给了test
	getchar();
	return 0;
}

说明这个指针是可以存储函数名的

void func(int v1,void(*p)(int){
	p(v1);			//相当于test(v1)

}

void test(int a){
	cout << "test()" << endl;
}

int main(){
	func(20, test);		//调用func将20传给V1,再将test传给指针P,最终结果是test(v1)

	getchar();
	return 0;
}

下面的传递函数名也是可以的

void func(int v1,void(*p)(int)=test){
	p(v1);
}

void test(int a){
	cout << "test(int)" << a << endl;
}

int main(){
	func(20);		//这个函数也是对的
	func(20, test);

	getchar();
	return 0;
}

如果函数的实参经常是同一个值,可以考虑使用默认参数
默认参数有什么价值如下

Sum(1,20)
Sum(2,20)
Sum(3,20)
Sum(4,20)
Sum(5,20)

Int Sum(int V1, int V2=20){

}
Int main(){
Sum(1)
Sum(2)
Sum(3)
Sum(4,25)
Sum(5)

}

C语言是没有默认参数这一说法的

在这里插入图片描述

函数重载和默认参数产生冲突时,我们就保留默认参数的函数吧,这个功能要多一些

6.2,默认参数的本质汇编

先看一下没有默认参数的程序的汇编代码

#include <iostream>
using namespace std;

int sum(int v1, int v2){
	return v1 + v2;
}

int main(){
	sum(1, 2);		#此处打断点
	sum(3, 4);

	getchar();
	return 0;
}

打断点,调试,右击反汇编

 23: 	sum(1, 2);
0034530E 6A 02                push        2  		#push传参
00345310 6A 01                push        1  		#push传参
00345312 E8 C0 BD FF FF       call        sum (03410D7h)  	#调用sum函数
00345317 83 C4 08             add         esp,8  		#牵扯到堆栈操作,先不讲
    24: 	sum(3, 4);
0034531A 6A 04                push        4  	#push传参
0034531C 6A 03                push        3  	#push传参
0034531E E8 B4 BD FF FF       call        sum (03410D7h)  	#调用sum函数,地址相同
00345323 83 C4 08             add         esp,8  

上面两个sum调用的地址相同,但传的参数不同
    18: 	return v1 + v2;
00B8518E 8B 45 08             mov         eax,dword ptr [v1]  
00B85191 03 45 0C             add         eax,dword ptr [v2]  	#add  V1与V2相加

下面是有默认参数的

int sum(int v1, int v2=4){
	return v1 + v2;
}

int main(){
	sum(1);
	getchar();
	return 0;
}

    23: 	sum(1);
0016530E 6A 04                push        4  
00165310 6A 01                push        1  
00165312 E8 C0 BD FF FF       call        sum (01610D7h)  
00165317 83 C4 08             add         esp,8  

发现是两个push,就算是有默认参数的情况,本质还是传递两个参数,其中是默认值,不存在调用函数在函数中将4赋值给V2

下面sum(1)和sum(1,4)的汇编代码是完全一样的,生成的机器码也是一样的(此处存疑)

    23: 	sum(1);
00A7530E 6A 04                push        4  
00A75310 6A 01                push        1  
00A75312 E8 C0 BD FF FF       call        sum (0A710D7h)  
00A75317 83 C4 08             add         esp,8  
    24: 	sum(1, 4);
00A7531A 6A 04                push        4  
00A7531C 6A 01                push        1  
00A7531E E8 B4 BD FF FF       call        sum (0A710D7h)  
00A75323 83 C4 08             add         esp,8  

首先内存地址是不一样的,但机器码是一样的(此处存疑),右击去掉地址显示

    23: 	sum(1);
6A 04                push        4  
6A 01                push        1  
E8 C0 BD FF FF       call        sum (0A710D7h)  
83 C4 08             add         esp,8  
   24: 	sum(1, 4);
6A 04                push        4  
6A 01                push        1  
E8 B4 BD FF FF       call        sum (0A710D7h)  
83 C4 08             add         esp,8  

将上面的赋值到一个软件(beyond compare),进行比较,红色的是不一样的,打开beyond compare,点击文本比较
我们发现只有一点不同,这里是E8的计算方法不同

E8 C0 BD FF FF       call        sum (0A710D7h) 
E8 B4 BD FF FF       call        sum (0A710D7h)

函数调用基本上是E8,既然我们调用的sum函数,sum函数地址是一样的,那为什么机器码有一点点不一样,E8右边的机器码是算出来的,与sum函数地址和下方的内存地址是有关联的

后面会详细讲汇编会有这个白皮书
Intel 机器码的白皮书PDF(天书)

在这里插入图片描述

Sum(1)确实等价于sum(1,4),只是因为call sum 处的位置不一样,所以生成的机器码有一点点的不同

不要纠结这个E8计算,知道这是个调用就行了

Push 4 是到栈里面(函数栈空间)
我们这课程时挖掘编程本质,利用C++和汇编挖掘编程本质

Hopper是Mac平台的
我们破解的是release优化后的exe文件

7,externC1作用

在这里插入图片描述

被extern “C”修饰的代码会按照C语言的方式去编译

#include <iostream>
using namespace std;

void func(){

}

void func(int v){

}

int main(){

	getchar();
	return 0;
}

在C语言中是没有函数重载的
这两个重载函数是写在CPP文件内的,编译器以看到有在这个文件,就会按照C++的标准去编译
,我们将extern “C”加到函数前

#include <iostream>
using namespace std;

extern  "C" void func(){

}

extern "C" void func(int v){

}

int main(){

	getchar();
	return 0;
}

我们运行此程序结果报错,不允许重载函数
我们是按照C语言的方式去编译被extern “C” 修饰的函数的

如果函数同时有声明和实现,要让函数声明被extern “C” 修饰,函数实现可以不修饰

extern "C" void func();
extern "C" void func(int v);

int main(){

	getchar();
	return 0;
}

void func(){

}

void func(int v){

}

此处报错,不能重载,证明extern “C” 放在声明前十有用的

 void func();
extern "C" void func(int v);

int main(){
	func();
	func(10);
	getchar();
	return 0;
}

void func(){
	cout << "func()" << endl;

}

void func(int v){
	cout << "func(int v)" << endl;
}

当代码变为上方时就不报错了,为什么,因为 void func();由C++编译器编译,extern “C” void func(int v);由C语言编译器编译,这两个函数不同时存在于C语言的编译,即两者编译后的函数名是不同的,所以不报错
如果两个函数都有extern “C” 意味着都通过C语言编译,C语言是不支持重载的。

下面两句去掉参数,是二意性的问题,有歧义

 void func();
extern "C" void func();

我们还可以这样使用

 extern "C"{
	 void func();
	 void func(int v);
 } 

8,externC-C与C++混合开发

extern "C"一般用在什么地方,用在C,C++混合开发
由于C,C++编译规则的不同,在C,C++混合开发时,可能经常出现以下操作

我们在开发时可能会经常用到第三方框架或者第三方的库(可能是用C语言写的开源库)

我们新建一个C文件math.c文件,里面函数如下

int sum(int v1, int v2){
	return v1 + v2;
}

int delta(int v1, int v2){
	return v1 - v2;
}

而main.cpp文件中为

#include <iostream>
using namespace std;

int sum(int v1, int v2);
int delta(int v1, int v2);

int main(){
	/*func();
	func(10);*/
	cout << sum(10,20) << endl;
	cout << delta(10,20) << endl;
	getchar();
	return 0;
}

此时在C++文件有对C文件中的函数的声明,但是运行后出现调用错误,这是因为两个文件的编译器不同,在C++文件中声明的函数编译后的名字与C文件中实现函数编译后的名字不同。所以在math.c找不到声明的函数,就调用报错了。

我们应告诉C++文件中sum和delta函数是按照C语言方式编译的,此时我们在之前加入extern "C"即可,声明在main.cpp文件中,实现在math.c文件中,我们只需在声明前加入extern "C"即可,如下

extern  "C"int sum(int v1, int v2);
extern  "C"int delta(int v1, int v2);

或者

extern  "C" {
	int sum(int v1, int v2);
	int delta(int v1, int v2);
}

所以C++就知道了这两个函数是C语言编译的,此时,就执行成功了

现实并不像上面只有两个文件简单
第三方框架或者第三方库要比上面的只有两个文件要复杂
将来库中的函数可能会很多,如果每个函数都要在文件前方声明的话会很复杂,别人使用你写的库会很麻烦,还需要自己声明。
我们怎么办,我们将声明放到头文件中,在头文件夹右击新建一个头文件math.h

.c文件使用来方函数实现的,.h文件为头文件一般用来存放函数声明的
如下math.h

//header file
int sum(int v1, int v2);
int delta(int v1, int v2);
int divide(int v1, int v2);

而main.cpp文件下我们改为如下

#include <iostream>
using namespace std;

extern  "C" {
	#include "math.h"				//主要改此地方即可
}

int main(){
	/*func();
	func(10);*/
	cout << sum(10,20) << endl;
	cout << delta(10,20) << endl;
	cout << divide(30,3) << endl;

	getchar();
	return 0;
}
#include "math.h"			//此句代码很简单,就相当于将math.h文件中的所有内容拷贝到main.cpp文件中,也就是将其中的所有声明拷贝到此文件下

如果我们开发第三方库、第三方框架给别人用肯定有一个头文件。
我们在使用别人的框架时,它可能给我们一个头文件、一个.C文件,因为有些东西我们只能用C语言这种比较底层的语言实现,效率会高一些。我们使用别人的库也很简单只需要包含一下别人的头文件即可,但是别人用C语言编写的,我们就要在包含的头文件先加一个extern “C”,这样就能识别出来C语言写的,我们调用就没问题了

我们在main.cpp文件下将头文件前加extern “C”,每次调用都要加extern “C”,挺麻烦的,我们直接在math.h这个头文件中将这些声明用extern "C"包起来,这样在main.cpp文件下,直接#include “math.h” 就行了如下
math.h文件下内容

//header file
extern  "C"{
	int sum(int v1, int v2);
	int delta(int v1, int v2);
	int divide(int v1, int v2);
}

Main.cpp文件下内容

#include <iostream>
using namespace std;

#include "math.h"				//此处之前无需extern  "C",头文件已经将声明按C语言编译								识别了
int main(){
	cout << sum(10,20) << endl;
	cout << delta(10,20) << endl;
	cout << divide(30,3) << endl;

	getchar();
	return 0;
}

上面程序是可以运行的。

那么如果我们直接用C文件other.c来调用这个头文件main.h行吗,不行,C语言的编译器不认识extern “C”,所以这样就会报错
Other.c

#include "math.h"			//此处就相当于Math.h中的声明内容

void other(){
	sum(10, 20);
}

Math.h内容
extern  "C"{
	int sum(int v1, int v2);
	int delta(int v1, int v2);
	int divide(int v1, int v2);
}

我们去掉extern "C"就行了,但是C++又麻烦了,又回到之前的状态了,此时我们应该怎么做呢?

我们现在希望extern “C”{}在C++环境下出现,但在C环境下消失,那应如何操作,很简单,我们要知道一个知识点宏定义这个是C语言的知识,在编译器会在C++环境下定义一个宏,宏其已经定义好了为#define __cplusplus,只要是C++环境它就会自动给你定义一个宏证明你是C++环境,只要是C++文件就默认有个宏#define __cplusplus(这个我们是看不到的,文件自动形成的,不用我们写),而C语言没有这个宏,我们可以通过判断这个宏来判断这个文件是否为C++文件

在math.h头文件中内容如下(很完全)

//header file:存放函数的声明
#ifdef __cplusplus
extern  "C"{
#endif

	int sum(int v1, int v2);
	int delta(int v1, int v2);
	int divide(int v1, int v2);

#ifdef __cplusplus
}
#endif

在这里,如定义了C++这个宏(也就是__cplusplus),在#ifdef __cplusplus 与 #endif之间的代码extern “C”{才能参与编译,下面的 }也是在这个条件中
这样,我们在C++环境和在C语言环境都可以直接使用#include “math.h” ,而无需加extern "C"之类的了
在other.c文件下代码为

#include "math.h"

#include <stdio.h>		//因为在C语言环境下,调用此头文件,调用此库可以使用下面的							printf函数,否则C++的cout是不认识的
void other(){
	printf("other - %d\n", sum(10, 20));		//打印此数值,验证此函数成功调用,\n换行
}

Main.cpp的内容如下
#include <iostream>
using namespace std;

#include "math.h"
extern "C" void other();

int main(){
	other();

	cout << sum(10,20) << endl;
	cout << delta(10,20) << endl;
	cout << divide(30,3) << endl;

	getchar();
	return 0;
}

如果是C++文件,默认在文件在最前面定义宏,相当于#define __cplusplus ,而调用#include math.h头文件,在里面从上向下执行,#ifdef __cplusplus(如果定义了这个宏__cplusplus就向下执行),执行到#endif结束,也就是执行在这两者之间的代码

在C语言环境,在此文件是没有这个宏#define __cplusplus,没有定义那么在执行头文件内的内容时,没有定义宏也就不执行#define __cplusplus与#endif之间的代码,即便将其拷贝到C语言文件下也是不会执行的

当我们开发第三方库的时候我们要加入#ifdef __cplusplus 换行 extern “C”{ 换行 #endif,以及#ifdef __cplusplus 换行 } 换行 #endif,也就是下面类似的代码

#ifdef __cplusplus
extern  "C"{
#endif
	int sum(int v1, int v2);
	int delta(int v1, int v2);
	int divide(int v1, int v2);
#ifdef __cplusplus
}
#endif

这样我们的库,C语言文件能调用,C++文件也能调用,我们会看到有些第三方库的头文件,有#ifdef和#endif

正常来说我们在自己的库math.c文件下也要添加对自己库调用的头文件math.h,因为在math.c文件中的函数可能会进行相互调用。

声明(#include “.h”)可以随便写,写多少分没关系,但实现只能有一份。

10,externC,定义宏防止头文件被重复包含

  • 我们经常使用#ifndef、#define、#endif来防止文件的内容被重复包含(任何版本都能用,而且可以针对某段代码)
  • #pragma once可以防止整个文件的内容被重复包含(版本不同可能此句不能用,针对所有文件)
    在这里插入图片描述

我们编写第三方库时,有可能包含头文件#include “math.h”写了很多份,后面想用的时候不确定前面有没有包含,就有写了一份,这样可能会导致重复包含头文件的现象,到时编译器编译的时候会比较臃肿,虽然这样写多少没关系,但预处理就是执行#include “math.h”(将头文件中的函数声明拷贝到此文件中),拷贝了很多次,为了防止重复包含,会在文件使用#ifndef#endif将所有内容包含,如下
Math.h文件中的内容

//header file:存放函数的声明
#ifndef ABC						#如果没有定义ABC这个宏
#define ABC						#就定义ABC这个宏(可以用来防止重复定义的现象)

#ifdef __cplusplus
extern  "C"{
#endif
	int sum(int v1, int v2);
	int delta(int v1, int v2);
	int divide(int v1, int v2);
#ifdef __cplusplus
}
#endif

#endif							#与ifndef 相互对应

这样写还是存在一个问题,将来会包含多个头文件,如果多个头文件下的#ifndef ABC 重复了,就会导致某些#include 的文件包含无法执行,相当于不存在。如果过防止重复包含的宏ABC一样,就会导致某些头文件不能执行
这是因为宏的命名不规范导致的,我们要是的每个头文件中的#ifndef ABC宏的命名是唯一的,有规范,这个很简单,用文件名作为宏名就行了,文件名是唯一的我们可以

#ifndef _MATH_H
#define _MATH_H
#endif

新建一个other.h文件,其中内容
#ifndef __OTHER_H
#define __OTHER_H
void other();
#endif

而math.h中的内容,定义的宏的名字以文件名前两个下划线,后面大写来实现的
#ifndef __MATH_H
#define __MATH_H

#ifdef __cplusplus
extern  "C"{
#endif
	int sum(int v1, int v2);
	int delta(int v1, int v2);
	int divide(int v1, int v2);
#ifdef __cplusplus
}
#endif

#endif

我们只需在头文件之前添加#pragma once这样就能保证此文件内容在被引用时不会重复包含
如新建一个test.h中内容如下

#pragma once
void test();

即便在main.cpp文件中写了很多

#include “test.h”
#include “test.h”
#include “test.h”

也是相当于只有一个

#include “test.h”

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1945535.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

防御和进攻编程

防御性编程是许多程序员都听说过的一个术语&#xff0c;对于某些程序&#xff0c;防御性编程是必不可少的。对于其他程序&#xff0c;它可能偶尔使用一下。除此之外&#xff0c;还有攻击性编程。 在本文中&#xff0c;我们将首先研究“正常编程”。我们首先研究它&#xff0c;…

android(安卓)最简单明了解释版本控制之MinSdkVersion、CompileSdkVersion、TargetSdkVersion

1、先明白几个概念 &#xff08;1&#xff09;平台版本&#xff08;Android SDK版本号&#xff09; 平台版本也就是我们平时说的安卓8、安卓9、安卓10 &#xff08;2&#xff09;API级别&#xff08;API Level&#xff09; Android 平台提供的框架 API 被称作“API 级别” …

Mongodb的通配符索引

学习mongodb&#xff0c;体会mongodb的每一个使用细节&#xff0c;欢迎阅读威赞的文章。这是威赞发布的第95篇mongodb技术文章&#xff0c;欢迎浏览本专栏威赞发布的其他文章。如果您认为我的文章对您有帮助或者解决您的问题&#xff0c;欢迎在文章下面点个赞&#xff0c;或者关…

Blender 4.2 安装GIS插件步骤

Blender 4 更新以后插件安装变得复杂&#xff0c;插件界面的安装按钮不显示&#xff0c;界面布局改变&#xff0c;怎么安装插件&#xff1a; 1. 在线安装&#xff1a; “编辑”&#xff08;Edit&#xff09;>进入偏好设置&#xff08;Preferences setting&#xff09;>…

文件粉碎销毁 硬盘粉碎销毁 废弃的文件如何销毁

废弃的文件可以采用多种方法进行销毁&#xff0c;具体取决于文件的敏感性和数量。以下是一些常见的废弃文件销毁方法&#xff1a; 1. 机械粉碎&#xff1a;这是一种常见的方法&#xff0c;尤其适用于含有敏感信息的文件。可以使用碎纸机将文件切碎&#xff0c;对于小批量的资料…

Python新手如何制作植物大战僵尸?这篇文章教会你!

引言 《植物大战僵尸》是一款非常受欢迎的塔防游戏&#xff0c;玩家需要种植各种植物来抵御僵尸的进攻。在这篇文章中&#xff0c;我们将使用Python编写一个简化版的植物大战僵尸游戏&#xff0c;以展示如何使用Python创建游戏。 游戏规则 玩家将种植不同类型的植物来防御僵尸…

微软“蓝屏”事件:对全球IT基础设施韧性与安全性的深刻反思

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 非常期待和您一起在这个小…

【ai】windows ollama安装qwen

可以直接github下载,或者官方网站下载:参考大神的介绍当前最新0.28安装windows的预览版本,出现一个服务? 直接跑会TLS超时? 配置代理 export https_proxy=http://192.168.50.65:7890 export http_proxy

JavaWeb笔记_Session

Session概述 Session是一种在服务端记录用户会话信息的技术 Session的创建和获取 /*** HttpServletRequest对象中的方法:* public HttpSession getSession()* 如果当前服务端没有session,那就在服务端新建一个session对象* 如果在服务端有这个session,那么就直…

【C++】类和对象之继承

目录 继承的概念和定义 继承的概念 继承的定义 继承的定义格式 继承关系和访问限定符 继承基类成员访问方式的变化 访问权限实例 基类和派生类对象赋值转换 继承中的作用域 派生类的默认成员函数 继承与友元 继承与静态成员 复杂的菱形继承及菱形虚拟继承 继承的…

别再只知道埋头苦学python了!!学了python后月入1w不在话下,不准你还不知道!!!

在Python接单的过程中&#xff0c;掌握一些技巧、注意相关事项以及选择合适的接单平台是非常重要的 一、Python接单要注意哪些 报酬问题&#xff1a;在接单前&#xff0c;务必明确客户所说的报酬是税前还是税后&#xff0c;以避免后期产生纠纷。时间管理&#xff1a;不要与客户…

nacos get changed dataId error, code: 403

nacos get changed dataId error, code: 403问题解决 问题出现原因&#xff1a;解决办法&#xff1a;需要在运行项目的配置添加权限账号和密码,重启服务 问题出现原因&#xff1a; 由于nacosserver开启了权限验证&#xff0c;项目启动时出现异常 nacos.core.auth.caching.ena…

Java基础06:变量,常量,作用域(狂神说Java)

一.变量 有了static&#xff0c;即类变量&#xff0c;就可以不用new了可以直接调用&#xff0c;类变量之后再细讲 二.常量 三.变量的命名规范

权限(linux)

权限就是文件权限&#xff08;linux万物皆文件&#xff09; 本文主要涉及文件/文件夹权限 涉及指令&#xff1a; shell&#xff1a; kernal &#xff1a; linux内核 shell &#xff1a; 外壳 shell可以方便交互与操作 bash是一个具体的shell su su 切换用户 su -root 变…

AppInventor导入导出项目以及打包apk安装包

AppInventor导入导出项目以及打包apk安装包 1.概述 当项目开发好了之后&#xff0c;如果想将项目分享给其他人&#xff0c;或者导入其他人开发的项目怎么办那。 如果给其他人安装你的项目&#xff0c;如何安装那&#xff1f; 2.自带导出和导入功能 导出项目&#xff0c;在P…

【网络】tcp_socket

tcp_socket 一、tcp_server与udp_server一样的部分二、listen接口&#xff08;监听&#xff09;三、accept接收套接字1、为什么还要多一个套接字&#xff08;明明已经有了个socket套接字文件了&#xff0c;为什么要多一个accept套接字文件&#xff1f;&#xff09;2、底层拿到新…

基于VMware(虚拟机) 创建 Ubunton 24.04

目录 1. 设置网络 1. 在安装ubuntu时设置网络 2.在配置文件中修改 2.设置 root 密码 3. 防火墙设置 1 安装防火墙 2 开启和关闭防火墙 3 开放端口和服务规则 4 关闭端口和删除服务规则 5 查看防火墙状态 4. 换源 1. 在创建的时切换源 2.修改源配置 1、Ubuntu24.04 …

MBR60200PT-ASEMI无人机专用MBR60200PT

编辑&#xff1a;ll MBR60200PT-ASEMI无人机专用MBR60200PT 型号&#xff1a;MBR60200PT 品牌&#xff1a;ASEMI 封装&#xff1a;TO-247 批号&#xff1a;最新 恢复时间&#xff1a;35ns 最大平均正向电流&#xff08;IF&#xff09;&#xff1a;60A 最大循环峰值反向…

学习华为IPD流程黑话2.0

目录 1、内容简介 2、概念六&#xff1a;管道管理 3、概念七&#xff1a;业务计划 4、概念八&#xff1a;IPMT 的投资活动 5、概念九&#xff1a;BETA、ESS、ESP 作者简介 1、内容简介 学习任何新事物都是从概念开始的。 以我个人最近遇到的一个事为例&#xff1a; 前…

TCP三次握手和四次挥手的理解

三次握手 第一次握手&#xff1a; 客户端发出 请求报文其中SYN应1&#xff0c;选择一个序列号x 第二次握手&#xff1a; 服务端接收到之后回复 确认报文&#xff0c;其中SYN应1&#xff0c;ACK1&#xff0c;确认号是x1&#xff0c;同时为自己初始化序列号y 第三次握手&…