一.1 库函数前面给大家介绍了系统调用,系统调用是内核直接向应用层提供的应用编程接口,譬如open、write、read、close等,关于这些系统调用后面会给大家进行详细介绍。编写应用程序除了使用系统调用之外,我们还可以使用库函数,本小节来聊一聊库函数。
库函数也就是C语言库函数,C语言库是应用层使用的一套函数库,在Linux下,通常以动态(.so)库文件的形式提供,存放在根文件系统/lib目录下,C语言库函数构建于系统调用之上,也就是说库函数其实是由系统调用封装而来的,当然也不能完全这么说,原因在于有些库函数并不调用任何系统调用,譬如一些字符串处理函数strlen()、strcat()、memcpy()、memset()、strchr()等等;而有些库函数则会使用系统调用来帮它完成实际的操作,譬如库函数fopen内部调用了系统调用open()来帮它打开文件、库函数fread()就利用了系统调用read()来完成读文件操作、fwrite()就利用了系统调用write()来完成写文件操作。
Linux系统内核提供了一系列的系统调用供应用层使用,我们直接使用系统调用就可以了呀,那为何还要设计出库函数呢?事实上,有些系统调用使用起来并不是很方便,于是就出现了C语言库,这些C语言库函数的设计是为了提供比底层系统调用更为方便、更为好用、且更具有可移植性的调用接口。
来看一看它们之间的区别:
l 库函数是属于应用层,而系统调用是内核提供给应用层的编程接口,属于系统内核的一部分;
l 库函数运行在用户空间,调用系统调用会由用户空间(用户态)陷入到内核空间(内核态);
l 库函数通常是有缓存的,而系统调用是无缓存的,所以在性能、效率上,库函数通常要优于系统调用;
l 可移植性:库函数相比于系统调用具有更好的可移植性,通常对于不同的操作系统,其内核向应用层提供的系统调用往往都是不同,譬如系统调用的定义、功能、参数列表、返回值等往往都是不一样的;而对于C语言库函数来说,由于很多操作系统都实现了C语言库,C语言库在不同的操作系统之间其接口定义几乎是一样的,所以库函数在不同操作系统之间相比于系统调用具有更好的可移植性。
以上便上它们之间一个大致的区别,从实现者的角度来看,系统调用与库函数之间有根本的区别,但从用户使用角度来看,其区别并不重要,它们都是C语言函数。在实际应用编程中,库函数和系统调用都会使用到,所以对于我们来说,直接把它们当做是C函数即可,知道你自己调用的函数是系统调用还是库函数即可,不用太过于区分它们之间的差别。
所以应用编程简单点来说就是:开发Linux应用程序,通过调用内核提供的系统调用或使用C库函数来开发具有相应功能的应用程序。
一.2 标准C语言函数库在Linux系统下,使用的C语言库为GNU C语言函数库(也叫作glibc,其网址为http://www.gnu.org/software/libc/),作为Linux下的标准C语言函数库。
进入到http://www.gnu.org/software/libc/网址,如下所示:
图 1.3.1 glibc官网
点击上面的Sources选项可以查看它的源码实现:
图 1.3.2 获取源码的方式
glibc源码的获取方式很简单,直接直接从git仓库下载,也可以通过ftp下载,如果大家有兴趣、或者想要了解某一个库函数它的具体实现,那么就可以获取到它源码来进行分析,好了,这里就不再多说了!
确定Linux系统的glibc版本
前面提到过了,C语言库是以动态库文件的形式提供的,通常存放在/lib目录,它的命名方式通常是libc.so.6,不过这个是一个软链接文件,它会链接到真正的库文件。
进入到Ubuntu系统的/lib目录下,笔者使用的Ubuntu版本为16.04,在我的/lib目录下并没有发现libc.so.6这个文件,其实是在/lib/x86_64-linux-gnu目录下,进入到该目录:
图 1.3.3 libc.so.6文件
可以看到libc.so.6链接到了libc-2.23.so库文件,2.23表示的就是这个glibc库的版本号为2.23。除此之外,我们还可以直接运行该共享库来获取到它的信息,如下所示:
图 1.3.4 确定glibc版本号
从打印信息可以看到,笔者所使用的Ubuntu系统对应的glibc版本号为2.23。
一.3 main函数对学习过C语言编程的读者来说,譬如单片机编程、Windows应用编程等,main函数想必大家再熟悉不过了,很多编程开发都是以main函数作为程序的入口函数,同样在Linux应用程序中,main函数也是作为应用程序的入口函数存在,main函数的形参一般会有两种写法,如果执行应用程序无需传参,则可以写成如下形式:
示例代码 1.4.1 main函数写法之无传参
- int main(void)
- {
- /* 代码 */
- }
如果在执行应用程序的时候需要向应用程序传递参数,则写法如下:
示例代码 1.4.2 main函数写法之有传参
- int main(int argc, char **argv)
- {
- /* 代码 */
- }
argc形参表示传入参数的个数,包括应用程序自身路径和程序名,譬如运行当前目录下的hello可执行文件,并且传入参数,如下所示:
./hello 112233
那么此时参数个数为2,并且这些参数都是作为字符串的形式传递给main函数:
argv[0]等于"./hello"
argv[1]等于"112233"
有传参时main函数的写法并不只有这一种,只是这种写法最常用,对于其它的写法,后面学习过程中如果遇到了再给大家进行讲解,这里暂时先不去管。
一.4 本书使用的开发环境对于编程开发,大家可能都比较关心开发环境的问题,譬如本书将使用什么IDE编写应用程序之类的问题,本小节将对开发环境的问题进行一个简单介绍。
首先,本书分为三篇内容,在前言部分已经向各位读者介绍了本书的内容规划,分为入门篇、提高篇以及进阶篇。因为是Linux应用编程,所以本书将会在Linux操作系统环境下编写示例代码,为了同正点原子的其它开发文档保持一致,选择Ubuntu操作系统,推荐大家使用16.04或14.04版本的Ubuntu系统,个人觉得这两个版本比较稳定;除了使用Ubuntu系统外,大家还可以选择其它Linux发行版,譬如CentOS、Redhat(红帽)等。
在Linux操作系统下,也有很多比较好用的IDE软件,可以帮助我们更为轻松的进行软件开发,譬如Eclipse、vscode等,如果你会使用Eclipse,可以在Ubuntu系统下安装Eclipse进行Linux应用开发,1.5.1小节介绍了如何在Ubuntu系统下安装Eclipse;如果你不会使用Eclipse,那就建议你使用vscode,同样本书也是使用vscode,与正点原子的驱动开发文档保持一致;vscode的安装方法和使用方法本书不再介绍,相信大家在自己的Ubuntu系统下已经安装好了,如果没有安装的,直接参考“开发板光盘资料A-基础资料/【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.5.pdf”文档第四章的4.5小节内容进行安装即可!
Tips:注意,本文档并不介绍Eclipse IDE的使用方法,笔者也用的不多,只是告诉大家可以在Ubuntu使用这个IDE,并且后面给出了安装方法,如果你以前经常使用它开发软件,你可以尝试使用它在Ubuntu下开发Linux应用程序,如果你没用过就不要去碰了。用什么IDE都不重要,哪怕你直接使用vi编写程序都可以,我们的重点是学习应用编程!而不是学习IDE怎么用。
vscode是一个代码编辑器,提供了很多好用的插件,譬如语法检测、高亮显示、智能补全等,大家可以根据自己的选择安装插件,在驱动开发指南文档中也都有介绍。
入门篇章节内容中,主要向大家介绍了Linux应用编程所涉及到的各个知识点以及各种系统调用和库函数,本部分内容所编写的示例代码,均是在Ubuntu系统下进行测试、验证,也就是在Ubuntu系统下执行测试程序以验证所学知识内容。所以在该部分内容中,本书使用vscode+gcc的方式开发应用程序,vscode作为代码编辑器、gcc作为编译器,使用vscode编写完代码之后再用gcc编译源码,得到可在PC端Ubuntu系统下运行的可执行文件。gcc的使用方法在驱动开发指南中有介绍,参考“开发板光盘资料A-基础资料/【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.5.pdf”文档第三章的内容。
提高篇章节内容中,将会以正点原子ALPHA/Mini I.MX6U开发板为例,向大家介绍如何编写应用程序控制开发板上的各种硬件外设,所以编写的应用程序是需要在开发板上运行的,ALPHA/Mini I.MX6U开发板是ARM架构硬件平台,所以源码需要使用交叉编译工具(ARM架构下的gcc编译器)进行编译,得到可在开发板上运行的可执行文件。所以在第二篇章节内容中,本书使用vscode+ARM gcc(交叉编译工具)的方式开发应用程序。
入门篇和提高篇章节内容中所编写的示例代码只有单个.c源文件,但是对于一个真正的Linux应用程序项目来说,其原文件的个数肯定不止一个,可能有几十上百个,这个时候可能就需要用到Makefile来组织、管理工程源文件。直接通过Makefile来组织工程,对于一个大型软件工程来说通常是比较困难的,需要程序员对Makefile能够熟练使用,譬如Linux内核源码、U-Boot源码等都是直接使用Makefile进行管理。对于一般程序员来说,很难做到熟练使用Makefile。在第三篇(进阶篇)章节内容中,会向大家介绍cmake工具,使用cmake来管理我们的源码工程,相比于直接使用Makefile,使用cmake会打打降低难度,提高我们的开发效率!
进阶篇章节内容中,会给大家介绍一些Linux应用编程实战小项目,通过学习进一步提升Linux应用编程能力,本部内容将会使用cmake+vscode进行开发,其实cmake在实际的项目当中用的还是比较多的,譬如很多的开源软件都会选择使用cmake来构建、配置工程源码,学习了之后,相信大家在今后的工作当中也会用到。
关于本小节的内容到这里就结束了,下一章开始将正式进入到应用编程学习之路,大家加油!