dll动态库加载失败导致程序启动报错以及dll库加载失败的常见原因分析与总结

目录

1、问题说明

2、dll库的隐式加载与动态加载

2.1、dll库的隐式加载

2.2、dll库的显式加载

3、使用Process Explorer查看进程加载的dll库信息以及动态加载的dll库有没有加载成功

3.1、使用Process Explorer查看进程加载的dll库信息

3.2、使用Process Explorer查看动态启动的库有没有加载成功

4、dll库加载失败原因详细分析与说明

4.1、dll位数与依赖它的模块位数不一致,导致dll库加载失败

4.2、dll库依赖的库有问题,导致dll加载失败

4.2.1、dll库依赖的库在当前系统中找不到

4.2.2、dll库调用其依赖的库中的接口,但该接口在被依赖的库中找不到

4.3、使用Dependency Walker工具查看dll库的依赖关系以及调用的接口

5、最后


       最近技术群一个朋友遇到一个程序启动异常以及dll库加载失败的问题,找我帮忙看一下,看看是什么原因导致的。其实这个问题不难,只要之前详细看过我博客相关文章的朋友,肯定能很快定位出来的。本文对dll库加载失败导致程序启动报错以及dll库加载失败的常见原因进行总结,以供大家参考。

1、问题说明

       朋友在用VS调试运行他们的程序时,程序中有异常,在VS的输出窗口中也看到了相关打印,如下所示:

​然后接下来,在使用LoadLibraryEx去加载某个dll库时也加载失败了,LoadLibraryEx返回NULL,如下所示:

​可以确定截图中绝对路径中的dll文件是存在的,但还是加载失败了。

       至于为什么要用LoadLibrary而要用扩展接口LoadLibraryEx,并且传入LOAD_WITH_ALTERED_SEARCH_PATH参数,,可以查看我之前写的文章:
查看开源操作系统ReactOS源码,解决dll库动态库加载失败问题(调用LoadLibrary加载失败)icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/129200442

我们在项目中遇到过几次使用LoadLibrary加载dll库失败的问题,dll库的路径是正确的(文件是存在的),但就是加载失败!当时深入研究了一下,参考Reactos开源系统中的regsvr32.exe中加载dll库的源码实现,将LoadLibrary改成了LoadLibraryEx,并传入LOAD_WITH_ALTERED_SEARCH_PATH参数。

2、dll库的隐式加载与动态加载

       程序启动时,系统会给程序分配指定大小的进程空间(虚拟内存空间),系统先将exe主程序依赖的多个dll库加载到进程空间中,然后再将exe主程序文件加载到进程空间中,然后进入main函数,程序开始运行。如果启动过程有dll库加载失败(非动态加载,且会弹出报错提示框),则程序启动终止,程序启动失败!

       程序中引用dll(调用dll中的接口),在dll加载时,可分隐式加载和动态显式加载两种方式。隐式加载是引入dll库对应的lib库,在链接时需要链接到代码中调用的dll的导出接口;

2.1、dll库的隐式加载

       所谓隐式加载,就是在程序中使用#pragma预编译指令引入dll库对应的lib导入库:

#pragma comment( lib, "libcurl.lib")

或者在VS的工程配置中配置导入库:

​对于隐式加载的库,在引入lib导入库之后,并包含dll库的头文件,就可以直接在代码中调用dll库的API接口了。

       隐式加载的dll库,在程序启动时就加载了(上面已经讲了程序启动时dll库的加载流程),如果dll库加载失败,则立即终止exe主程序的启动流程,程序启动失败。

2.2、dll库的显式加载

       dll库的显式加载,是调用LoadLibrary或者LoadLibraryEx去动态地加载dll库。

       对于显式加载的库,需要调用GetProcAddress接口去获取dll库中API接口的函数,然后通过该地址去调用API函数。比如如下的一段代码:

