Windows核心编程 静态库与动态库

资源文件 .rc 文件 会被 rc.exe 变成 .res 文件(二进制文件) 在链接时链接进入 .exe 文件

一、如何保护源码

程序编译链接过程

不想让别人拿到源代码,但是想让其使用功能,根据上图观察,把自己生成的obj给对方,对方拿到obj后,链接到自己的程序中。

新建一个控制台项目进行测试,目录结构

Math.h

Math.cpp

test.cpp

编译后,会生成一个 Math.obj的文件

再新建一个工程使用Math.obj

首先,包含头文件,其次需要导入 .obj文件

方式一:直接托进解决方案里;

方式二:项目-属性-链接器-输入-附加依赖项-箭头-编辑-添加obj文件(一行一个obj文件)

项目目录结构

Math.h

如何兼容C?

test.c  

链接时报错

原因:
C语言的名称粉碎是:_Sub,_Add;
C++的名称粉碎是: ?Sub@@YAHHH@Z,?Add@@YAHHH@Z
编译器拿着“?Sub@@YAHHH@Z”,在obj中匹配C的_Sub,当然匹配不上

解决办法:告诉编译器,名称粉碎的时候,按照C的名称粉碎规则进行粉碎。

C++ 项目使用时,函数声明加上extern "C"后,C++支持extern "C"语法,能够直接使用
C项目使用时,由于函数声明上extern "C",但是C不支持该语法,不认识,所以编译不通过

解决办法:头文件被C 包含的时候前面不加extern "C"   int Add(int n1, int n2);
                  头文件被C++ 包含的时候,声明前面加上extern "C" ,说明用C风格名称粉碎去找实现extern "C"

条件编译宏:这样使用的时候就可以不管是C包含还是Cpp包含了

//要想C 和C++ 都能所用该obj,声明的前面必须加上extern "C",生成的obj文件名称粉碎是C风格的。C++可以使用,C也可以使用
//C++ 项目使用时,函数声明加上extern "C"后,C++支持extern "C"语法,能够直接使用
//C   项目使用时,由于函数声明上extern "C",但是C不支持该语法,不认识,所以编译不通过。//解决办法:头文件被C 包含的时候前面不加extern "C"   int Add(int n1, int n2);
//          头文件被C++ 包含的时候,声明前面加上extern "C" ,说明用C风格名称粉碎去找实现extern "C"  int Add(int n1, int n2);//条件编译宏:这样使用的时候就可以不管是C包含还是Cpp包含了
#ifdef __cplusplus
extern "C"
{
#endif // __cplusplusint Add(int n1, int n2);#ifdef __cplusplus
}
#endif // __cplusplus

上述的obj的方法中,当有很多obj时候,需要拷贝很多的obj,很不方便,考虑将这些obj合并成一个大的“obj”,这时就引出了静态库的概念。

补充:

#pragma once 是一种预处理指令,用于确保头文件只被编译一次。当一个头文件被多次包含在不同的源文件中时,使用 #pragma once 可以防止重复包含,从而避免编译错误和重复定义的问题。

#pragma once 的作用类似于传统的头文件保护宏(header guard),但更加简洁和方便。传统的头文件保护宏需要在头文件开头和结尾分别使用条件编译语句,如 #ifndef HEADER_NAME_H#define HEADER_NAME_H#endif,以确保头文件只被编译一次。而 #pragma once 只需要在头文件的开头使用一次,即可达到相同的效果。

使用 #pragma once 的好处是可以提高编译速度,因为编译器可以直接根据指令判断是否需要重新编译头文件。而传统的头文件保护宏需要进行条件判断,会增加编译时间和额外的预处理工作。

二、静态库 动态库 概述

函数和数据被编译进一个二进制文件(通常扩展名为.lib)。在使用静态库的情况下,在编译可执行文件时,链接器从库中复制这些函数和数据并把它们和应用程序的其他模块组合起来创建最终的可执行文件(.exe文件)。

本质:把所有的obj文件全部打包到一个.lib文件内。

缺点:

  1. 维护困难:如果.lib更新,使用的工程如需更新,则必须重新编译。
  2. 磁盘冗余:如果很多工程使用,就要拷贝很多份.lib文件,这些lib都是一样的
  3. 无法很好的同时兼容C和C++
  4. 其他语言无法使用

