版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
《嵌入式系统原理与开发教程》Linux系统应用程序开发基础主讲人:赖树明东莞理工学院01Linux系统程序设计基础03Linux系统Makefile使用04Linux文件IO编程05课程作业任务02Linux静态库和动态库01Linux系统程序设计基础Linux程序设计概述GCC编译器学习Linux系统标准main函数Linux系统程序设计基础Linux程序设计概述01在Windows系统中,开发工具多以集成开发环境IDE的形式展现给最终用户。例如,VS2019集成了编辑器,宏汇编ml,C/C++编译器cl,资源编译器rc,调试器,文档生成工具,nmake。在Linux系统中,开发工具被切割成一个个独立的小工具,各自处理不同的问题。编辑器(emacs,vim,gedit)用来进行编辑程序调试器(gdb)用来调试程序编译器(GCC)用来编译和链接程序本节重点讲解GCC使用方法,这是踏入Linux编程世界的第一步,要求必须掌握。Linux历史Linux程序设计概述Linux系统程序设计基础GCC编译器01GNU是一个计划或者叫运动,在GNU计划开发了很多的工作和项目,包括核心的gcc和glibc。Linux系统下的gcc(GNUCCompiler)是GNU推出的功能强大、性能优越的多平台编译器。gcc是可以在多种硬体平台上编译出可执行程序的超级编译器,其执行效率与一般的编译器相比平均效率要高20%~30%。gcc编译器能将C、C++语言源程序、汇程式化序和目标程序编译、连接成可执行文件。在Linux系统中,可执行文件没有统一的后缀,系统从文件的属性来区分可执行文件和不可执行文件。而gcc则通过后缀来区别输入文件的类别。Linux历史GCC编译器介绍Linux系统程序设计基础GCC编译器学习01.c为后缀的文件,C语言源代码文件;
.a为后缀的文件,是由目标文件构成的档案库文件;
.C或.cc或.cxx为后缀的文件,是C++源代码文件;
.h为后缀的文件,是程序所包含的头文件;
.i为后缀的文件,是已经预处理过的C源代码文件;
.ii为后缀的文件,是已经预处理过的C++源代码文件;
.m为后缀的文件,是Objective-C源代码文件;
.o为后缀的文件,是编译后的目标文件;
.s为后缀的文件,是汇编语言源代码文件;
.S为后缀的文件,是经过预编译的汇编语言源代码文件。Linux历史GCC支持的文件扩展名Linux系统程序设计基础GCC编译器学习01C语言从源代码到可执行文件需要经历四个相互关联的步骤,如下所示:GCC编译程序流程预处理(Preprocessing)源代码文件中以#开头的代码进行处理,如#include,#define,#if等),生成不带#开头代码的源中间文件编译(Compilation)把预处理的文件编译生成以.s为后缀的汇编代码文件。汇编(Assembly)把上一步得到的汇编文件使用汇编器进行编译,生成以.o为后缀的二进制目标文件,但是这时的文件还不能执行。链接(Linking)把所有的.o文件使用链接器进行链接,整合成一个可执行程序。Linux系统程序设计基础GCC编译器学习01gcc常用选项选项含义-c只编译不链接,源文件编译成目标代码(.o),默认生成目标文件“.o”-S(注意是大写S)生成汇编代码-E只进行预编译,不做其他处理-g在可执行程序中包含标准调试信息-o<输出目标文件名>指定输出文件名-O<0~3优化级别>代码优化选项,-ON:指定优化级别,0<=N<=3,默认为1-v打印出编译内部编译各过程的命令行信息和编译器的版本-Wall打开所有警告-Werror把所有警告当成错误处理-w关闭所有警告-static链接静态库,默认情况下gcc只链接动态库(静态库:以.a为扩展名,动态库:带.so扩展名的库文件)静态链接:生成的可执行程序包含所有调用的函数实现代码动态链接:生成的可执行程序是运行时才在so动态库中搜索其调用的函数的。-I<头文件路径>指定编译时包含的头文件路径-L<库文件路径>指定链接时第三库文件的路径-l
<库文件名>(L的小字字母)指定链接时库文件的名称
在Linux系统中库文件命名lib库名.so[*]或lib库名.a,指定库:-l库名Linux系统程序设计基础GCC编译器学习01main.c源码清单:#include<stdio.h>externintadd(int,int);externintsub(int,int);intmain(void){printf(“1+2:%d\r\n",add(1,2));printf(“20-10:%d\r\n",sub(20,10));}gcc编译示例add.c源码清单:intadd(inta,intb){returna+b;}sub.c源码清单:intsub(inta,intb){returna-b;}编译过程:分别编译三个文件为二进制目标文件gcc-Emain.c-omain.igcc-Smain.i-omain.sgcc-cmain.s-omain.ogcc-Eadd.c-oadd.igcc-Sadd.i-oadd.sgcc-cadd.s-oadd.ogcc-Esub.c-osub.igcc-Ssub.i-osub.sgcc-csub.s-osub.o链接:gccmain.osub.oadd.o-omain运行:./maingcc编译器也支持一次编译生成可执行程序,如上面三个c程序,执行以下指令:gccmain.csub.cadd.c-omain可以直接生成可执行程序main。Linux系统程序设计基础Linux标准main函数01在Linux系统或Windows系统中,启动应用程序时可以给程序传递参数,例如:“ls-l/home”中,“ls”是程序名,
“-l”是参数1,“/home”是参数2,如有更多依次类推。程序要接收启动的参数,由需要通过main函数来完成。示例:以下app-arg.c把命令行中输入的参数数量及每个参数全部输出。app-arg.c源码清单:#include<stdio.h>intmain(intargc,char*argv[]){ inti=0; printf("argc:%d\r\n",argc); for(i=0;i<argc;i++){ printf("argv[%d]:%s\r\n",i,argv[i]); }}系统级标准main函数说明:argc记录了启动程序时参数个数,程序名本身算一个参数;argv是一个char*类型的数组,每个元素保存了命令行为每个字符串参数地址,通过argv的下标可以取得启动程序时传递给程序的参数。编译:$gccapp-arg.c-oapp-arg运行测试:$./app-argxyd123
argc:4argv[0]:./app-argargv[1]:xydargv[2]:123argv[3]:结果分析:以空格分隔的,每个字符串是一个参数,分别保存在argv数组中。02Linux静态库和动态库静态库和动态库相关编译选项静态库的创建及使动态库的创建及使动态库与静态库的比较Linux静态库和动态库概述02函数库是能提供一系列功能函数给第三方程序调用的文件,可分为静态库和动态库(也称为共享库),在使用上对应静态链接和动态链接。编译链接程序时,链接器会根据命令选项采用动态库还是静态库中的函数。函数库定义动态库命名:
libname.so.x.y.zlib:固定名称前缀;name:真正的库名称;
so:固定后缀x:主版本号;
y:次版本号;z:发行版本号
,其中.x.y.z是可选的静态库命名:
libname.a库命名规则
把功能函数写到单独的C文件中,使用编译器封装成.a文件,第三方程序要调用其中的函数时通过编译选项-static-l库名
形式把调用的函数对应代码复制一份到当前程序中,运行时候就再依赖静态库文件。静态库使用原理把功能函数写到单独的C文件中,使用编译器封装成libname.so.x.y.z动态库文件,第三方程序要调用其中的函数时通过编译选项-l库名
形式来找到库,记录函数符号,并没有复制代码到当前程序中,而是运行时才加载动态库文件到,找到函数执行。动态库使用原理Linux静态库和动态库静态库和动态库相关编译选项02如果gcc编译命令中指定该选项,表示代码中使用到来自第三方库的函数必须存在静态库,不能是动态库,否则链接失败。-static选项如果gcc编译指令中指定该选项,表示优先使用动态库链接,即代码中使用来自第三方库的函数,如果同时存在动态库和静态库,会优先使用动态库链接,如果只存在静态库,才会使用静态库文件链接。-shared选项如果gcc编译指令中指定该选项,表示要生成相对地址位置无关的目标代码,通常使用gcc的-shared选项把生成位置无关目标文件编译成动态库文件。-fPIC选项在使用第三方库时,如果没有把第三方库文件复制到系统存放库文件的标准路径中,则要在编译阶段使用-L选项来指定第三方库的存放位置。而-I后面跟着的是库文件存放路径,后面可以加空格,也可以不加空格直接写库文件路径,路径可以是相对路径,也可以使用绝对路径。-L<dir>选项链接时,采用链接名为libname.a静态库或libname.so动态库的库文件或其中的一个。若两个库文件都存在,则根据编译时指定的选项(-static或-shared或不写)来进行链接。-lname选项aad.c:intadd(inta,intb){intc;c=a+b;returnc;}Linux静态库和动态库03编写功能函数静态库的创建及使用sub.c:intsub(inta,intb){intc;c=a-b;returnc;}aad.h:#ifndef__ADD_H__#define__ADD_H__intadd(inta,intb);#endifsub.h:#ifndef__SUB_H__#define__SUB_H__intsub(inta,intb);#endifgcc-cadd.c-oadd.ogcc-csub.c-osub.o编译函数为.o文件ar-crsvlibfunction.aadd.osub.o把.o文件封装成.a执行上面命令后会生成libfunction.a静态库文件,后面就可以供第三方程序使用。main.c:#include<stdio.h>#include"add.h"//引入函数声明#include"sub.h"//引入函数声明intmain(void){printf("2+1=%d;\r\n2-1=%d\r\n",add(2,1),sub(2,1));return0;}Linux静态库和动态库03测试代码编写静态库的创建及使用工程文件存放结构├──include│├──add.h│└──sub.h└──src├──add.c├──libfunction.a├──main.c└──sub.c说明:编译为.a文件后,可以把add.c、sub.c删除,我们只需要.a文件即可。编译:gcc-omainmain.clibfunction.a-I../include/
--->这种是把库当成源文件一样使用或者:gcc-omainmain.c-I../include-L./-lfunction--->这种是标准的使用方法编译生成main可执行程序后可以删除.a文件,不会影响到后面可执行程序的运行,因为它已复制一份代码到其中。运行:./mainaad.c:intadd(inta,intb){intc;c=a+b;returnc;}Linux静态库和动态库03编写功能函数动态库的创建及使用sub.c:intsub(inta,intb){intc;c=a-b;returnc;}aad.h:#ifndef__ADD_H__#define__ADD_H__intadd(inta,intb);#endifsub.h:#ifndef__SUB_H__#define__SUB_H__intsub(inta,intb);#endifgcc-fPIC-cadd.c-oadd.ogcc-fPIC-csub.c-osub.o编译函数为.o文件gcc-olibfunction.so-sharedadd.osub.o把.o文件封装成.a执行上面命令后会生成libfunction.so动态库文件,后面就可以供第三方程序使用。补充:也支持把上三条指令合并成为条:gcc-olibfunction.so-shared-fPICadd.csub.cmain.c:#include<stdio.h>#include"add.h"//引入函数声明#include"sub.h"//引入函数声明intmain(void){printf("2+1=%d;\r\n2-1=%d\r\n",add(2,1),sub(2,1));return0;}Linux静态库和动态库03测试代码编写动态库的创建及使用工程文件存放结构├──include│├──add.h│└──sub.h└──src├──add.c├──libfunction.so├──main.c└──sub.c说明:编译为.so文件后,可以把
add.c、sub.c删除,我们只需要
.so文件即可。编译:gcc-omainmain.clibfunction.so-I../include/
--->这种是把库当成源文件一样使用或者:gcc-omainmain.c-I../include-L./-lfunction--->这种是标准的使用方法编译生成main可执行程序后可以删除.a文件,不会影响到后面可执行程序的运行,因为它已复制一份代码到其中。运行:./mainLinux静态库和动态库03运行测试程序动态库的创建及使用由于采用动态库,生成main可执行程序在链接时并没有把调用的add,sub函数实现代码复制到程序当中,而在运行时候在系统库环境变量中指定的路径去搜索动态库文件,加载到内在中,然后再调用已经加载到内在的so文件中的函数代码,因此,要想正确运行使用了动态库的程序,必须把依赖的动态库也复制到系统的环境变量指定的目标当中,一般可以复制到/lib目录即可,当然其他系统环境变量指定的库存放目录也是可以的。复制动态库文件到/lib目录中:sudocplibfunction.a/lib/运行测试程序:./main
补充:如果不复制到系统环境变量指定的库路径,也可以通过临时导出环境变量的方式,不过不推荐使用这种,每次都需要操作一次,比较麻烦:exportLD_LIBRARY_PATH=$PWD
。另一种方法:修改/etc/ld.so.conf配置文件,把动态库文件所在路径写到其中,然后
执行“sudoldconfig”命令让系统重新加载
/etc/ld.so.conf配置文件可,系统会到指定路径加载动态库
。03Linux系统Makefile使用Makefile概述Makefile语法Makefile编程进阶少量代码可以直接使用:gcc源代码列表-o可执行程序名命令直接编译,如果文件数量庞大,每次都使用上面命令编译则会耗费大量时间(输入命令、哪怕是没有修改地的文件都重新编译都需要花费时间)Linux中使用makefile来解决文件编译的问题,Makefile中定义了一系列的规则,指定哪文件的编译顺序,哪些文件需要重新编译,或实现更复杂的功能操作。Makefile就像一个Shell脚本一样,Makefile文件中也使用Linux系统的命令。Makefile可以实现自动化编译”,只需要一个make命令,整个工程完全自动编译,极大的提高了软件开发的效率。使用make命令来是解释Makefile编写的编译规则。Makefile使用:把命令终端工作目录切换到包含编写好的Makefile的目录,输入make命令即可。Linux系统Makefile使用03概述Makefile概述目标1:[依赖1依赖2依赖3…]TAB键
所有依赖都实现后会执行的命令语句1TAB键
所有依赖都实现后会执行的命令语句2……依赖1:[依赖1
…]TAB键
所有依赖都实现后会执行的命令语句1……目标2:[依赖1依赖2依赖3…]TAB键
所有依赖都实现后会执行的命令语句1……Linux系统Makefile使用03Makefile文件格式Makefile语法Makefile中的目标名,根据需要随意定义;目标名一般是对应于硬盘上的一个同名文件,目标规则下方的命令语句必须以tab键开头,命令语句可根据需要编写,可以使用Linux各种命令依赖可以是一个文件,也可以是Makefile中一个目标规则;命令终端输入make命令后,make程序会在当前目录查找名为Makefile/makefile的文件,执行其中的第一个目标规则下的命令;某些操作并不需要生成文件,则PHONY关键字声明为一个伪目标;声明伪目标的语法:PHONY:目标名1目标名2
如:PHONY:rebuildclean被声明为伪目标的目的规则名make程序不会视为文件,不会关心硬盘上是否存在这个文件,而是无条件执行目标名下对应的命令;main:main.oadd.o gcc-omainmain.oadd.omain.o:main.c gcc-cmain.cadd.o:add.c gcc-cadd.c#把clean声明为伪目标PHONY:cleanclean: rm*.omainLinux系统Makefile使用03Makefile入门示例Makefile语法注意:和shell编程相同,在Makefile中,也用“#”号表示注释执行Makefile中指定目标:
语法:make[目标名]说明:不指定目标名则默认第一个目标
例如:make执行第一个目标规则下方的命令语句makeclean执行clean目标下方的命令语句makeadd.o 执行add.o目标下方的命令语句make命令会根据Makefile递归推导目标规则依赖实现方法,当所有依赖实现Makefile中可以定义变量,与C语言不同,没有类型说明,值全部以字符形式表示;Makefile定义变量:直接命名一个合法标识符,同时对它进行初始化Makefile变量初始化有四种方法:使用=
:=+=?=+==:最普通的赋值方式
;:=:
变量值简单变量展开(类似于C语言赋值),是覆盖之前的值,通常采用这种形式;
?=:如果当前变量没有赋值才进行赋值,这种方式一般用于可以通过make变量名=新值方式来替换Makefile中的定义的默认值;
比如:makefile中定义变量var?=123,命令行输入makevar=567,则var值会被567替换;+=:追加上原来的数值,如:A+=123A+=www则结果是A值为123www
变量引用:要取得变量的值,使用$(变量名)或$变量名,建议使用$(变量名)
Linux系统Makefile使用03使用普通变量Makefile编程进阶APP_NAME?=mainOBJS:=main.oadd.o$(APP_NAME):$(OBJS) gcc-o$(APP_NAME)$(OBJS)main.o:main.c gcc-cmain.c-omain.oadd.o:add.c gcc-cadd.c-oadd.o#声明伪目标 PHONY:rebuildcleanclean: rm-rf$(APP_NAME)$(OBJS)*~Linux系统Makefile使用03Makefile普通变量示例Makefile编程进阶edu118@localhost:02_makefile$makecleanrm-rfmainmain.oadd.o*~edu118@localhost:02_makefile$makegcc-cmain.c-omain.ogcc-cadd.c-oadd.ogcc-omainmain.oadd.o运行生成的可执行程序:edu118@localhost:02_makefile$./mains:3hellolinux!在Makefile规范中定义了一些具有特定含义的变量,我们可以使用这些变量进一步优化Makefile,以下列出比较常用的。Linux系统Makefile使用03使用自动变量Makefile编程进阶变量名说明$@当前规则的目标名$<当前规则的第一个依赖如main:main.oadd.o,则命令中$<结果是main.o$^当前规则的所有依赖文件,以空格分隔,如main:main.oadd.o,则命令中$^结果是main.oadd.o$?规则中日期新于目标文件的所有相关文件列表,逗号分隔$(@D)目标文件的目录名部分,如目标名是bin/main则命令中$(@D)得到bin$(@F)目标文件的文件名部分,如目标名是bin/main则命令中$(@F)得到main#外部可以传入新值替换这里的mainAPP_NAME?=mainOBJS:=main.oadd.o$(APP_NAME):$(OBJS) gcc-o$@
$^main.o:main.c gcc-c$<
-o$@add.o:add.c gcc-c$<
-o$@#声明伪目标 PHONY:rebuildcleanclean: rm-rf$(APP_NAME)$(OBJS)*~Linux系统Makefile使用03Makefile自动变量示例Makefile编程进阶edu118@localhost:02_makefile$makecleanrm-rfmainmain.oadd.o*~编译:edu118@localhost:02_makefile$makegcc-cmain.cgcc-cadd.cgcc-omainmain.oadd.o运行生成的可执行程序:edu118@localhost:02_makefile$./mains:3hellolinux!前面使用使用变量优化的Makefile还存在很多不足的,比如每添加增加一个C文件,必须增加一个生成.o文件的规则,如下:mul.o:mul.c gcc-c$<
-o$@Makefile规则中支持使用模式规则,所谓模式匹配规则即是通过匹配模式找字符串,你可以理解为类似命令的通配置符。Makefile中规定使用“%”匹配1或多个任意字符串。上面的mul.o:mul.c演变为:%.o:%.c,表示任意目标文件依赖文件是与目标文件同名.c的文件;Linux系统Makefile使用03模式规则%.o:%.cMakefile编程进阶#外部可以传入新值替换这里的mainAPP_NAME?=mainOBJS:=main.oadd.o$(APP_NAME):$(OBJS) gcc-o$@
$^%.o:%.c gcc-c$<
-o$@#声明伪目标 PHONY:rebuildcleanclean: rm-rf$(APP_NAME)$(OBJS)*~Linux系统Makefile使用03Makefile模式规则示例Makefile编程进阶edu118@localhost:02_makefile$makecleanrm-rfmainmain.oadd.o*~编译:edu118@localhost:02_makefile$makegcc-cmain.cgcc-cadd.cgcc-omainmain.oadd.o运行生成的可执行程序:edu118@localhost:02_makefile$./mains:3hellolinux!#外部可以传入新值替换这里的mainAPP_NAME?=mainOBJS:=main.oadd.o$(APP_NAME):$(OBJS) gcc-o$@
$^%.o:%.c gcc-c$<
-o$@#声明伪目标 PHONY:rebuildcleanclean: rm-rf$(APP_NAME)$(OBJS)*~Linux系统Makefile使用03模式规则使用示例Makefile编程进阶edu118@localhost:03_makefile$makecleanrm-rfmainmain.oadd.o*~编译:edu118@localhost:03_makefile$makegcc-cmain.cgcc-cadd.cgcc-omainmain.oadd.o运行生成的可执行程序:edu118@localhost:03_makefile$./mains:3hellolinux!上面的Makefile当项目文件增加时,还需要再修改Makefile,如:修改OBJS:=main.oadd.o变量值解决:使用Makefile函数的来遍历指定目录的c文件,然后把后缀名去掉,得到目标名文件列表常用几个Makefile函数有wildcard,patsubst,addprefixLinux系统Makefile使用03Makefile函数使用Makefile编程进阶#外部可以传入新值替换这里的mainAPP_NAME?=main#得到当前目录的c文件列表SRCS:=$(wildcard*.c)#把.c替换为.o,得到.o文件列表OBJS:=$(patsubst
%.c,%.o,$(SRCS))$(APP_NAME):$(OBJS) gcc-o$@
$^%.o:%.c gcc-c$<
-o$@#声明伪目标 PHONY:rebuildcleanclean: rm-rf$(APP_NAME)$(OBJS)*~Makefile函数使用示例edu118@localhost:04_makefile$makecleanrm-rfmainmain.oadd.o*~编译:edu118@localhost:04_makefile$makegcc-cmain.cgcc-cadd.cgcc-omainmain.oadd.oedu118@localhost:04_makefile$makemake:'main'isuptodate.运行生成的可执行程序:edu118@localhost:04_makefile$./mains:3hellolinux!04Linux文件IO编程Linux文件类型Linux文件IO分类非缓冲文件IO编程缓冲文件IO编程--选修Linux系统中一切皆文件,对普通文件和硬件设备的操作都是文件操作。Linux系统文件分为普通文件,管道文件,目录文件,链接文件,套接字文件和设备文件。普通文件:也称磁盘文件,并且能够进行随机的存储数据(能够自由seek定位到某一个位置);管道文件:是一个从一端发送数据,另一端接收数据的数据通道;目录文件:它包含了保存在目录中文件列表的简单文件。套接字文件:在Linux中,套接字也可以当作文件来进行处理。设备文件:Linux下各种硬件设备在内核中都表现为一个设备文件,通过设备文件操作对应的硬件设备。
设备文件有两种类型:字符型设备和块设备。
字符设备文件:是以字节为单位进行顺序读写的设备,如鼠标、键盘、终端、打印机、声卡,串口等,各类传感器;
块设备文件:以块为单位来读出或者写入数据,如CD-ROM、硬盘、SD卡,U盘等大容量存储类型设备;对文件操作一般步骤:打开文件,读/写/移动读写指针等操作,关闭文件。
其中打开文件是第一步,是为后续操作做准备的。Linux文件IO编程Linux文件类型04在Linux系统中,对文件进行读写操作有两套机制:缓冲文件IO操作和非缓冲文件IO操作;缓冲文件IO操作:这种文件操作就是标准C库提供的文件操作;对文件读写是带缓冲区的,一般使用来操作硬盘上的数据文件,如电影,音乐,文档等普通文件。非缓冲文件IO操作:这种文件操作就是Linux系统特有的文件操作,只能在Linux系统中使用,对文件读写的读写是不带缓冲区的,一般使用来操作硬件设备在系统中的设备文件,如鼠标,键盘,传感器,UART等对应的设备文件,当然,它也可以使用来操作硬盘上的数据文件,如电影,音乐,文档。IO分类Linux文件IO编程Linux文件IO分类04文件描述符:是内核为了高效的管理已经被打开的文件所创建的编号,它是一个非负整数,用于指代被打开的文件,所有执行I/O操作的系统调用都是通过文件描述符完成的。当某个程序打开文件时,操作系统返回相应的文件描述符,程序为处理该文件必须引用此描述符。所谓的文件描述符是一个低级的正整数。最前面的三个文件描述符(0,1,2)分别与标准输入(stdin),标准输出(stdout)和标准错误(stderr)对应。每次打开的文件时(含socket)必须使用当前进程中最小可用的文件描述符号码。Linux文件IO编程非缓冲文件IO编程04主要函数文件描述符open(打开)、read(读)、write(写)、close(关闭)、lseek(调整读写位置)Linux文件IO编程非缓冲文件IO编程04open函数头文件:#include<sys/types.h>#include<sys/stat.h>#include<fcntl.h>原型1:intopen(constchar*pathname,intflags);原型2:intopen(constchar*pathname,intflags,mode_tmode);功能:打开一个现有文件或创建一个新文件
参数:pathname:带路径的文件名,表示要操作哪个文件。
flags:文件打开方式,常用值:O_RDONLY:只读模式O_WRONLY:只写模式O_RDWR:可读可写O_CREAT:如果文件不存在,则创建;mode:当创建新文件时设置文件访问权限的初始值,和用户掩码umask有关;返回值:>=0:打开成功,值为文件描述符,这个数值关联pathname表示的文件;-1:打开失败。mode可取值:S_IXOTH:其他用户有执行权0o001S_IWOTH:其他用户有写权限0o002S_IROTH:其他用户有读权限0o004S_IRWXO:其他用户有全部权限(权限掩码)0o007S_IXGRP:组用户有执行权限0o010S_IWGRP:组用户有写权限0o020S_IRGRP:组用户有读权限0o040S_IRWXG:组用户有全部权限(权限掩码)0o070S_IXUSR:拥有者具有执行权限0o100S_IWUSR:拥有者具有写权限0o200S_IRUSR:拥有者具有读权限0o400S_IRWXU:拥有者有全部权限(权限掩码)0o700Linux文件IO编程非缓冲文件IO编程04read函数头文件:#include<unistd.h>原型:ssize_tread(intfd,void*buf,size_tcount);功能:从fd关联的文件中读取最多count字节数据保存到buf指向的内存首地址;参数:fd:要读取得文件描述符;buf:数据指针,指向保存读取到的数据的内存地址;count:要读取的字节数量;返回值:出错:-1;0:读文件结束;>0:成功读取到的字节数;0<=返回值<count时表示已经把文件数据读取完成;==count并不能判断已经读取完成,即使文件只count字节。注意:每读取成功1字节,文件读写位置指针增加1。write函数头文件:#include<unistd.h>原型ssize_twrite(intfd,constvoid*buf,size_tcount);功能:把buf指向的内存开始的数据,写入到fd关联的文件中,最多写入ount字节;参数:fd:要读取得文件描述符;buf:数据指针,指向要写入的数据的内存首地址;count:要写入的的字节数量;返回值:-1:出错:>0:成功写入到的字节数;注意:每写入成功1字节,文件读写位置指针增加1注意1:文件读写位置是共用一个读写位置指针的,每读写成功1字节,则文件读写位置相应增加1字节;注意2:读写函数调用都是从当前文件读写位置开始读写数据的;Linux文件IO编程非缓冲文件IO编程04lseek函数头文件:#include<sys/types.h>#include<unistd.h>原型:off_tlseek(intfd,off_toffset,intwhence);参数:fd:之前用open()获得的一个文件描述符;offset:要调整的读写偏移量,可正可负,如何由whence来确定的;whence参数分为下列三种:SEEK_SET:最终的读写位置=offset。SEEK_CUR:最终的读写位置=当前读写位置+offsetSEEK_END:最终的读写位置=文件末尾+offset返回值:>=0:调用成功,值含义是距离文件开头字节偏移。-1:调用失败,错误代码保存在全局变量errno中,常见的erron的错误代码:EBADF:fildes不是一个打开的文件描述符。ESPIPE:文件描述符被分配到一个管道、套接字或FIFO。EINVAL:whence取值不当。close函数头文件:#include<unistd.h>原型:intclose(intfd);;参数:fd:之前用open()获得的一个文件描述符;返回值:成功0,否则返回-1,同时失败原因会被记录在errno中。常见的错误原因有:EBADF:fd不是有效的文件描述符EINTR:close()被某个信号处理程序中断EIO:关闭文件时发生了IO错误Linux文件IO编程非缓冲文件IO编程04创建/写/读示例Linux文件IO编程非缓冲文件IO编程04创建/写/读示例测试结果非缓冲文件IO使用描述符(实际上一个整数)关联一个打开的文件,而缓冲文件IO使用文件流指针FILE*来关联一个已经打开的文件;文件结构FILE体内部成员一般不需要关注,编程时只通过fopen得到的FILE*类型指针来进行后面的fread,fwrite,fseek,ftell,fclose等函数进行操作。Linux文件IO编程缓冲文件IO编程04主要函数文件结构指针fopen(打开)、fread(块读)、fwrite(块写)、fclose(关闭)、fseek(调整读写位置),ftell(查询当前文件读写位置值);fgtc(单字符读)、fgets(读取字符串)、fptc(单字符写)、fputs(字符串写);Linux文件IO编程缓冲文件IO编程04fopen函数头文件:#include<stdio.h>原型:FILE*fopen(constchar*path,constchar*mode);功能:打开一个现有文件或创建一个新文件
参数:path字符串包含欲打开的文件路径及文件名;
mode字符串则代表着流形态;"r":以只读方式打开文件,该文件必须存在。
"r+":以可读写方式打开文件,该文件必须存在。
“w”:只写方式打开文件,若文件存在则文件长度清为0,即清空文件;
若文件不存在则建立该文件。
“w+”:打开可读写文件,若文件存在则文件长度清为零,即清空文件;
若文件不存在则建立该文件。“a”:以追加只写文件,若文件不存在,则会建立该文件,
如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留。
"a+":以追加方式可读可写方式打开文件,若文件不存在,则会建立该文件,
如果文件存在,写入的数据会被加到文件尾后,即文件原先的内容会被保留。返回值:非NULL:打开成功,值是指向该流的文件指针NULL:打开失败,错误代码存在全局的errno变量中。调用f
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 教育机构校长劳动合同3篇
- 新版项目软件合同3篇
- 提前终止月嫂合同3篇
- 安装工程合同中的技术解析3篇
- 数码相机购买合同3篇
- 摩托车位转让合同模板3篇
- 教育培训机构店长劳动合同3篇
- 旅游产品销售员招聘合同3篇
- 新版银行抵押贷款合同3篇
- 新媒体合作协议书3篇
- GB 1886.342-2021食品安全国家标准食品添加剂硫酸铝铵
- 期末复习必背作文 鲁教版八年级上册英语全册
- 2023年全科医师转岗培训理论考试试题及答案
- 电力系统三相短路电流计算的基本方法课件
- 福建省莆田市各县区乡镇行政村村庄村名明细及行政区划代码
- 汉字起源与发展(最新完美版)
- 网络安全的应急预案(精选13篇)
- 抚顺齐隆化工有限公司5万吨-年热聚树脂项目环境影响评价文件
- sy4209-《石油天然气建设工程施工质量验收规范-天然气净化厂建设工程》
- 2021年12月英语六级听力试题、原文及答案 两套
- AQL2.5抽检标准
评论
0/150
提交评论