BOOL AutoRegsvr32( LPCTSTR lpszDllPath )
{if ( lpszDllPath == NULL ){return;}// 改用LoadLibraryEx,并使用LOAD_WITH_ALTERED_SEARCH_PATH参数,避免部分系统无法加载到dll的问题。HINSTANCE hInstLib = LoadLibraryEx( lpszDllPath, NULL, LOAD_WITH_ALTERED_SEARCH_PATH );if ( NULL == hInstLib ){return FALSE;}typedef HRESULT (*DllRegisterServer)(void);DllRegisterServer dllRegisterServer = (DllRegisterServer)GetProcAddress( hInstLib, "DllRegisterServer" );if ( dllRegisterServer != NULL ){HRESULT hr = dllRegisterServer();}else{FreeLibrary(hInstLib);return FALSE;}FreeLibrary(hInstLib);return TRUE;
}

       显式加载的dll库,不会在程序启动时加载,而是在代码执行到LoadLibrary或者LoadLibraryEx函数的调用时才会动态的加载。如果dll库加载失败,也不会导致程序启动失败。

       当动态加载dll库失败时,不像通过dll对应的lib导入库隐式调用dll库,不会弹出报错提示框。对于通过dll对应的lib导入库隐式调用dll库加载失败时(一般是在程序启动时),如果依赖的库在当前系统中找不到,则会报出类似如下的错误:

​       如果程序中调用的接口在对应的dll库中找不到(一般是库与库之间的版本不一致导致的),则会报类似如下的错误:

如果是动态加载dll失败,不管是哪种原因,都不会弹出上述报错提示框。


       在这里,给大家重点推荐一下我的几个热门畅销专栏,欢迎订阅:(博客主页还有其他专栏,可以去查看)

专栏1:该精品技术专栏的订阅量已达到520多个,专栏中包含大量项目实战分析案例,有很强的实战参考价值,广受好评!专栏文章持续更新中,预计更新到200篇以上!欢迎订阅!)

C++软件调试与异常排查从入门到精通系列文章汇总icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/125529931

本专栏根据多年C++软件异常排查的项目实践,系统地总结了引发C++软件异常的常见原因以及排查C++软件异常的常用思路与方法,详细讲述了C++软件的调试方法与手段,以图文并茂的方式给出具体的项目问题实战分析实例(很有实战参考价值),带领大家逐步掌握C++软件调试与异常排查的相关技术,适合基础进阶和想做技术提升的相关C++开发人员!

考察一个开发人员的水平,一是看其编码及设计能力,二是要看其软件调试能力!所以软件调试能力(排查软件异常的能力)很重要,必须重视起来!能解决一般人解决不了的问题,既能提升个人能力及价值,也能体现对团队及公司的贡献!

专栏中的文章都是通过项目实战总结出来的,包含大量项目问题实战分析案例,有很强的实战参考价值!专栏文章还在持续更新中,预计文章篇数能更新到200篇以上!

专栏2:(本专栏涵盖了C++多方面的内容,是当前重点打造的专栏,订阅量已达160多个,专栏文章已经更新到400多篇,持续更新中...)

C/C++实战进阶(专栏文章,持续更新中...)icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/category_11931267.html

以多年的开发实战为基础,总结并讲解一些的C/C++基础与项目实战进阶内容,以图文并茂的方式对相关知识点进行详细地展开与阐述!专栏涉及了C/C++领域多个方面的内容,包括C++基础及编程要点(模版泛型编程、STL容器及算法函数的使用等)、数据结构与算法、C++11及以上新特性(不仅看开源代码会用到,日常编码中也会用到部分新特性,面试时也会涉及到)、常用C++开源库的介绍与使用、代码分享(调用系统API、使用开源库)、常用编程技术(动态库、多线程、多进程、数据库及网络编程等)、软件UI编程(Win32/duilib/QT/MFC)、C++软件调试技术(排查软件异常的手段与方法、分析C++软件异常的基础知识、常用软件分析工具使用、实战问题分析案例等)、设计模式、网络基础知识与网络问题分析进阶内容等。

专栏3:  

C++常用软件分析工具从入门到精通案例集锦汇总(专栏文章,持续更新中...)icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/131405795

常用的C++软件辅助分析工具有SPY++、PE工具、Dependency Walker、GDIView、Process Explorer、Process Monitor、API Monitor、Clumsy、Windbg、IDA Pro等,本专栏详细介绍如何使用这些工具去巧妙地分析和解决日常工作中遇到的问题,很有实战参考价值!

专栏4:   

