再议:__cdecl与__stdcall 调用约定在动态链接库调用中不同的表现

2023-12-16 16:08

本文主要是介绍再议:__cdecl与__stdcall 调用约定在动态链接库调用中不同的表现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

dll中函数声明

                  dll中函数名                   

 

void __declspec(dllexport) add(int, int)

?add@@YAXHH@Z

 

extern "C" void __declspec(dllexport) add(int, int)

add

 

void __declspec(dllexport) __stdcall add(int, int)

?add@@YGXHH@Z

 

extern "C" void __declspec(dllexport) __stdcall add(int, int)

_add@8

 

主调程序必需用相同的符号调用,但c++各个编译器生成符号不同,因此为了方便,也为了与c的混合编程,一般用c++编译器编写dll时,

声明为 extern "C" ,而主调程序也同时声明为extern "C" (或 不声明,只靠C++编译器生成一个混乱的名称,但这是就要求 dll 的编译器与主调程序的编译器相同,

则它们生成的混乱名称也相同)

 

即使是 extern "C" 碰到 stdcall 时,当采用GetProcAddress也找不到 add 这个函数,(因为名称已经变掉了,静态调用时,编译器也会按stdcall生成的函数名_add@8 在外部查找函数,所以不影响,但动态调用则不行),无语了,这时候要 def 文件 登场

 

当启用def文件后,不再需要__declspec(dllexport)申明相关函数为导出函数

LIBRARY "xxx"

EXPORTS
       add

 

dll中文件名都变成add,但不影响导入库lib文件,也就是静态调用动态链接库时,仍要确保 两边 编译器 生成的函数名一样,所以通常推荐两边同时声明 extern "C" ,不然很难确保不同的c++编译器生成相同的函数名,这里为了搞清问题,两边使用同一种编译器,考虑了四种情况。

 

例子:

在动态链接库 xxx.dll 代码中,(已启用def文件,所以省略__declspec(dllexport))

void add(int, int)
{
return ;
}

以c++方式导出函数,编译器处理为?add@@YAXHH@Z
主调程序中,GetProcAddress调用成功,静态调用时,函数申明分别为

函数申明错误 
extern "C" void add(int,int);无法解析的外部符号_add链接不通过
void add(int,int);/(同一个编译器给了个同一函数名来外部链接)        链接通过

 

在xxx.dll 代码中,

extern "C" void add(int, int)
{
return ;
}

以c方式导出函数,编译器处理为_add(为什么不是add?)
因为启用了def文件,所以主调程序的GetProcAddress无论如何都成功,而静态调用情况如下:

函数声明错误 
extern "C" void add(int,int);/同样生成_add外部链接成功链接通过
void add(int,int);"void __cdecl add(int,int)" (?add@@YAXHH@Z)       链接不通过

 

参考:

http://blog.csdn.net/wujian53/article/details/706975
http://hi.baidu.com/luosiyong/item/88a97b0e3bd0ee8802ce1b0b
http://blog.csdn.net/friday5pm/article/details/1532212

 

相关代码下载,dll 文件的位置需在运行时再调整,否则提示找不到xx.dll

http://download.csdn.net/detail/silyvin/5467251

 

这篇关于再议:__cdecl与__stdcall 调用约定在动态链接库调用中不同的表现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/501070

相关文章

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

Java调用Python脚本实现HelloWorld的示例详解

《Java调用Python脚本实现HelloWorld的示例详解》作为程序员,我们经常会遇到需要在Java项目中调用Python脚本的场景,下面我们来看看如何从基础到进阶,一步步实现Java与Pyth... 目录一、环境准备二、基础调用:使用 Runtime.exec()2.1 实现步骤2.2 代码解析三、

Python Flask实现定时任务的不同方法详解

《PythonFlask实现定时任务的不同方法详解》在Flask中实现定时任务,最常用的方法是使用APScheduler库,本文将提供一个完整的解决方案,有需要的小伙伴可以跟随小编一起学习一下... 目录完js整实现方案代码解释1. 依赖安装2. 核心组件3. 任务类型4. 任务管理5. 持久化存储生产环境

Python如何调用另一个类的方法和属性

《Python如何调用另一个类的方法和属性》在Python面向对象编程中,类与类之间的交互是非常常见的场景,本文将详细介绍在Python中一个类如何调用另一个类的方法和属性,大家可以根据需要进行选择... 目录一、前言二、基本调用方式通过实例化调用通过类继承调用三、高级调用方式通过组合方式调用通过类方法/静

C#控制台程序同步调用WebApi实现方式

《C#控制台程序同步调用WebApi实现方式》控制台程序作为Job时,需同步调用WebApi以确保获取返回结果后执行后续操作,否则会引发TaskCanceledException异常,同步处理可避免异... 目录同步调用WebApi方法Cls001类里面的写法总结控制台程序一般当作Job使用,有时候需要控制

Python用Flask封装API及调用详解

《Python用Flask封装API及调用详解》本文介绍Flask的优势(轻量、灵活、易扩展),对比GET/POST表单/JSON请求方式,涵盖错误处理、开发建议及生产环境部署注意事项... 目录一、Flask的优势一、基础设置二、GET请求方式服务端代码客户端调用三、POST表单方式服务端代码客户端调用四

Python跨文件实例化、跨文件调用及导入库示例代码

《Python跨文件实例化、跨文件调用及导入库示例代码》在Python开发过程中,经常会遇到需要在一个工程中调用另一个工程的Python文件的情况,:本文主要介绍Python跨文件实例化、跨文件调... 目录1. 核心对比表格(完整汇总)1.1 自定义模块跨文件调用汇总表1.2 第三方库使用汇总表1.3 导

使用Python的requests库调用API接口的详细步骤

《使用Python的requests库调用API接口的详细步骤》使用Python的requests库调用API接口是开发中最常用的方式之一,它简化了HTTP请求的处理流程,以下是详细步骤和实战示例,涵... 目录一、准备工作:安装 requests 库二、基本调用流程(以 RESTful API 为例)1.

Python调用LibreOffice处理自动化文档的完整指南

《Python调用LibreOffice处理自动化文档的完整指南》在数字化转型的浪潮中,文档处理自动化已成为提升效率的关键,LibreOffice作为开源办公软件的佼佼者,其命令行功能结合Python... 目录引言一、环境搭建:三步构建自动化基石1. 安装LibreOffice与python2. 验证安装

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示