动态链接库(DLL) 通常不能直接运行,也不能接收信息,只有在其他模块调用动态链接库中的函数时,才能发挥作用。通常我们把完成某种功能的函数放在一个动态链接库中,提供给其他程序调用。DLL就是整个windows操作系统的基础。动态链接库不能直接运行,也不能接收消息。他们是一些独立的文件。

Windows API中所有的函数都包含在DLL中,其中有3个重要的DLL:

  • Kernel32.dll:包含用于管理内存、进程和线程的函数、例如CreateThread函数。
  • User32.dll:它包含用于执行用户界面任务(如窗口的创建和消息的传送)的函数。例如CreateWindow函数。
  • GDI32.dll:它包含用于画图和显示文本的函数。

使用动态链接库的好处:

  1. 可以采用多种编程语言来编写。
  2. 增强产品的功能(扩展插件)
  3. 提供二次开发的平台(扩展插件)
  4. 简化项目管理(一个团队负责自己团队的dll)
  5. 可以节省磁盘空间和内存
  6. 有助于资源的共享
  7. 有助于实现应用程序的本地化。

三、静态链接库创建与使用

VS2019中直接找到静态链接库,一路确认即可

不适用预编译头即可

项目目录:

pch.h framework.h 文件是作用是减少重复文件编译,提升性能有关。不用管

如果想建立一个自己的静态链接库,直接添加 .h  .cpp文件即可,编译后就可以得到 .lib 文件

使用静态库和使用 .obj 类似

1. 添加头文件,使用者才能知道传的什么参数以及其他
2. 拷贝lib文件和.h头文件到VS工程根目录
3. 添加lib文件到工程的方式(用法):
  a. 直接拖入项目中
  b. 依赖项添加.lib文件
  c. 代码内添加.lib文件 # pragma comment(lib,lib路径)

如何把两个 obj 合成为 lib

静态库中还可以放 全局变量,类(通过源文件右击添加-类)

四、动态链接库创建

新建>>类向导>>项目类型>>.dll动态链接库。

动态链接库中有导出函数和非导出函数:

  • 导出函数:DLL提供给其他应用程序调用的函数
  • 非导出函数:给DLL内的函数调用的函数,中间函数等。

如果想导出函数给外面的工程使用,需要指定函数,告诉编译器哪个函数需要导出

从DLL中导出函数:
为了让DLL导出一些函数,需要在每一个将要被导出的函数前面添加标识符__declspec(dllexport) 

编译:生成DLL文件和LIB文件

LIB文件:称为DLL的导入库文件,是一个特殊的库文件,和静态库文件有着本质上的区别,引入库文件包含该DLL导出的函数和变量的符号名;而DLL文件包含该DLL实际函数和数据。

工程结构:

CTest.h

#pragma once
class __declspec(dllexport) CTest
{
public:void Show();
};

Add.h

#pragma once
#ifdef __cplusplus
extern "C" {
#endif // __cplusplus__declspec(dllexport) int Add(int n1, int n2);__declspec(dllexport) extern int g_nVal;
#ifdef __cplusplus
}
#endif // __cplusplus

Sub.h

#pragma once
#ifdef __cplusplus
extern "C" {
#endif // __cplusplus__declspec(dllexport) int Sub(int n1, int n2);
#ifdef __cplusplus
}
#endif // __cplusplus

Add.cpp

#include "Add.h"
int Add(int n1, int n2)
{return n1 + n2;
}
int g_nVal = 0x12345678;

CTest.cpp

#include "CTest.h"
#include <iostream>
using namespace std;
void CTest::Show()
{cout << "CTest::Foo()" << endl;
}

dll.cpp

#include <iostream>
#include "CTest.h"
#include "Add.h"
#include "Sub.h"
int main()
{std::cout << Add(1, 2) << std::endl;std::cout << Sub(2, 1) << std::endl;CTest test;test.Show();
}

Sub.cpp

#include "Sub.h"
int Sub(int n1, int n2)
{return n1 - n2;
}

编译后生成以下文件:

在下面动态链接库的debug目录下:生成了dll文件;dll.exp 文件是一个输出库文件。