VC++常用功能开发汇总(专栏文章,持续更新中...)icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/124272585

将10多年C++开发实践中常用的功能,以高质量的代码展现出来。这些常用的高质量规范代码,可以直接拿到项目中使用,能有效地解决软件开发过程中遇到的问题。

专栏5: 

C++ 软件开发从入门到精通(专栏文章,持续更新中...)icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/category_12695902.html

根据多年C++软件开发实践,详细地总结了C/C++软件开发相关技术实现细节,分享了大量的实战案例,很有实战参考价值。


3、使用Process Explorer查看进程加载的dll库信息以及动态加载的dll库有没有加载成功

       我们可以使用Process Explorer工具查看程序加载的哪些dll库,以及这些dll库的详细信息。还可以查看动态加载的库有没有加载成功。

3.1、使用Process Explorer查看进程加载的dll库信息

       可以使用Process Explorer工具查看进程加载的dll库列表,在dll列表中可以看到加载的dll库的详细信息。打开Process Explorer后:

​在进程列表中找到目标进程,点击选中该进程,下方就会显示本进程加载的dll库列表,在dll列表中可以查看这些库的详细信息,比如加载的dll库的路径、库的版本等。通过库的路径或版本,确定是否加载了正确版本的dll库。

       注意,如果是在电脑上第一次启动Process Explorer,需要在第一次启动该工具时点击工具栏中的“View DLLs”图标按钮:

​才会显示目标进程加载的库列表。默认情况下显示的是句柄占用信息。

       前段时间有粉丝朋友告诉我,微软官方最近更新升级了Process Explorer工具(微软官方提供的工具),软件的主界面发生了变化,询问新版本如何查看进程占用的dll列表,于是我到微软官网下载了该工具。其实很简单,要查看进程加载的dll列表,直接点击下方的“DLLs”标签页即可

3.2、使用Process Explorer查看动态启动的库有没有加载成功

       上面我们说了,隐式加载的dll,会在程序启动时加载,如果dll库加载失败,则程序的启动会被立即终止;而动态加载的dll,不会在程序启动时加载,不会影响程序的启动。所以,程序启动起来后,只会存在动态加载的dll库加载失败,隐式加载的dll库肯定都成功加载起来了,否则程序会启动失败。

       如何判断动态启动的库有没有加载成功呢?其实很简单,只要看dll库列表中有没有这个给dll就知道了。

       那程序运行时,我们怎么感知动态加载的dll库没有加载成功呢?只能根据程序业务来判断,即执行到动态库相关的业务,没有正常执行时,则可能时动态加载的dll库加载失败导致的。在我们的项目中,底层若干业务模块就是动态加载的,上层产品根据自己的需要去选择性加载部分业务模块即可,我们在产品开发联调时经常遇到dll库动态加载失败的问题,所以我在处理这方面问题时比较有经验一点。

4、dll库加载失败原因详细分析与说明

       这里我们讨论一下dll加载失败的原因,不管是通过dll库的导入库隐式加载的,还是通过调用LoadLibrary或者LoadLibraryEx动态加载的,原因都是一样的,主要有以下三个原因:

1)问题dll的位数与依赖它的模块位数不一致;
2)问题dll依赖的库在当前系统中找不到;
3)问题dll调用底层库的接口,在底层库中找不到。

       dll隐式加载失败与dll动态加载失败不同的地方在于现象有所有不同。dll隐式加载失败时(一般在程序启动时),会弹出报错提示框;而dll动态加载失败时,则不会弹出报错提示框,开发人员只能根据业务异常去感知。至于弹出什么样的报错提示框,上面我已经讲过了,在此不再赘述。

4.1、dll位数与依赖它的模块位数不一致,导致dll库加载失败

        如果dll库加载失败,我们可以确认一下dll库的位数是否与依赖该dll的模块(依赖该dll的模块可能是其他dll,也可能是exe主程序)的位数一致,因为32位dll库是不能和64位模块混用的,位数必须要一致才能使用。当然这种情况在项目中比较少见,这只是一种可能的原因,实际项目中比较少,一般不用关注这种情况。比如32位的exe主程序是不能加载使用64位dll库的,如果将位数不同的模块混在一起,程序启动时回报0xC000007B错误,如下所示:

