C语言静态库、动态库的封装和注意事项

news2024/9/29 11:26:48

1、动态库、静态库介绍

参考博客:《静态库和动态库介绍以及Makefile》;

2、代码目录结构和编译脚本

参考博客:《实际工作开发中C语言工程的目录结构分析》;

3、编写库的流程

(1)明确需求:需求是否合理、需求的使用场景、需求可能遇到的出错情况;
(2)编写设计文档:功能实现的流程图、代码框架;
(3)设计文档的评审:开设计文档的评审会;
(4)编写对外提供的库头文件:拟定库对外的头文件;
(5)头文件的评审;
(6)创建代码的工程目录,编写代码;
(7)能编译出库文件后,编写测试demo程序;
(8)demo程序测试成功,交付库或者和相关模块进行联调;
(9)根据调试情况修改bug;
(10)上传代码,对外提供的头文件要上传到公共目录,采用映射的方式使用头文件;

3、注意事项

3.1、考虑代码规范性

(1)定义变量后注意要初始化,内存申请后先清零;
(2)内存申请后要判断是否成功,指针使用前判断是否为NULL;
(3)一个文件一般只定义一个全局变量,如果需要多个全局变量可以封装成一个结构体;
(4)初始化函数要考虑被重复调用的可能:记录下是否已经初始化成功,初始化成功之后被再次调用则直接返回成功;
(5)对外提供的库头文件要对每个函数和数据类型写清楚注释;库内部使用的函数和结构体只对重要的写注释;
(6)不对本文件之外提供的函数和全局变量用static修饰,定义成局部的;
(7)函数的传参,如果函数内部不需要修改或者不能修改,用const修饰;
(8)函数的大小一般不超过200行,太长的函数考虑拆分;
(9)函数的命名要规范,可以防止函数重名,一般是库的名字开头;比如函数名aaa_xxx:aaa是库的名字,xxx是函数的功能;
(10)尽量不要用数字来表示含义,用枚举来表示则逻辑比较清晰;
(11)函数对传参要做判断,特别是要判断指针是否为NULL;
(12)内存管理:注意谁申请谁释放的原则,不要造成内存泄漏;

3.2、考虑跨平台

(1)需要在不同机器架构间共用的头文件,要考虑32位、64位机器的数据类型所占字节数的差异,不用使用在不同位数机器中有歧义的数据类型;比如指针类型void *:在32位机器中是4字节,在64位机器中是8字节;
(2)如果库中调用linux的标准API,需要考虑这个API在不同的linux版本中是否有差异。可能有些高版本的linux已经不建议使用某些接口,有新的接口来替换旧的接口;
(3)结构体在不同的

3.3、考虑扩展性

(1)数据类型在定义时可保留一些字节;比如在定义结构体时,重要的、将来可能会添加元素的结构体可保留一些字节,将来扩展结构体时去占用这些保留字节,能保证结构体大小不会发生改变;
(2)函数接口保持一定的抽象性;比如将一些操作以函数指针的方式供上层调用,只定义函数指针的格式,将来可以很方便的替换掉函数指针的实体;
(3)定义函数接口时,要保留将来可能会用到但现在用不到的接口,保持架构的完整性。比如定义了库的初始化函数,对应要定义一个库的反初始化函数,虽然这个接口现在用不上,将可能会用到;
(4)库的头文件采用映射的方式,有利于同步;

3.4、考虑库的调试手段

(1)封装打印函数:区分一般打印、调试打印、出错打印,可以通过编译宏或者程序在运行时接收参数的形式,正常运行时关闭一般打印和调试打印,在需要调试时打再打开;
(2)可以将库的打印注册到内核的/proc文件系统中,效果可以参考海思芯片的"/proc/umap"调试手段;
(3)编译库时,Makefile添加"-g"选项,支持gdb调试;
(4)可以考虑错误码,不同的情况返回不同的错误码,便于定位问题;(错误码用枚举不要用纯数字)

3.5、考虑兼容性