LIB文件:称为DLL的导入库文件,是一个特殊的库文件,和静态库文件有着本质上的区别,引入库文件包含该DLL导出的函数和变量的符号名;而DLL文件包含该DLL实际函数和数据。

查看导出函数工具-DEPENDS,拖进去使用即可

五、动态链接库的两种调用方式

动态链接库的使用

  1. 静态调用:在程序编译的时候将DLL的信息植入可执行文件
  2. 动态调用:在程序中用语句显示地加载DLL,编译器不需要知道任何关于DLL的信息。

显式加载和隐式加载是在使用动态链接库(DLL)时的两种加载方式。下面我将为你解释这两种加载方式的区别:

  1. 隐式加载(Implicit Loading):

    • 在编译时,程序会将对 DLL 的引用嵌入到可执行文件中。
    • 在程序运行时,操作系统会自动加载并初始化 DLL。
    • 隐式加载不需要手动加载 DLL 或指定 DLL 的路径。
    • 函数调用时,直接使用函数名进行调用,编译器会根据嵌入的引用找到对应的函数地址。
    • DLL 的导入函数表会在程序加载时自动解析,可以直接访问 DLL 中的函数。
  2. 显式加载(Explicit Loading):

    • 程序需要显式地通过代码来加载 DLL 并获取其函数地址。
    • 使用 LoadLibrary 函数加载 DLL,并返回一个句柄,表示已加载的 DLL。
    • 使用 GetProcAddress 函数根据函数名获取 DLL 中的函数地址。
    • 加载后的 DLL 需要手动卸载,使用 FreeLibrary 函数释放 DLL 句柄。
    • 函数调用时,需要通过函数指针来调用 DLL 中的函数。

显式加载和隐式加载主要的区别在于加载时机和加载方式。隐式加载在程序运行时自动加载 DLL,并且可以直接调用 DLL 中的函数。而显式加载需要手动加载 DLL,并使用函数指针来调用 DLL 中的函数。显式加载提供了更大的灵活性和控制权,适用于需要在运行时动态加载和卸载 DLL 的情况,而隐式加载则更加简单和方便。

六、动态链接库的静态加载

静态调用步骤:

  1. 新建应用工程。
  2. 通过编译器供给应用程序关于DLL的名称,以及DLL函数的链接参考(.h文件)。这种方式不需要在程序中用代码将DLL加载到内存。
  3. 将DLL和LIB文件拷贝到工程目录下
  4. 将lib文件添加到工程
    1. 方式一:项目>>属性>>链接>>依赖项>>lib名称
    2. 方式二:拖入到项目
  5. 添加头文件>>直接调用头文件中的函数即可。
     

新建一个控制台项目,目录结构如下

Add.h

#pragma once
#ifdef __cplusplus
extern "C" {
#endif // __cplusplus__declspec(dllimport) int Add(int n1, int n2);__declspec(dllimport) extern int g_nVal;
#ifdef __cplusplus
}
#endif // __cplusplus

Sub.h

#pragma once
#ifdef __cplusplus
extern "C" {
#endif // __cplusplus__declspec(dllimport) int Sub(int n1, int n2);
#ifdef __cplusplus
}
#endif // __cplusplus

CTest.h 

#pragma once
class __declspec(dllimport) CTest
{
public:void Show();
};

.cpp

#include <iostream>
#include "CTest.h"
#include "Add.h"
#include "Sub.h"
#pragma comment(lib,"dll.lib")int main()
{std::cout << Add(1, 2) << std::endl;std::cout << Sub(3, 4) << std::endl;std::cout << std::hex << g_nVal << std::endl;CTest test;test.Show();
}

动态链接库与可执行文件放在同一目录下:

lib文件放到根目录下

方式一:使用 extern 声明外部函数

extern 关键字在C和C++中都有着重要的作用,它的具体含义取决于它所修饰的变量或函数。

在C语言中,extern 关键字用于声明一个变量或函数是在别处定义的,告诉编译器该变量或函数的定义在其他地方,不在当前文件中。具体来说:

  1. 外部变量声明:在C语言中,当你在一个文件中使用了一个全局变量,而该变量的定义在另外一个文件中时,你可以使用 extern 来声明该变量,以便编译器知道该变量的定义在其他地方。

    // 在一个文件中声明外部变量
    extern int global_var; // 声明global_var是在其他文件中定义的全局变量
    
  2. 外部函数声明extern 也可以用于声明外部函数,在这种情况下,它告诉编译器该函数的定义在其他地方,不在当前文件中。

    // 外部函数声明
    extern void external_function(); // 声明external_function是在其他文件中定义的函数
    

 