​       什么情况下会出现模块与模块之间的位数不一致呢?比如我们在日常开发调试过程中遇到的一个问题,发布32位程序时,要将相关C++运行时库一起打包发布,结果错误拷贝了64位运行时库(现在大家用的基本都是Win10系统,系统中既有32位的库,也有64位的库),导致程序启动报0xC000007B错误。关于0xC000007B错误的实战分析案例,可以查看我之前写的文章:

使用Dependency Walker和Process Explorer排查程序启动时缺少ucrtbase.dll等运行时库以及报0xC000007B错误icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/131505299       至于如何查看dll、exe等二进制文件的位数,可以用dumpbin.exe或者PE查看工具EXE Explorer,相关查看方法可以查看我的文章:
使用Dumpbin工具查看C++二进制文件的位数、时间戳及dll库的依赖关系icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/140153214如何查看exe和dll等二进制文件的生成时间(时间戳)和位数(32位/64位)icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/140043291

4.2、dll库依赖的库有问题,导致dll加载失败

       dll库依赖的库有问题(当前问题dll依赖的更下层的库),主要有两种情况:

1)dll库依赖的库在当前系统中找不到;
2)dll库中调用了其依赖库中的接口,但该接口在被依赖的库中找不到。

4.2.1、dll库依赖的库在当前系统中找不到

        如果dll库依赖的库在当前系统中找不到,则该dll库会加载失败(加载dll之前,会先将当前dll依赖的库先加载起来,依赖的库都加载起来后,才会去加载当前的dll)。

        如果该dll库是隐式加载的,则一般在程序启动时加载,会报类似这个错误:

​这个问题在我们项目中也经常出现,引发这类问题可能有两个典型的场景:

1)一个场景是,在发布程序时,没有将程序模块依赖的C/C++运行时dll库一起发布,而这些运行时库在其他电脑上可能没有,所以我们在发布程序时要将相关运行时库带上。
2)另一个场景是,底层模块因为重构或者业务需要,新增了一个dll模块,没有通知上层产品,导致上层软件产品在打包时没有将新增的dll库打包进去。

4.2.2、dll库调用其依赖的库中的接口,但该接口在被依赖的库中找不到

       如果dll库调用其依赖的库中的接口在被依赖的库中找不到,则该dll库也会加载失败。如果该dll库是隐式加载的,则在程序启动时会报类似这个错误:

​出现这类问题,一般是库与库的版本不一致导致的。在编译时引用了正确的dll库的导入库lib,编译是没问题的,但在程序打包时拷贝的是老版本的dll库,导致程序运行时出现调用的接口在底层的库中找不到。

       至于为什么在依赖库中找不到接口呢?可能找不到的接口,在新版本库中被删除了,或者接口名称被修改了,亦或是接口的参数被修改了(比如增加了参数、删除了参数或者修改了参数类型)。关于接口参数被修改导致找不到接口的实例:

​可以查看我之前写的文章:

使用Process Explorer和Dependency Walker排查dll动态库没法调试的问题(dll库加载失败导致没法动态调试)icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/140803687        还有一种情况就是,程序中调用了Windows系统的系统dll库中的某个接口,该接口只在高版本的Windows系统中才有,比如只在Win10系统才有,Win7系统中的系统库中没有这个接口,这样将程序拷贝到Win7系统中运行就会报接口找不到的错误,因为Win7系统库中没有这个接口。

4.3、在某些系统上会出现LoadLibrary加载dll库失败的问题,需要使用LoadLibraryEx接口

       我们在项目中遇到过几次使用LoadLibrary加载dll库失败的问题,dll库的路径是正确的(文件是存在的),但就是加载失败!当时深入研究了一下,参考Reactos开源系统中的regsvr32.exe中加载dll库的源码实现,将LoadLibrary改成了LoadLibraryEx,并传入LOAD_WITH_ALTERED_SEARCH_PATH参数。

4.4、使用Dependency Walker工具查看dll库的依赖关系以及调用的接口

       对于上述dll依赖的库问题,无论是依赖的库找不到,还是调用的接口在下层库中找不到,直接使用Dependency Walker工具打开这个问题dll查看即可找到问题。

       关于使用Dependency Walker查看库的依赖关系排查问题的项目问题实战分析案例,可以查看我之前写的文章:
使用Process Explorer和Dependency Walker排查C++程序中dll库动态加载失败问题icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/140731158使用Dependency Walker和Process Explorer排查瑞芯微工具软件RKPQTool.exe启动报错的问题icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/140731614使用Process Explorer和Dependency Walker排查dll动态库没法调试的问题(dll库加载失败导致没法动态调试)icon-default.png?t=O83Ahttps://blog.csdn.net/chenlycly/article/details/140803687

5、最后

       本文根据多年的项目实战经验,对dll库加载失败导致程序启动报错以及dll库加载失败的常见原因进行了总结,有一定的实战参考价值。本文的内容不仅对开发新人有用,这些内容很多有若干年开发经验的人可能也不太了解(他们对系统特性不太了解),对这些人可能也很有用。

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

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

相关文章

JAVA开源项目 旅游管理系统 计算机毕业设计

本文项目编号 T 063 ,文末自助获取源码 \color{red}{T063,文末自助获取源码} T063,文末自助获取源码 目录 一、系统介绍二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景5.2 国内外研究现状5.3 可行性分析5.4 用例设计 六、核…

SpringBoot项目 | 瑞吉外卖 | 短信发送验证码功能改为免费的邮箱发送验证码功能 | 代码实现

0.前情提要 之前的po已经说了单独的邮箱验证码发送功能怎么实现: https://blog.csdn.net/qq_61551948/article/details/142641495 这篇说下如何把该功能整合到瑞吉项目里面,也就是把原先项目里的短信发送验证码的功能改掉,改为邮箱发送验证…

前端学习第一天笔记 HTML5 CSS初学以及VSCODE中的常用快捷键

