Visual Studio 2022静态库与动态库创建及使用完全指南

在C++开发中,库(Library)是代码复用的重要方式。本教程将详细介绍如何在Visual Studio 2022中创建和使用静态库(.lib)和动态库(.dll),每种库类型都会有完整的创建步骤和实际示例。

第一部分:静态库的创建与使用

1. 创建静态库项目

  1. 打开VS2022,选择"创建新项目"
  2. 搜索"静态库",选择"C++静态库"模板
  3. 项目命名为"MathStaticLib",解决方案命名为"LibraryDemo"
  4. 点击"创建"

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. 配置静态库项目

  1. 右键项目 → 属性
  2. 确保配置为"Debug"和"x64"(根据你的需求选择)
  3. 配置属性 → 常规 → 配置类型:静态库(.lib)
  4. C/C++ → 预编译头 → 设置为"不使用预编译头"
  5. 点击"应用" → “确定”

4. 生成静态库

  1. 菜单栏选择"生成" → “生成解决方案”(Ctrl+Shift+B)
  2. 在输出窗口查看生成结果,确认没有错误
  3. 生成的静态库文件位于:解决方案目录\x64\Debug\MathStaticLib.lib

5. 创建使用静态库的控制台应用

  1. 在解决方案中添加新项目
  2. 选择"控制台应用"模板,命名为"StaticLibClient"
  3. 右键"StaticLibClient"项目 → 属性
  4. 配置属性 → VC++目录 → 包含目录:添加$(SolutionDir)MathStaticLib
  5. 链接器 → 常规 → 附加库目录:添加$(SolutionDir)\x64\Debug
  6. 链接器 → 输入 → 附加依赖项:添加MathStaticLib.lib

6. 编写测试代码

修改StaticLibClientmain.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. 设置项目依赖

  1. 右键解决方案 → 项目依赖项
  2. 设置"StaticLibClient"依赖于"MathStaticLib"
  3. 生成解决方案并运行

第二部分:动态库的创建与使用

1. 创建动态库项目

  1. 在解决方案中添加新项目
  2. 搜索"动态链接库",选择"DLL"模板
  3. 命名为"StringDynamicLib"
  4. 点击"创建"

2. 添加动态库代码

删除自动生成的dllmain.cpppch文件。

添加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. 配置动态库项目

  1. 右键项目 → 属性
  2. 配置属性 → 常规 → 配置类型:动态库(.dll)
  3. C/C++ → 预处理器 → 预处理器定义:添加STRINGDYNAMICLIB_EXPORTS
  4. 生成解决方案

4. 生成动态库文件

生成成功后,在输出目录中会生成:

  • StringDynamicLib.dll (动态库文件)
  • StringDynamicLib.lib (导入库文件)

5. 创建使用动态库的控制台应用

  1. 在解决方案中添加新项目
  2. 选择"控制台应用"模板,命名为"DynamicLibClient"
  3. 右键项目 → 属性
  4. 配置属性 → VC++目录 → 包含目录:添加$(SolutionDir)StringDynamicLib
  5. 链接器 → 常规 → 附加库目录:添加$(SolutionDir)\x64\Debug
  6. 链接器 → 输入 → 附加依赖项:添加StringDynamicLib.lib

6. 编写测试代码

修改DynamicLibClientmain.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. 部署动态库

  1. StringDynamicLib.dll复制到以下任一位置:

    • DynamicLibClient项目的x64\Debug目录
    • 系统PATH包含的目录
    • 与可执行文件相同的目录
  2. 设置项目依赖:

    • 右键解决方案 → 项目依赖项
    • 设置"DynamicLibClient"依赖于"StringDynamicLib"
  3. 生成解决方案并运行

静态库与动态库对比

特性静态库(.lib)动态库(.dll)
编译时行为代码直接嵌入到可执行文件中运行时动态加载
文件部署只需可执行文件需要可执行文件和DLL文件
内存使用可能增加可执行文件大小多个程序可共享同一个DLL
更新维护需要重新编译整个程序只需替换DLL文件
加载速度启动快首次加载稍慢
适用场景小型工具、嵌入式系统大型应用、插件系统

常见问题

Q1: 程序找不到DLL文件?
A: 确保DLL文件位于:

  1. 可执行文件同一目录
  2. 系统PATH包含的目录
  3. Windows系统目录

Q2: 静态库和动态库可以混合使用吗?
A: 可以,一个程序可以同时链接静态库和动态库。