方式二:__declspec(dllimport) 声明外部函数

除了使用extern 关键字表明函数是外部定义的之外,还可以使用标识符:__declspec(dllimport) 来表明函数是从动态链接库中引入的。
__declspec(dllimport) 与使用extern 关键字这种方式相比,再使用__declspec(dllimport) 标识符声明外部函数时,它将告诉编译器该函数是从动态链接库中引入的,编译器可以生成运行效率更高的代码。所以调用的函数来自于动态链接库,则应该使用这种方式来声明外部函数。

标准来说,无论是全局变量,还是函数都是需要使用关键字dllimport

使用宏优化导关键字dllimport

代码如下:

#pragma once
#ifdef DLL_EXPORT
#define DLL_API __declspec(dllexport)
#else
#define DLL_API __declspec(dllimport)
#endif 

解释一下这段代码的含义:

  1. #ifdef DLL_EXPORT:这个条件编译指令用于检查是否定义了DLL_EXPORT宏。如果定义了,表示当前是在编译DLL库的源代码,需要导出函数和数据。如果没有定义,则表示当前是在使用DLL的客户端代码,需要导入函数和数据。

  2. #define DLL_API __declspec(dllexport):如果DLL_EXPORT被定义了,那么将DLL_API宏定义为__declspec(dllexport)__declspec(dllexport)是在Windows平台上用于标记要导出的函数和数据的修饰符。

  3. #else:如果DLL_EXPORT未被定义,执行下面的代码块。

  4. #define DLL_API __declspec(dllimport):将DLL_API宏定义为__declspec(dllimport)__declspec(dllimport)是在Windows平台上用于标记要导入的函数和数据的修饰符。

通过这种方式,可以在编写DLL库时使用DLL_API宏来修饰要导出的函数和数据,而在使用DLL库的客户端代码中使用DLL_API宏来修饰要导入的函数和数据。这样可以保证在编译时正确地处理导出和导入函数的修饰符。

动态链接库创建优化

预处理器包含宏:DLL_EXPORT

DLL_API 替换 __declspec(dllexport),并包含 common.h 头文件

加载动态链接库优化

不需要包含这个宏即可, .h 文件 包含 common.h 文件 替换宏即可

读取全局变量来说,最好直接无脑加 extern 关键字

补充:在实现动态链接库时,可以不导出整个类,而只导出该类中的某些函数,在导出类的成员函数的时候需要注意,该函数必须具有public类型的访问权限。

兼容 C

如果使用C++语言编写了一个DLL,那么使用C语言编写的客户端程序访问DLL中的函数就会出现问题,因为后者将使用函数原始名称来调用DLL中的函数,而C++编译器已经对该名称进行了改编,所以C语言编写的客户端程序就找不到所需的DLL导出函数。

#pragma once
#ifdef DLL_EXPORT
#define DLL_API extern "C" __declspec(dllexport)
#else
#define DLL_API extern "C" __declspec(dllimport)
#endif 

利用限定符 extern “C” 可以解决C++和C语言之间的相互调用是函数命名问题。但是这种方法有一个缺陷,就是不能用于导出一个类的成员函数和全局变量,只能用于导出全局函数这种情况。
如果导出函数的调用约定发生了变化,那么即使使用了限定符 extern “C” ,该函数的名字仍然会发生改编。
在这种情况下,可以通过一个称为模块定义文件(DEF) 的方式来解决名字改编问题。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ps找不到msvcp140.dll怎么办?亲测5个有效的修复方法分享

运行Photoshop时提示找不到MSVCP140.dll&#xff0c;这是因为计算机MSVCP140.dll文件丢失或者损坏。msvcp140.dll是微软Visual C 2015运行库的一部分&#xff0c;它包含了许多用于支持C运行的函数和类。当我们在使用某些程序时&#xff0c;如果这个程序依赖于msvcp140.dll&…

Day35力扣打卡