前端学习笔记 VsCode常用快捷键列表HTML5标题标签标签之段落、换行、水平线标签之图片图片路径详解标签之超文本链接标签之文本列表标签之有序列表列表标签之无序列表标签之表格表格之合并单元格Form表单表单元素文本框 密码框 块元素与行内元素(内联元素&#xff0…

阿里云部署1Panel(失败版)

官网脚本部署不成功 这个不怪1panel,这个是阿里Linux 拉不到docker的下载源,懒得思考 正常部署直接打开官网 https://1panel.cn/docs/installation/online_installation/ 但是我使用的阿里云os(Alibaba Cloud Linux 3.2104 LTS 64位) 我执行不管用啊装不上docker 很烦 curl -s…

力扣 简单 110.平衡二叉树

文章目录 题目介绍解法 题目介绍 解法 平衡二叉树:任意节点的左子树和右子树的高度之差的绝对值不超过 1 //利用递归方法自顶向下判断以每个节点为根节点的左右子树的最大深度是否大于1 class Solution {public boolean isBalanced(TreeNode root) {if(root null){return tr…

SPARK调优:AQE特性(含脑图总结)

学完AQE需要能够回答如下的几个问题: 什么是AQE?AQE的实现原理是什么?AQE的特性有哪些?使用什么参数实现?AQE每个特性可以解决什么问题?什么问题是AQE不能解决的 HL:学习脑图如下 SparkAQE是spa…

【2024版本】Mac/Windows IDEA安装教程

IDEA 2024版本真的很强大,此外JDK发布了最新稳定版 JDK21 ,只有新版本支持JDK 21、JDK22。原来数据库插件不支持redis等一些NoSql的数据库的连接,如果要使用需要自己单独装收费的插件。直接打开idea就很吃内存了,再打开其他一大堆…

案例-猜数字游戏

文章目录 效果展示初始画面演示视频 代码区 效果展示 初始画面 演示视频 猜数字游戏 代码区 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width,…

C++游戏开发详解:从入门到实践

目录 引言 使用C进行游戏开发的优势 常用的C游戏引擎和工具 C游戏引擎比较 开发工具 C游戏开发核心概念与代码示例 面向对象编程&#xff08;OOP&#xff09; 封装 继承 多态 内存管理 手动内存管理 智能指针 内存池 并发编程 多线程 同步机制 游戏开发流程 …

【python】追加写入excel

输出文件运行前&#xff08;有两张表&#xff0c;“表1”和“Sheet1”&#xff09;&#xff1a; 目录 一&#xff1a;写入单表&#xff08;删除所有旧工作表&#xff0c;写入新表&#xff09;二&#xff1a;写入多表&#xff08;删除所有旧工作表&#xff0c;写入新表&#x…

docker环境下配置cerbot获取免费ssl证书并自动续期

文章目录 实践场景了解certbot查看nginx的映射情况操作目标配置nginx配置的ssl证书设置自动续签 实践场景 本人使用docker部署了一个nginx容器&#xff0c;通过容器卷&#xff0c;实现本地html&#xff0c;ssl&#xff0c;conf和ngiinx容器映射的&#xff0c; 经常需要手动部署…

【GEE学习第三期】GEE常用函数总结

【GEE学习第三期】GEE常用函数总结 数据统计类ee.List.sequence函数 图像处理类ee.Geometry类‌defaultVisualizationVis函数 数据输入输出数值与绘图导出影像 参考 数据统计类 ee.List.sequence函数 用法如下&#xff1a; ee.List.sequence &#xff08;开始&#xff0c;结…

【Spring】@RequestMapping、@RestController和Postman

文章目录 1.RequestMapping 注解介绍2. RequestMapping 使用3. RequestMapping 是 GET 还是 POST 请求&#xff1f;GET 请求POST 请求指定 GET/POST 方法类型 2. Postman 介绍1. 创建请求2. 传参介绍1. 普通传参2. form-data3. x-www-form-urlencoded form 表单&#xff0c;对应…

PWM 模式

一、介绍 PWM&#xff08;脉宽调制&#xff0c;Pulse-width modulation&#xff09;是一种通过调节脉冲信号的宽度来控制电能输出的方法。PWM是一种方波信号&#xff0c;通常在电子和电气工程中用于调节功率输送&#xff0c;控制电机速度&#xff0c;调节LED亮度&#xff0c;以…

【CSS Tricks】css动画详解

目录 引言一、动画关键帧序列二、动画各属性拆解1. animation-name2. animation-duration3. animation-delay3.1 设置delay为正值3.2 设置delay为负值 4. animation-direction5. animation-iteration-count6. animation-fill-mode7. animation-play-state8. animation-timing-f…

MySQL的驱动安装

1、下载并安装MySQL 下载地址&#xff1a; 建议在下列框中选择LTS长期支持版本&#xff0c;下载对应的MSI安装文件。 安装完成后&#xff0c;将MySQL的环境bin路径添加到环境变量中。 可以运行MySQL Configurator进行配置&#xff0c;主要设置密码&#xff0c;并初始化。其余…

网络基础 【HTTP】

&#x1f493;博主CSDN主页:麻辣韭菜&#x1f493;   ⏩专栏分类&#xff1a;Linux初窥门径⏪   &#x1f69a;代码仓库:Linux代码练习&#x1f69a; &#x1f4bb;操作环境&#xff1a; CentOS 7.6 华为云远程服务器 &#x1f339;关注我&#x1faf5;带你学习更多Linux知识…

JVM Class类文件结构

国庆节快乐 2024年10月2日17:49:22 目录 前言 magic 数 文件版本 使用JClassLib观察class文件 一般信息 接口 常量池 字段 方法 常量池计数器 常量池 类型 CONSTANT_Methodref_info CONSTANT_Class_info 类型结构总表 访问标志 类索引, …

通信协议感悟

本文结合个人所学&#xff0c;简要讲述SPI&#xff0c;I2C&#xff0c;UART通信的特点&#xff0c;限制。 1.同步通信 UART&#xff0c;SPI&#xff0c;I2C三种串行通讯方式&#xff0c;SPI功能引脚为CS&#xff0c;CLK&#xff0c;MOSI&#xff0c;MISO&#xff1b;I2C功能引…

【api连接ChatGPT的最简单方式】

通过api连接ChatGPT的最简单方式 建立client 其中base_url为代理&#xff0c;若连接官网可省略&#xff1b;配置环境变量 from openai import OpenAI client OpenAI(base_url"https://api.chatanywhere.tech/v1" )或给出api和base_url client OpenAI(api_key&…