Q3: 如何调试动态库?
A: 在动态库项目中设置断点,确保调试器可以访问库的源代码和PDB文件。

Q4: 为什么动态库函数需要导出声明?
A: 导出声明(__declspec(dllexport))告诉编译器哪些函数应该对外可见,没有导出的函数无法从外部调用。

Q5: 如何查看动态库导出的函数?
A: 使用Visual Studio自带的dumpbin工具:

dumpbin /exports StringDynamicLib.dll

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

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

相关文章

从24GHz到71GHz:Sivers半导体的广泛频率范围5G毫米波产品解析

在5G技术的浪潮中&#xff0c;Sivers半导体推出了创新的毫米波无线产品&#xff0c;为通信行业带来高效、可靠的解决方案。这些产品支持从24GHz到71GHz的频率&#xff0c;覆盖许可与非许可频段&#xff0c;适应高速、低延迟的通信场景。 5G通信频段的一点事儿及Sivers毫米波射频…

LocalDateTime序列化总结

版权说明&#xff1a; 本文由CSDN博主keep丶原创&#xff0c;转载请保留此块内容在文首。 原文地址&#xff1a; https://blog.csdn.net/qq_38688267/article/details/146703276 文章目录 1.背景2.序列化介绍常见场景关键问题 3.总体方案4.各场景实现方式WEB接口EasyExcelMybat…

分享一个Pyside6实现web数据展示界面的效果图

今天又是有问题直接找DS的一天&#xff0c;每日一问&#xff0c;今天我的问题是“怎么将pyside6生成的界面转成web界面&#xff0c;使用python语言实现web界面”&#xff0c;等了一会&#xff0c;DS给我提供了两种方案&#xff0c;方案如下&#xff1a; 然后&#xff0c;让我们…

GAMMA数据处理(十)

今天向别人请教了一个问题&#xff0c;刚无意中搜索到了一模一样的问题 不知道这个怎么解决... ok 解决了 有一个GAMMA的命令可转换 但是很奇怪 完全对不上 转换出来的行列号 不知道为啥 再试试 是因为经纬度坐标的小数点位数 de as

[从零开始学习JAVA ] 深入多线程

前言&#xff1a; 当今软件开发领域中&#xff0c;多线程编程已成为一项至关重要的技能。然而&#xff0c;要编写出高效、可靠的多线程程序并不容易。多线程编程面临着许多挑战&#xff0c;如线程安全性、资源共享、死锁等问题。因此&#xff0c;对于初学者来说&#xff0c;深入…

【Python NetworkX】图结构 图绘制

【Python NetworkX】图结构 & 图绘制 1. 简介 & 安装1.1 简介1.2 安装1.3 导入 2. 图2.1 无向图2.2 有向图2.3 重边无向图2.4 重边有向图2.5 图属性 3. 节点3.1 添加节点3.2 移除节点3.3 节点属性3.4 检查节点状态 4. 边4.1 添加边4.2 移除边4.3 边属性4.4 检查边状态 …

Kubernetes》k8s》Containerd 、ctr 、cri、crictl

containerd ctr crictl ctr 是 containerd 的一个客户端工具。 crictl 是 CRI 兼容的容器运行时命令行接口&#xff0c;可以使用它来检查和调试 k8s 节点上的容器运行时和应用程序。 ctr -v 输出的是 containerd 的版本&#xff0c; crictl -v 输出的是当前 k8s 的版本&#x…

【湖北工业大学2025年ACM校赛(同步赛)】题解

比赛链接 A. 蚂蚁上树 题目大意 给定一棵 n n n 个结点的树&#xff0c;根结点为 1 1 1。每个 叶结点 都有一只蚂蚁&#xff0c;每过 1 1 1 秒钟&#xff0c;你可以选一些蚂蚁往其 父结点 走一步&#xff0c;但是要求任意两只蚂蚁都不能在同一个 非根结点 上。 问至少要…

CS2 DEMO导入blender(慢慢更新咯)

流程&#xff1a;cs2-sourcefilmmaker-blender 工具&#xff1a;cs2tools&#xff0c;cs2manager&#xff0c;blender&#xff0c;blender插件sourceio&#xff0c;source2viewer 导入sfm 工具界面 选择这个 sourceio插件 sourceIO其中新版本导入相机路径不见了&#xff0c…

一周学会Flask3 Python Web开发-SQLAlchemy数据迁移migrate

