Visual Studio 2022静态库与动态库创建及使用完全指南
在C++开发中,库(Library)是代码复用的重要方式。本教程将详细介绍如何在Visual Studio 2022中创建和使用静态库(.lib)和动态库(.dll),每种库类型都会有完整的创建步骤和实际示例。
第一部分:静态库的创建与使用
1. 创建静态库项目
- 打开VS2022,选择"创建新项目"
- 搜索"静态库",选择"C++静态库"模板
- 项目命名为"MathStaticLib",解决方案命名为"LibraryDemo"
- 点击"创建"
2. 添加静态库代码
在"头文件"文件夹中添加MathFunctions.h:
// MathFunctions.h
#pragma oncenamespace MathStatic
{// 计算数字的平方int square(int x);// 计算两个数的最大公约数int gcd(int a, int b);// 判断数字是否为偶数bool isEven(int num);
}
在"源文件"文件夹中添加MathFunctions.cpp:
// MathFunctions.cpp
#include "MathFunctions.h"namespace MathStatic
{int square(int x) {return x * x;}int gcd(int a, int b) {while (b != 0) {int temp = b;b = a % b;a = temp;}return a;}bool isEven(int num) {return (num % 2) == 0;}
}
3. 配置静态库项目
- 右键项目 → 属性
- 确保配置为"Debug"和"x64"(根据你的需求选择)
- 配置属性 → 常规 → 配置类型:静态库(.lib)
- C/C++ → 预编译头 → 设置为"不使用预编译头"
- 点击"应用" → “确定”
4. 生成静态库
- 菜单栏选择"生成" → “生成解决方案”(Ctrl+Shift+B)
- 在输出窗口查看生成结果,确认没有错误
- 生成的静态库文件位于:
解决方案目录\x64\Debug\MathStaticLib.lib
5. 创建使用静态库的控制台应用
- 在解决方案中添加新项目
- 选择"控制台应用"模板,命名为"StaticLibClient"
- 右键"StaticLibClient"项目 → 属性
- 配置属性 → VC++目录 → 包含目录:添加
$(SolutionDir)MathStaticLib - 链接器 → 常规 → 附加库目录:添加
$(SolutionDir)\x64\Debug - 链接器 → 输入 → 附加依赖项:添加
MathStaticLib.lib
6. 编写测试代码
修改StaticLibClient的main.cpp:
#include <iostream>
#include "MathFunctions.h"int main()
{std::cout << "静态库使用示例:" << std::endl;int num = 5;std::cout << num << "的平方是: " << MathStatic::square(num) << std::endl;int a = 56, b = 98;std::cout << a << "和" << b << "的最大公约数是: " << MathStatic::gcd(a, b) << std::endl;std::cout << num << "是" << (MathStatic::isEven(num) ? "偶数" : "奇数") << std::endl;return 0;
}
7. 设置项目依赖
- 右键解决方案 → 项目依赖项
- 设置"StaticLibClient"依赖于"MathStaticLib"
- 生成解决方案并运行
第二部分:动态库的创建与使用
1. 创建动态库项目
- 在解决方案中添加新项目
- 搜索"动态链接库",选择"DLL"模板
- 命名为"StringDynamicLib"
- 点击"创建"
2. 添加动态库代码
删除自动生成的dllmain.cpp和pch文件。
添加StringUtils.h:
// StringUtils.h
#pragma once// DLL导出宏
#ifdef STRINGDYNAMICLIB_EXPORTS
#define STRING_API __declspec(dllexport)
#else
#define STRING_API __declspec(dllimport)
#endifnamespace StringDynamic
{// 反转字符串STRING_API void reverse(char* str, int length);// 统计字符出现次数STRING_API int countChar(const char* str, char c);// 连接两个字符串STRING_API char* concatenate(const char* str1, const char* str2);
}
添加StringUtils.cpp:
// StringUtils.cpp
#include "StringUtils.h"
#include <cstring>
#include <cstdlib>namespace StringDynamic
{STRING_API void reverse(char* str, int length) {int start = 0;int end = length - 1;while (start < end) {std::swap(str[start], str[end]);start++;end--;}}STRING_API int countChar(const char* str, char c) {int count = 0;while (*str) {if (*str == c) count++;str++;}return count;}STRING_API char* concatenate(const char* str1, const char* str2) {size_t len1 = strlen(str1);size_t len2 = strlen(str2);char* result = (char*)malloc(len1 + len2 + 1);strcpy(result, str1);strcat(result, str2);return result;}
}
3. 配置动态库项目
- 右键项目 → 属性
- 配置属性 → 常规 → 配置类型:动态库(.dll)
- C/C++ → 预处理器 → 预处理器定义:添加
STRINGDYNAMICLIB_EXPORTS - 生成解决方案
4. 生成动态库文件
生成成功后,在输出目录中会生成:
StringDynamicLib.dll(动态库文件)StringDynamicLib.lib(导入库文件)
5. 创建使用动态库的控制台应用
- 在解决方案中添加新项目
- 选择"控制台应用"模板,命名为"DynamicLibClient"
- 右键项目 → 属性
- 配置属性 → VC++目录 → 包含目录:添加
$(SolutionDir)StringDynamicLib - 链接器 → 常规 → 附加库目录:添加
$(SolutionDir)\x64\Debug - 链接器 → 输入 → 附加依赖项:添加
StringDynamicLib.lib
6. 编写测试代码
修改DynamicLibClient的main.cpp:
#include <iostream>
#include "StringUtils.h"int main()
{std::cout << "动态库使用示例:" << std::endl;char str[] = "Hello Dynamic Library";std::cout << "原始字符串: " << str << std::endl;StringDynamic::reverse(str, strlen(str));std::cout << "反转后字符串: " << str << std::endl;char target = 'l';std::cout << "字符'" << target << "'出现次数: " << StringDynamic::countChar(str, target) << std::endl;const char* str1 = "Hello, ";const char* str2 = "World!";char* combined = StringDynamic::concatenate(str1, str2);std::cout << "连接后的字符串: " << combined << std::endl;free(combined); // 释放动态分配的内存return 0;
}
7. 部署动态库
-
将
StringDynamicLib.dll复制到以下任一位置:DynamicLibClient项目的x64\Debug目录- 系统PATH包含的目录
- 与可执行文件相同的目录
-
设置项目依赖:
- 右键解决方案 → 项目依赖项
- 设置"DynamicLibClient"依赖于"StringDynamicLib"
-
生成解决方案并运行
静态库与动态库对比
| 特性 | 静态库(.lib) | 动态库(.dll) |
|---|---|---|
| 编译时行为 | 代码直接嵌入到可执行文件中 | 运行时动态加载 |
| 文件部署 | 只需可执行文件 | 需要可执行文件和DLL文件 |
| 内存使用 | 可能增加可执行文件大小 | 多个程序可共享同一个DLL |
| 更新维护 | 需要重新编译整个程序 | 只需替换DLL文件 |
| 加载速度 | 启动快 | 首次加载稍慢 |
| 适用场景 | 小型工具、嵌入式系统 | 大型应用、插件系统 |
常见问题
Q1: 程序找不到DLL文件?
A: 确保DLL文件位于:
- 可执行文件同一目录
- 系统PATH包含的目录
- Windows系统目录
Q2: 静态库和动态库可以混合使用吗?
A: 可以,一个程序可以同时链接静态库和动态库。
Q3: 如何调试动态库?
A: 在动态库项目中设置断点,确保调试器可以访问库的源代码和PDB文件。
Q4: 为什么动态库函数需要导出声明?
A: 导出声明(__declspec(dllexport))告诉编译器哪些函数应该对外可见,没有导出的函数无法从外部调用。
Q5: 如何查看动态库导出的函数?
A: 使用Visual Studio自带的dumpbin工具:
dumpbin /exports StringDynamicLib.dll
相关文章:
Visual Studio 2022静态库与动态库创建及使用完全指南
在C开发中,库(Library)是代码复用的重要方式。本教程将详细介绍如何在Visual Studio 2022中创建和使用静态库(.lib)和动态库(.dll),每种库类型都会有完整的创建步骤和实际示例。 第…...
C++中常见符合RAII思想的设计有哪些
文章目录 **一、标准库中的 RAII 类**1. **智能指针**2. **文件操作类**3. **锁管理类**4. **容器类**5. **线程管理** **二、自定义 RAII 类的常见场景**1. **数据库连接**2. **图形资源管理(如 OpenGL 纹理)**3. **网络套接字**4. **事务处理**5. **临…...
CUDA Memory Fence 函数的功能与硬件实现细节
CUDA Memory Fence 函数的功能与硬件实现细节 Memory Fence 的基本功能 CUDA中的memory fence函数用于控制内存操作的可见性顺序,确保在fence之前的内存操作对特定范围内的线程可见。主要功能包括: 排序内存操作:确保fence之前的内存操作在…...
CSS学习笔记5——渐变属性+盒子模型阶段案例
目录 通俗易懂的解释 渐变的类型 1、线性渐变 渐变过程 2、径向渐变 如何理解CSS的径向渐变,以及其渐变属性 通俗易懂的解释 渐变属性 1. 形状(Shape) 2. 大小(Size) 3. 颜色停靠点(Color Sto…...
[Java微服务架构]4_服务通信之客户端负载均衡
欢迎来到啾啾的博客🐱,一个致力于构建完善的Java程序员知识体系的博客📚,记录学习的点滴,分享工作的思考、实用的技巧,偶尔分享一些杂谈💬。 欢迎评论交流,感谢您的阅读😄…...
基于SpringBoot实现的高校实验室管理平台功能四
一、前言介绍: 1.1 项目摘要 随着信息技术的飞速发展,高校实验室的管理逐渐趋向于信息化、智能化。传统的实验室管理方式存在效率低下、资源浪费等问题,因此,利用现代技术手段对实验室进行高效管理显得尤为重要。 高校实验室作为…...
吴恩达深度学习复盘(1)神经网络与深度学习的发展
一、神经网络的起源与生物学动机 灵感来源 神经网络的最初动机源于对生物大脑的模仿。20 世纪 50 年代,科学家试图通过软件模拟神经元的工作机制(如树突接收信号、轴突传递信号),构建类似人类大脑的信息处理系统。 生物神经元的简…...
用Python实现资本资产定价模型(CAPM)
使用 Python 计算资本资产定价模型(CAPM)并获取贝塔系数(β)。 步骤 1:导入必要的库 import pandas as pd import yfinance as yf import statsmodels.api as sm import matplotlib.pyplot as plt 步骤 2࿱…...
Linux进程管理之子进程的创建(fork函数)、子进程与线程的区别、fork函数的简单使用例子、子进程的典型应用场景、父进程等待子进程结束后自己再结束
收尾 进程终止:子进程通过exit()或_exit()终止,父进程通过wait()或waitpid()等待子进程终止,并获取其退出状态。?其实可以考虑在另一篇博文中来写 fork函数讲解 fork函数概述 fork() 是 Linux 中用于创建新进程的系统调用。当…...
妙用《甄嬛传》中的选妃来记忆概率论中的乘法公式
强烈推荐最近在看的不错的B站概率论课程 《概率统计》正课,零废话,超精讲!【孔祥仁】 《概率统计》正课,零废话,超精讲!【孔祥仁】_哔哩哔哩_bilibili 其中概率论中的乘法公式,老师用了《甄嬛传…...
虚幻基础:UI
文章目录 控件蓝图可以装载其他控件蓝图可以安装其他蓝图接口 填充:相对于父组件填充水平框尺寸—填充—0.5:改变填充的尺寸填充—0.5:改变与父组件的距离 锚点:相对于父组件的控件坐标系原点,屏幕比例改变时ÿ…...
【MySQL篇】事务管理,事务的特性及深入理解隔离级别
目录 一,什么是事务 二,事务的版本支持 三,事务的提交方式 四,事务常见操作方式 五,隔离级别 1,理解隔离性 2,查看与设置隔离级别 3,读未提交(read uncommitted&a…...
项目实战-角色列表
抄上一次写过的代码: import React, { useState, useEffect } from "react"; import axios from axios; import { Button, Table, Modal } from antd; import { BarsOutlined, DeleteOutlined, ExclamationCircleOutlined } from ant-design/icons;const…...
fetch`的语法规则及常见用法
fetch() 是 JavaScript 用于发送 HTTP 请求的内置 API,功能强大,语法简洁。以下是 fetch 的语法规则及常见用法。 1. fetch 基本语法 fetch(url, options).then(response > response.json()) // 解析 JSON 响应体.then(data > console.log(data))…...
如何排查java程序的宕机和oom?如何解决宕机和oom?
排查oom 用jmap生成我们的堆空间的快照Heap Dump(堆转储文件),来分析我们的内存占用 用可视化工具,例如java中的jhat分析Heap Dump文件 ,它分析完会通过一个浏览器打开一个可视化页面展示分析结果 根据oom的类型来调…...
26_ajax
目录 了解 接口 前后端交互 一、安装服务器环境 nodejs ajax发起请求 渲染响应结果 get方式传递参数 post方式传递参数 封装ajax_上 封装ajax下 了解 清楚前后端交互就可以写一些后端代码了。小项目 现在写项目开发的时候都是前后端分离 之前都没有前端这个东西&a…...
代理模式(Proxy Pattern)实现与对比
代理模式(Proxy Pattern)实现与对比 1. 虚拟代理(Virtual Proxy) 定义:延迟加载对象,避免资源浪费。 适用场景:大文件或资源的加载(如图片、数据库连接)。 代码示例 /…...
MySQL - 数据库基础操作
SQL语句 结构化查询语言(Structured Query Language),在关系型数据库上执行数据操作、数据检索以及数据维护的标准语言。 分类 DDL 数据定义语言(Data Definition Language),定义对数据库对象(库、表、列、索引)的操作。 DML 数据操作语言(Data Manip…...
Spring Boot热部署插件
在实际开发中,我们修改某些代码或页面都需要重启应用后才能生效,如果每次都手动重启,会降低了开发效率;热部署是指当我们修改代码后,服务能自动重启加载新修改的内容,这样大大提高了我们开发的效率…...
pip install cryptacular卡住,卡在downloading阶段
笔者安装pip install cryptacular卡在downloading阶段,但不知道为何 Collecting cryptacularCreated temporary directory: /tmp/pip-unpack-qfbl8f08http://10.170.22.41:8082 "GET http://repo.huaweicloud.com/repository/pypi/packages/42/69/34d478310d6…...
AI大模型从0到1记录学习 day09
第 8 章 面向对象之类和对象 8.1 面向过程和面向对象 面向过程编程(Procedural Programming)和面向对象编程(OOP)是两种不同的编程范式,它们在软件开发中都有广泛的应用。 Python是一种混合型的语言,既支持…...
【FW】ADB指令分类速查清单
1. 设备管理 指令核心作用adb devices列出已连接设备adb reboot重启设备adb reboot bootloader进入Bootloader模式adb reboot recovery进入Recovery模式adb root获取Root权限(需设备支持)adb remount挂载系统分区为可读写 2. 应用管理 指令核心作用adb…...
Kafka中的消息是如何存储的?
大家好,我是锋哥。今天分享关于【Kafka中的消息是如何存储的?】面试题。希望对大家有帮助; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 在 Kafka 中,消息是通过 日志(Log) 的方式进行存储的。…...
Altium Designer——同时更改多个元素的属性(名称、网络标签、字符串标识)
右键要更改的其中一个对象,选择查找相似… 进入到筛选界面,就是选择你要多选的对象的共同特点(名字、大小等等),我这里要更改的是网络标签,所以我选择Text设置为一样。 点击应用就是应用该筛选调节&#…...
当模板方法模式遇上工厂模式:一道优雅的烹饪架构设计
当模板方法模式遇上工厂模式:一道优雅的烹饪架构设计 模式交响曲的实现模板方法模式搭建烹饪骨架(抽象类)具体菜品(子类) 工厂模式 模式协作的优势呈现扩展性演示运行时流程控制 完整代码 如果在学习 设计模式的过程中…...
c++位运算总结
在C中,位运算是对二进制位进行操作的运算,主要有以下几种: 1. 按位与( & ):两个操作数对应位都为1时,结果位才为1,否则为0。例如 3 & 5 , 3 二进制是 0000 0011…...
企业级知识库建设:自建与开源产品集成的全景解析 —— 产品经理、CTO 与 CDO 的深度对话
文章目录 一、引言二、主流产品与方案对比表三、自建方案 vs. 开源产品集成:技术路径对比3.1 自建方案3.2 开源产品集成方案 四、结论与个人观点 一、引言 在当今数据驱动的商业环境中,构建高质量的知识库已成为企业数字化转型的关键一环。本博客分别从…...
Python小练习系列 Vol.6:单词搜索(网格回溯)
🧠 Python小练习系列 Vol.6:单词搜索(网格回溯) 🔍 本期我们来挑战一道 LeetCode 上经典的网格型回溯题 —— 单词搜索,考察对 DFS 状态恢复的掌握! 🧩 一、题目描述 给定一个 m x…...
shell脚本--MySQL简单调用
实现功能 增 数据库的创建,数据表的创建已经实现 创建用户 删 删除数据库, 删除库下的某个表, 删除某个用户 改 暂无 查 查看所有的数据库, 查看某个库下的所有数据表, 查看某个表的结构, 查…...
vue3项目配置别名
vue3项目配置别名 src别名的配置TypeScript 编译配置如果出现/别名引入报找不到的问题 src别名的配置 在开发项目的时候文件与文件关系可能很复杂,因此我们需要给src文件夹配置一个别名!!! // vite.config.ts import {defineCon…...
