调用约定小记

//callTest.h
#include <stdio.h>
#include <stdlib.h>

#define A

#ifdef A
//只声明不实现是导不出函数的
__declspec(dllexport) int __cdecl test1(int pa);
//c==test
//cpp==?test1@@YAHH@Z

//c==_test2@4
//cpp==?test2@@YGHH@Z
__declspec(dllexport) int __stdcall test2(int pa);
#else
//extern "C"只对cpp文件起作用
extern "C" 
{
	//cpp==_test3@4==由被调用者自己处理堆栈平衡,往外pop 4个字节,即把int型的pa参数出栈
	__declspec(dllexport) int __stdcall test3(int pa);
	//cpp==test4
	__declspec(dllexport) int __cdecl test4(int pa);
}
#endif // 1

//得到结论1
//源文件是以.c结尾的话就使用c编译器编译
//源文件是以.cpp结尾的话就使用cpp编译器编译(可以使用exter "C"指定cpp中的某部分使用c编译器去编译生成)

//得到结论2
//用c编译器和__cdecl是不会改变函数名,但需要调用者自己清理堆栈
//用c编译器和__stdcall会改变函数名,在函数名前加下划线在函数名后加@并加上出栈字节数

//得到结论3
//cpp编译器会改名,不管是__cdecl还是__stdcall,导出的函数名都面目全非,可以认为就是乱码了,正常是无法根据函数名调用
//callTest.c   callTest.cpp
#include "../Header/callTest.h"

int __cdecl test1(int pa)
{
	return 0;
}

int __stdcall test2(int pa)
{
	return 0;
}

int __stdcall test3(int pa)
{
	return 0;
}

int __cdecl test4(int pa)
{
	return 0;
}

//得到结论1
//源文件是以.c结尾的话就使用c编译器编译
//源文件是以.cpp结尾的话就使用cpp编译器编译(可以使用exter "C"指定cpp中的某部分使用c编译器去编译生成)

//得到结论2
//用c编译器和__cdecl是不会改变函数名,但需要调用者自己清理堆栈
//用c编译器和__stdcall会改变函数名,在函数名前加下划线在函数名后加@并加上出栈字节数

//得到结论3
//cpp编译器会改名,不管是__cdecl还是__stdcall,导出的函数名都面目全非,可以认为就是乱码了,正常是无法根据函数名调用

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值