(1)对外的头文件,使用基本的、通用的数据类型;比如使用int,不用使用自己封装的Int32等类型;
(2)对外头文件要低的耦合性,不用和其他的库绑定;
(3)C语言写的库,要兼容C++调用;具体参考博客:https://blog.csdn.net/weixin_42031299/article/details/126688788;

3.6、考虑库的交付

(1)在编译库时,把静态库和动态库都编译出来,让使用库的人自行选择;
(2)提供的demo代码,要写一个Readme文档,介绍使用方法;
(3)交互库时,最好以工程文件夹的方式提供,里面编写一个Makefile,让使用者可以一键编译出demo程序。推荐博友有机会熟悉下海思芯片的SDK开发包,里面的文档、脚本写的很详细,几乎就是一键编译;

推荐

给大家推荐一个学校嵌入式知识的网站,博主在大学时候学习嵌入式知识、找工作的时候都在用这个网站,网站里有C语言、Linux等等的笔试题、面试常问问题等等知识,无论是学习基础知识、面试刷题、交流工作经验都是不错的选择。大家一起进步,欢迎留言交流。
链接:学习神器跳转
在这里插入图片描述
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

chatgpt:人工智能的一次突破,如何正确的创建用户及使用

Chatgpt的正确创建及使用 chatgpt最近在国内也开始有声音了,其实早在去年12月初,该网站就已经可以在国外进行使用,而且很快渗透到了国外各行各业各个年龄段 ,最火的当属国外很多学生用它来生成论文,关键是语句通顺&am…

如何开启多个独立Chrome浏览器

一、简介 作为测试或者开发人员,有些情况下会用到 Chrome 浏览器,但有时是同一个 Chrome 浏览器无法为我们提供隔离开的不同环境。这样 我们就需要清理 cache 、切换账号等,降低了我们的工作效率。今天的主题是如何开启多个独立的 Chrome 浏…

LayUI模板引擎渲染数据

前端模板引擎介绍 接上节Spring boot项目开发实战——(LayUI实现前后端数据交换与定义方法渲染数据) 模板引擎能简化开发,极大提高效率,小编之前使用过JSP和Thymeleaf,以及python的jinja2,这些是后端的模…

spring(二)-----------如何注入bean

我们从第三方框架mybatis为引,看看如何往spring中注入一个bean 1、纯mybatis开发生成一个mapper对象 如果不使用spring的情况下,mybatis想生成一个mapper对象大概需要做下面的操作: 假设我们有了一个TMapper接口,此时获取该map…

12款开源数据资产(元数据)管理平台选型分析(三)

如上,是ChatGPT的百度指数和微信指数,继2022年12月上旬技术圈火热之后,因为微软、谷歌等巨头的推广加持,ChatGPT成为全球大众热源的话题。各大媒体都在消费这波舆论红利,打开微信公众号,劈天盖地各种姿势的…

前后端学习

最近和锴哥想搞一下前后端接口的事儿,但是不会,所以打算再学学前后端的基础知识,之后好抄作业,做缝纫机;达哥觉得我浮躁,这次一定要支棱起来;这次开始,面向openai学习。 前后端学习1…

ChatGPT (可能)是怎么炼成的

学习自李宏毅老师的课https://www.youtube.com/watch?ve0aKI2GGZNg 1.学习文字接龙 学习方式 GPT只需要在网上阅读大量的句子,不需要人工标注即可学习到大量句子接龙的知识 然而实际上,“你好”后面可以接的字有很多。实际上,GPT学的就是…

3、Go基础数据类型

目录一、Go数据类型二、字符串三、强制类型转换一、Go数据类型 基础数据类型 类型长度(字节)默认值说明bool1falsebyte10uint8,取值范围[0,255]rune40Unicode Code Point, int32int, uint4或8032 或 64 位,取决于操作系统int8, uint810-128 ~ 127, 0 ~…

Freemarker介绍

2. Freemarker介绍 FreeMarker 是一个用 Java 语言编写的模板引擎,它基于模板来生成文本输出。FreeMarker与 Web 容器无关,即在 Web 运行时,它并不知道 Servlet 或 HTTP。它不仅可以用作表现层的实现技术,而且还可以用于生成 XML…