锋哥原创的Flask3 Python Web开发 Flask3视频教程&#xff1a; 2025版 Flask3 Python web开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili 模型类(表)不是一成不变的&#xff0c;当你添加了新的模型类&#xff0c;或是在模型类中添加了新的字段&#xff0c;甚至是修改…

Postman CORS 测试完全指南:轻松模拟跨域请求,排查 CORS 相关问题

在使用 Postman 进行 API 测试时&#xff0c;通常不会遇到跨域问题&#xff0c;因为 Postman 是一个独立的客户端应用程序&#xff0c;不同于在浏览器中运行的 JavaScript 代码&#xff0c;它没有同源策略&#xff08;SOP&#xff09;的限制。跨域资源共享&#xff08;CORS&…

【图像处理基石】什么是refocus?

1. Refocus 的定义 Refocus&#xff08;重新对焦&#xff09;是一种通过算法调整图像或视频焦点的技术&#xff0c;允许用户在拍摄后选择焦点&#xff0c;实现类似光场相机的“先拍照后对焦”效果。其核心是通过多视角信息或深度估计&#xff0c;生成不同焦平面的图像&#xff…

kettle从入门到精通 第九十三课 ETL之kettle kettle 调用web service接口5种方法,一文彻底搞懂

场景&#xff1a;群里有小伙伴向我求助如何调用web service接口&#xff0c;趁着周末时间&#xff0c;给兄弟们搞demo。 1、本次使用的web service服务接口地址是http://ws.webxml.com.cn/WebServices/WeatherWS.asmx?opgetSupportCityDataset&#xff0c; 此接口根据用户输入…

电子电气架构 --- 域控架构下,汽车连接器的挑战和变化

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 周末洗了一个澡,换了一身衣服,出了门却不知道去哪儿,不知道去找谁,漫无目的走着,大概这就是成年人最深的孤独吧! 旧人不知我近况,新人不知我过…

[MySQL] 库的操作 表的操作

1.库的操作 1.创建数据库 这里就是一个创建数据库的例子&#xff0c;框内的东西可以不填&#xff0c;因为有默认设置&#xff0c;而这些东西是什么呢&#xff1f; 2.字符集和校验规则 2.1查看字符集校验规则 show variables like ‘character_set_database’; show variable…

Let’s Encrypt 宣布推出短期证书与 IP 地址支持,推动 Web 安全迈向新高度

2025 年 1 月 16 日&#xff0c;全球领先的免费 SSL/TLS 证书颁发机构 Let’s Encrypt 正式宣布两项重大功能更新计划&#xff1a;推出六天有效期证书&#xff08;Short-Lived Certificates&#xff09;及支持以 IP 地址为主体的证书申请。两项功能将于 2025 年起陆续开放&…

十二、Cluster集群

目录 一、集群简介1、现状问题2、集群作用 二、集群结构设计1、集群存储设2、消息通信设计 三、Cluster集群三主三从结构搭建1、redis.conf配置文件可配置项2、配置集群3、链接集群4、命令客户端连接集群并使用 四、集群扩容1、添加节点2、槽位分配3、添加从节点 五、集群缩容1…

Linux进程管理之子进程的创建(fork函数)、子进程与线程的区别、fork函数的简单使用例子、子进程的典型应用场景、父进程等待子进程结束后自己再结束

收尾 进程终止&#xff1a;子进程通过exit()或_exit()终止&#xff0c;父进程通过wait()或waitpid()等待子进程终止&#xff0c;并获取其退出状态。&#xff1f;其实可以考虑在另一篇博文中来写 fork函数讲解 fork函数概述 fork() 是 Linux 中用于创建新进程的系统调用。当…

【AI论文】挑战推理的边界:大型语言模型的数学基准测试

摘要&#xff1a;近年来&#xff0c;大型推理模型的迅猛发展导致现有用于评估数学推理能力的基准测试趋于饱和&#xff0c;这凸显出迫切需要更具挑战性和严谨性的评估框架。为填补这一空白&#xff0c;我们推出了OlymMATH&#xff0c;这是一项全新的奥林匹克级数学基准测试&…

典范硬币系统(Canonical Coin System)→ 贪心算法

【典范硬币系统】 ● 典范硬币系统&#xff08;Canonical Coin System&#xff09;是指使用贪心算法总能得到最少硬币数量解‌的货币面值组合‌。 ● 给定一个硬币系统 &#xff0c;若使其为典范硬币系统&#xff0c;则要求其各相邻面值比例 &#xff0c;及各开区间 内各金额…