打卡记录 相邻字符不同的最长路径&#xff08;树状DP&#xff09; 链接 若节点也存在父节点的情况下&#xff0c;传入父节点参数&#xff0c;若是遍历到父节点&#xff0c;直接循环里 continue。 class Solution:def longestPath(self, parent: List[int], s: str) -> in…

一个C语言程序的分析:运行速度和文件大小以及变量初始值

环境 Ubuntu 22.04gcc 11.4.0Window 11Microsoft Visual Studio Community 2022 (64-bit) - Current Version 17.6.2 运行速度 一个C程序 test1.c 如下&#xff1a; int array[30000][30000];int main() {for (int i 0; i < 30000; i)for (int j 0; j < 30000; j) …

VMware Workstation 与 Device/Credential Guard 不兼容 解决办法

问题描述 问题描述&#xff1a; VMware 启动虚拟机会报错。无法运行。 错误信息&#xff1a;VMware Workstation 与 Device/Credential Guard 不兼容。在禁用 Device/Credential Guard 原因分析&#xff1a; 通常原因是 Window 系统开启了 内置的Hyper-V 虚拟机。 解决方案&…

sftp 从windows10向linux(centos7)传输文件

前言背景&#xff1a;该示例是需要从windows10向本地linux系统传输一个qt安装文件&#xff0c;不想或者无法安装xftp这些传输工具&#xff0c;直接通过命令传输&#xff1b; 首先保证windows10 ping通linux系统ip&#xff0c;linux ping 通windows10系统&#xff1b; 注意&am…

计算机网络的标准化工作及相关组织

一、国际化组织 计算机网络的标准化工作由一些主要的组织来进行管理和推动。以下是几个主要的计算机网络标准化的国际组织及其相关的标准&#xff1a; 1. 国际标准化组织&#xff08;ISO&#xff09;&#xff1a;国际标准化组织负责制定各种行业的标准&#xff0c;包括计算机…

【Java程序员面试专栏 专业技能篇】Java SE核心面试指引(一):基础知识考察

关于Java SE部分的核心知识进行一网打尽,包括四部分:基础知识考察、面向对象思想、核心机制策略、Java新特性,通过一篇文章串联面试重点,并且帮助加强日常基础知识的理解,全局思维导图如下所示 本篇Blog为第一部分:基础知识考察,子节点表示追问或同级提问 基本概念 …

⑩⑥ 【MySQL】详解 触发器TRIGGER,协助 确保数据的完整性,日志记录,数据校验等操作。

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ 触发器 ⑩⑥ 【MySQL】触发器详解1. 什么是触发…

【智能家居】5、主流程设计以及外设框架编写与测试

目录 一、主流程设计 1、工厂模式结构体定义 &#xff08;1&#xff09;指令工厂 inputCmd.h &#xff08;2&#xff09;外设工厂 controlDevices.h 二、外设框架编写 1、创建外设工厂对象bathroomLight 2、编写相关函数框架 3、将浴室灯相关操作插入外设工厂链表等待被调…

设计模式-组合模式-笔记

“数据结构”模式 常常有一些组件在内部具有特定的数据结构&#xff0c;如果让客户程序依赖这些特定数据结构&#xff0c;将极大地破坏组件的复用。这时候&#xff0c;将这些特定数据结构封装在内部&#xff0c;在外部提供统一的接口&#xff0c;来实现与特定数据结构无关的访…

六大排序(插入排序、希尔排序、冒泡排序、选择排序、堆排序、快速排序)未完

文章目录 排序一、 排序的概念1.排序&#xff1a;2.稳定性&#xff1a;3.内部排序&#xff1a;4.外部排序&#xff1a; 二、插入排序1.直接插入排序2.希尔排序 三、选择排序1.直接选择排序方法一方法二直接插入排序和直接排序的区别 2.堆排序 四、交换排序1.冒泡排序2.快速排序…

Unity Meta Quest 一体机开发(七):配置玩家 Hand Grab 功能

文章目录 &#x1f4d5;教程说明&#x1f4d5;玩家物体配置 Hand Grab Interactor⭐添加 Hand Grab Interactor 物体⭐激活 Hand Grab Visual 和 Hand Grab Glow⭐更新 Best Hover Interactor Group &#x1f4d5;配置可抓取物体&#xff08;无抓取手势&#xff09;⭐刚体和碰撞…