Python数据结构:概念、栈

1.概念 数据结构是指相互之间存在着一种或多种关系的数据元素的集合和该集合中数据元素之间的关系组成。简单来说,数据结构就是设计数据以何种方式组织并存储在计算机中。 比如:列表、集合与字典等都是一种数据结构。 N.Wirth:“程序数据结构算法’ 2.分类 数据结…

基于comsol软件弯曲单模光纤模拟仿真

在本节中,主要基于实验室实际光纤单模圆柱光纤进行模拟,与comsol案例库文件在分析过程和建模有些差异: 模拟主要通过以下三个步骤进行:模型的几何构建、物理场的添加研究、结构处理分析来进行。 下面是第一步骤:几何…

unity2022.1.8之后版本的新的输入行为控制对象变化

文章目录unity2022.1.8之后版本的新的输入行为控制对象变化怎么导入?如何使用?unity2022.1.8之后版本的新的输入行为控制对象变化 我们先了解大概的逻辑。我们要设置触发行为的方式并且让他和对象的行为绑定,再将行为和对象绑定,…

SpringBoot开发规范部分通用模板+idea配置【项目通用-1】

SpringBoot开发规范通用模板 1 分页插件使用 通过MybatisPlus配置分页插件拦截器 Configuration MapperScan("com.xuecheng.content.mapper") public class MybatisPlusConfig {//定义分页的拦截器Beanpublic MybatisPlusInterceptor getMybatisPlusInterceptor() {…

Pascal版本的 - freopen

参数 filename -- 这是包含要打开的文件的名称的字符串。 mode -- 这是包含文件访问模式的字符串。它包括 - 高级编号模式&说明1个 “r” 打开文件进行读取。该文件必须存在。 2个 “w” 创建一个用于写入的空文件。如果已存在同名文件,则删除其内容并将该文件…

【Java容器(jdk17)】ArrayList深入源码,就是这么简单

ArrayList深入源码一、ArrayList源码解析1. MIXIN 的混入2. 属性说明3. 构造方法4. 其他方法(核心)iterator 和 listIterator 方法add方法remove 方法sort方法其他二、ArrayList 为什么是线程不安全的?体现哪些方面呢?三、ArrayLi…

(day12) 自学Java——集合进阶(双列集合)

目录 1.双列集合特点 Map遍历三种方式 2.HashMap 3.LinkedHashMap 4.TreeMap 5.源码解析 6.可变参数(形参个数可变) 7.Collections 8.综合练习 1.双列集合特点 ①双列集合一次需要存一对数据,分别为键和值 ②键不能重复,值可以重复 ③键和值是一…

全志H616——用C语言的形式操作数据库

sqlite3_open(const char *filename, sqlite3 **ppDb)该例程打开一个指向 SQLite 数据库文件的连接,返回一个用于其他 SQLite 程序的数据库连接对象。sqlite3_close(sqlite3*)该例程关闭之前调用 sqlite3_open() 打开的数据库连接。所有与连接相关的语句都应在连接关…

【Linux】环境变量本地变量

文章目录环境变量基本概念常见环境变量和环境变量相关的命令为什么带./运行我们的可执行程序本地变量环境变量的组织方式环境变量具有全局属性环境变量 基本概念 环境变量(environment variables)一般是指在操作系统中用来指定操作系统运行环境的一些参数 如:我们…

gRPC的简单应用

gRPC的简单应用 gRPC是由开发的一个高性能、通用的开源RPC框架,主要面向移动应用开发且基于HTTP/2协议标准而设计,同时支持大多数流行的编程语言。 官网:https://grpc.io/ 安装protoc 工具 https://protobuf.dev/ 安装Go插件 旧版本直接…

学习HandlerThread

HandlerThread是一个扩展了Thread的类。也就意味着它和普通的Thread类的调用没有什么区别,仍然要调用start()。 如上图所示,扩展后的HandlerThread类有一个Looper和Handler。 关于这一块的知识可以参考一下《关于Handler我们应该知道的知识》 HandlerTh…