android5.1.1移植教程,iTOP4412开发板Android5.1.1移植教程

对于传统的操作系统来说,普通的 I/O 操作一般会被内核缓存,这种 I/O 被称作缓存 I/O。本文所介绍的文件访问机制不经过操作系统内核的缓存,数据直接在磁盘和应用程序地址空间进行传输,所以该文件访问的机制称作为直接 I/O。Linux 中就提供了这样一种文件访问机制,对于那种将 I/O 缓存存放在用户地址空间的应用程序来说,直接 I/O 是一种非常高效的手段。

“Linux 中一切皆文件”这句话已经不知道说了多少遍了,后面也会提到很多次。那么在深入学习之前,肯定要掌握对 Linux 文件的各种操作,包括读、写、创建等基本知识。

本章配套视频为:

“视频 05_01 文件 IO 之 open 打开操作”

“视频 05_02 文件 IO 之 creat 创建操作”

“视频 05_03 文件 IO 之 write 写操作”

“视频 05_04 文件 IO 之 read 读操作”

16.1 Linux 中 中 IO  的概念介绍

所有的 I/O 操作都是通过读文件或者写文件来完成的。在这里,把所有的外围设备,包括键盘和显示器,都看成是文件系统中的文件。

什么是缓存 I/O

缓存 I/O 又被称作标准 I/O,大多数文件系统的默认 I/O 操作都是缓存 I/O。在 Linux 的缓存 I/O 机制中,操作系统会将 I/O 的数据缓存在文件系统的页缓存( page cache ) 中,也就是说,数据会先被拷贝到操作系统内核的缓冲区中,然后才会从操作系统内核的缓冲区拷贝到应用程序的地址空间。缓存 I/O 有

以下这些优点:

缓存 I/O 使用了操作系统内核缓冲区,在一定程度上分离了应用程序空间和实际的物理设备。

缓存 I/O 可以减少读盘的次数,从而提高性能。

当应用程序尝试读取某块数据的时候,如果这块数据已经存放在了页缓存中,那么这块数据就可以立即返回给应用程序,而不需要经过实际的物理读盘操作。当然,如果数据在应用程序读取之前并未被存放在页缓存中,那么就需要先将数据从磁盘读到页缓存中去。对于写操作来说,应用程序也会将数据先写到

页缓存中去,数据是否被立即写到磁盘上去取决于应用程序所采用的写操作机制:如果用户采用的是同步写机制( synchronous writes ), 那么数据会立即被写回到磁盘上,应用程序会一直等到数据被写完为止;如果用户采用的是延迟写机制( deferred writes ),那么应用程序就完全不需要等到数据全部被写回到磁盘,数据只要被写到页缓存中去就可以了。在延迟写机制的情况下,操作系统会定期地将放在页缓存中的数据刷到磁盘上。与异步写机制( asynchronous writes )不同的是,延迟写机制在数据完全写到磁盘上的时候不会通知应用程序,而异步写机制在数据完全写到磁盘上的时候是会返回给应用程序的。所以延迟写机制本身是存在数据丢失的风险的,而异步写机制则不会有这方面的担心。

缓存 I/O  的缺点

在缓存 I/O 机制中,DMA 方式可以将数据直接从磁盘读到页缓存中,或者将数据从页缓存直接写回到磁盘上,而不能直接在应用程序地址空间和磁盘之间进行数据传输,这样的 话,数据在传输过程中需要在应用程序地址空间和页缓存之间进行多次数据拷贝操作,这些数据拷贝操作所带来的 CPU 以及内存开销是

非常大的。

对于某些特殊的应用程序来说,避开操作系统内核缓冲区而直接在应用程序地址空间和磁盘之间传输数据会比使用操作系统内核缓冲区获取更好的性能,下边这一小节中提到的自缓存应用程序就是其中的一种。

小贴士-关于文件的相关知识的一点说明

在 Linux 中,文件非常重要,所以在 linux 系统中提供了丰富的文件操作函数。

在系统编程中,只会介绍编程中用到的最终的 API 和必须掌握的知识,还有大量的和文件相关的知识,例如高级 IO,制作文件系统,虚拟文件系统,文件共享,网络文件系统等等。

如果真是要详细的介绍整个 linux 文件,可能好几千页的书都介绍不完,而且对于大家学习初期也没有太大的帮助,得不偿失。

手册中还会以其它形式介绍文件相关的知识,有的时候会介绍一步一步介绍如何操作,例如制作文件系统,NFS 网络启动;有时候会介绍如何使用,例如内核教程中介绍的虚拟文件系统等等。大家如果感兴趣,在学习教程之外可以通过学习相关知识,去了解更加具体的含义。

文件对于 Linux 实在是太重要了,不过大家学习了本章的知识和实验,了解其他和文件系统相关的知识,在后面 Linux 编程中遇到的问题基本可以自行解决了。

函数头文件

在所有的 Linux 系统中,如果需要对文件的进行操作,只要包含如下 4 个头文件即可。

#include

#include

#include

#include

上面四个头文件中包含了打开,关闭,创建,读文件,写文件的函数,还有标志位,以及在不同 32 位以及 64 位系统下数据长度的宏变量定义。

16.2  打开文件函数 open

使用 open 函数的时候会返回一个文件句柄,文件句柄是文件的唯一识别符 ID。对文件的操作必须从读取句柄开始。

先来看一下函数 open 的两个原型。

int open(const char *path, int oflags);

有两个参数的‘open’函数主要用于创建文件,在本章的 10.5.5 小节会和创建文件的函数 creat 的同时介绍具体用法,并给出例子。

int open(const char *path, int oflags,mode_t mode);

open 函数可以建立一个文件或者设备的访问路径。在打开或创建文件时可以指定文件的属性及用户的

权限等参数。

第一个参数 path 表示:路径名或者文件名。路径名为绝对路径名,例如开发板中的 led 驱动的设备节点/dev/leds。

第二个参数 oflags 表示:打开文件所采取的动作。下 面 三 个 选 项 是 必 须 选 择 其 中 之 一的 。

O_RDONLY 文件只读

O_WRONLY 文件只写

O_RDWR 文件可读可写

下面是可以任意选择的。

O_APPEND 每次写操作都写入文件的末尾

O_CREAT 如果指定文件不存在,则创建这个文件

O_EXCL 如果要创建的文件已存在,则返回 -1,并且修改 errno 的值

O_TRUNC 如果文件存在,并且以只写/读写方式打开,则清空文件全部内容

O_NOCTTY 如果路径名指向终端设备,不要把这个设备用作控制终端。

O_NONBLOCK 如果路径名指向 FIFO/块文件/字符文件,则把文件的打开和后继 I/O 设置为非阻塞模式

(nonblocking mode),后面会介绍什么是阻塞和非阻塞。

O_NDELAY 和 O_NONBLOCK 功能类似,调用 O_NDELAY 和使用的 O_NONBLOCK 功能是一样的。第三个参数 mode 表示:设置创建文件的权限。

S_IRUSR,S_IWUSER,S_IXUSR,S_IRGRP,S_IWGRP,S_IXGRP,S_IROTH,S_IWOTH,S_IXOTH. 其中 R:读,W:写,X:执行,USR:文件所属的用户,GRP:文件所属的组,OTH: 其他用户。第三个参数可以直接使用参数代替,参考 10.4.5 小节‘Linux 权限’。

前面用过的‘chmod 0777 helloworld’命令,其中的含义是一样的,只不过 chmod 是在文件创建之后再修改权限。

open  函数代码

编写简单的 open.c 文件测试 open 函数。首先添加头文件,如下图所示。

首先添加头文件,如下图所示。

213585306_1_20210121021803350.jpg

然后 main 函数如下。

213585306_2_20210121021803491

上图中打开了三个文件分别属于不同的情况。

/dev/leds 已经在开发板中存在,属于驱动的设备节点,在 linux 驱动教程中会具体介绍

/bin/test1 和/bin/test2 都不存在

使用 open 函数调用上面三个文件,如果出错就会打印错误,然后打印句柄。

编译运行测试

在 Ubuntu 系统下,如下图所示,进入前面实验创建的目录“/home/linuxsystemcode/iofile”,使用命令“mkdir iofile”新建 iofile 文件夹,将源码 open.c 拷贝进去,进入新建的文件夹 iofileopen,如下图所示。

213585306_3_20210121021803741

使用命令“arm-none-linux-gnueabi-gcc -o open open.c -static”编译 open 文件,如下图所示,使用命令“ls”可以看到生成了 open 可执行文件。

213585306_4_2021012102180423

这里介绍 U 盘拷贝代码的方法,也可以编译进文件系统,具体方法参考 10.3.5 小节。将编译成的可执行文件 open,拷贝到 U 盘,启动开发板,插入 U 盘,加载 U 盘,运行程序如下。

213585306_5_20210121021804303

如上图所示,可以看到打开/dev/leds 成功,这个是板载 LED 的内核驱动,调用的时候, 还调用了内核驱动中的函数,这个函数会打印“LEDS_CTL DEBUG:Device Opened Success!”和“LEDS_CTL DEBUG:DeviceOpened Success!”

然后打印句柄 ID,/dev/leds fd is 3

调用“/bin/test1”会报错“open /bin/test1 failed”,这种打开文件的方式是 linux 中标准的用法,几乎所有对文件的 open 操作都会加上出错报警的语句。

创建“/bin/test2”会打印“/bin/test2 fd is 4”,表明创建“/bin/test2 ”成功了。

使用命令“ls /bin/test2”,查看一下对应目录‘/bin’下应该新建了“test2”,如下图所示。

213585306_6_20210121021804600

另外的“dev/leds”本身就存在,如下图所示,这是驱动的设备节点文件,在后面的实验会介绍如何操作调用,在 linux 驱动实验中会介绍这个设备节点文件是如何生成的。

213585306_7_20210121021804913

16.3  创建函数 creat 和 和 open

creat  函数介绍

关于 creat 函数,首先这个单词并不是表示创建的意思,创建的英文单词是“create”, 这是早期的一个小的拼写错误,却一直沿用下来。

在介绍 open 函数的时候,可以看到 open 函数有两种形式,一个是两个参数一个是三个参数,早期的时候 open 只有三个参数的形式,三个参数的形式会导致 open 函数无法打开一个未创建的文件,也就是无法建立文件,所以就有了这个 creat 函数。

现在 creat 函数可以完全用 open 替代,考虑到在阅读代码的时候可能会碰到,所以简单介绍一下。

creat 函数原型如下。

int creat(const char * pathname, mode_t mode);

creat 函数只有两个参数,参数的含义和 open 类似。大家看到这个函数的时候知道它是创建文件的就成,在写代码的时候完全可以用 open 代替。

creat  函数例程

编写简单的 creat.c 文件测试 creat 函数。首先添加头文件,如下所示。

213585306_8_20210121021805116.jpg

然后 main 函数如下所示。

213585306_9_20210121021805366

第 22 行、27 行、31,open 可以打开已有的文件,也可以打开不存在的文件,即创建文件,创建文件的时候需要在参数中添加标志位 O_CREAT。在第 27 行代码中,没有添加标志位,运行的时候肯定会报错,这么写是希望大家能够记住这个参数。

第 36 行是使用 creat 函数创建文件“test3”,注意一下 creat 函数和 open 函数在创建文件的时候,参数的区别。

编译运行测试

在 Ubuntu 系统下,如下图所示,进入前面实验创建的目录“/home/linuxsystemcode/iofile”,将源码creat.c 拷贝进去,如下图所示。

213585306_10_20210121021805694

使用命令“arm-none-linux-gnueabi-gcc -o creat creat.c -static”编译 creat.c 文件,如下图所示,使用命令“ls”可以看到生成了 creat 可执行文件。

213585306_11_2021012102180637

这里介绍 U 盘拷贝代码的方法,也可以编译进文件系统,具体方法参考 10.3.5 小节

将编译成的可执行文件 creat,拷贝到 U 盘,启动开发板,插入 U 盘,加载 U 盘,运行程序如下。

213585306_12_20210121021806350

如上图所示。

打开文件"/dev/leds"成功,这个文件已经存在

打开文件"/bin/test1"失败,因为没有添加参数 O_CREAT,这个文件不存在,新建的时候需要参数

O_CREAT。

打开文件"/bin/test2"成功,不存在这个文件,创建成功。

打开文件"/bin/test3"成功,不存在这个文件,使用 creat 新建成功。

如下图所示,使用命令“ls /bin/test* ”在“/bin”目录下可以看到新建的文件 test2 和 test3。

213585306_13_20210121021806459.jpg

16.4  关闭函数 close

任何一个文件在操作完成之后都需要关闭,这个时候需要调用 close 函数。

close  函数介绍

调用 close 函数之后,会取消 open 函数建立的映射关系,句柄将不再有效,占用的空间将被系统释放。

close 函数在头文件“#include ”中,close 函数的使用和参数都比较简单.

int close(int fd);

参数 fd,使用 open 函数打开文件之后返回的句柄。返回值,一般很少使用 close 的返回值。

close  函数例程

调用很简单,在下一个实验中会永到 close 函数。

16.5  写函数 write

对文件进行写操作,write 函数使用的比较多。

write  函数介绍

write 函数在头文件“#include ”中。

函数原型为 ssize_t write(int fd,const void *buf,size_t count) 参数 fd,使用 open 函数打开文件之后返

回的句柄。

参数*buf,需要写入的数据。

参数 count,将参数*buf 中最多 count 个字节写入文件中。

返回值为 ssize 类型,出错会返回-1,其它数值表示实际写入的字节数。

write  函数例程

编写简单的 write.c 文件测试 write 函数。首先添加头文件,如下所示。

//标准输入输出头文件

#include

//文件操作函数头文件

#include

#include

#include

#include

#include

然后 main 函数如下所示。

213585306_14_20210121021806725

如上图代码所示。

在 16 行定义了 buffer_write 字符数组。

在 18 行,进行写操作之前,必须得到文件的句柄,在这一行中使用 open 函数创建和打开文件“/bin/testwrite"。

在 23 行中会调用 write 函数,将 buffer_write 字符数组中的内容写到新建的文件中。

在 31 行调用 close 函数,将"/bin/testwrite"文件关闭。

后面测试的时候可以在超级终端中,使用 vi 编辑器打开"/bin/testwrite"文件,可以看到这个文件中有字符 Hello Write Function!.

编译运行测试

在 Ubuntu 系统下,如下图所示,进入前面实验创建的目录“/home/linuxsystemcode/iofile”,将源码write.c 拷贝进去,如下图所示。

213585306_15_20210121021806959

使用命令“arm-none-linux-gnueabi-gcc -o write write.c -static”编译 write.c 文件,如下图所示,使用命令“ls”可以看到生成了 write 可执行文件。

213585306_16_20210121021807147

这里介绍 U 盘拷贝代码的方法,也可以编译进文件系统。

将编译成的可执行文件 write,拷贝到 U 盘,启动开发板,插入 U 盘,加载 U 盘,运行程序。如下图所示,打印出了 Write Function OK!。

213585306_17_20210121021807397.jpg

在代码中定义的文件是"/bin/testwrite",使用 vi 编辑器打开文件,如下图所示,程序执行运行成功。

213585306_18_20210121021807522

16.6  文件的读 read  函数

对文件进行写操作,read 函数使用的比较多。

read  函数介绍

read 函数在头文件“#include ”中。

函数原型为 ssize_t read(int fd,void *buf,size_t len)

参数 fd,使用 open 函数打开文件之后返回的句柄。

参数*buf,读出的数据保存的位置。

参数 len,每次最多读 len 个字节。

返回值为 ssize 类型,出错会返回-1,其它数值表示实际写入的字节数,返回值大于 0 小于 len 的数

值都是正常的。

read  函数例程

编写简单的 read.c 文件测试 read 函数。

首先添加头文件和定义读函数缓冲区为 1000,如下图所示。

//标准输入输出头文件

#include

//文件操作函数头文件

#include

#include

#include

#include

#include

然后 main 函数如下图所示。

213585306_19_20210121021807803

如上代码所示。 在 1-9 行,头文件。

在 20 行,使用 open 函数打开或者新建"/bin/testwrite"文件。

在 23 行,使用 write 函数将 buffer 中的内容写到"/bin/testwrite"文件中。在 35 行,使用 read 函数,将"/bin/testwrite"文件中的内容读出来。

在 38 行,使用打印函数 printf 打印 read 函数读出的数据。在 39 行,调用 close 函数关闭打开的文件,程序结束。

最终测试的时候,除了会出现"/bin/testwrite"文件,还会打印 read 函数读取的数据。

编译运行测试

在 Ubuntu 系统下,如下图所示,进入前面实验创建的目录“/home/linuxsystemcode/iofile”,将源码read.c 拷贝进去,如下图所示。

213585306_20_2021012102180853

使用命令“arm-none-linux-gnueabi-gcc -o read read.c -static”编译 read.c 文件,如下图所示,使用命令“ls”可以看到生成了 read 可执行文件。

213585306_21_20210121021808303

这里介绍 U 盘拷贝代码的方法,也可以编译进文件系统。

将编译成的可执行文件 read,拷贝到 U 盘,启动开发板,插入 U 盘,加载 U 盘,运行程序。如下图所示,可以看到打印出了“Files Content is Hello Write Function”,使代码中预期的结果。

213585306_22_20210121021808600

213585306_23_20210121021808897

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

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

相关文章

岗位po是什么意思_劳务派遣是什么意思,劳务派遣和正式员工区别在哪

劳务派遣也是劳动活动的一种,虽然性质相对于临时工,但是劳务派遣仍然属于劳动合同法所调整的,不过与正式员工存在一定区别。因此接下来将由找法网小编为您介绍关于劳务派遣是什么意思,劳务派遣和正式员工区别在哪及其相关方面的知…

Android对性能要求,Android性能管理

管理Android设备的功耗和性能有助于确保应用程序在各种硬件上始终如一地顺畅运行。在Android 7.0及更高版本中,OEM可以实现对持续性能提示的支持,使应用程序能够保持一致的设备性能,并指定专属核以提高CPU密集型前台应用程序的性能。持续性能…

layui导航栏页面滚动固定_帮你搞定长滚动网页的设计!这5种设计策略需谨记!...

网站中的内容再好,总得让用户能看到,才有价值,而矛盾在于,首屏的空间始终是有限的。所以,我们总希望用户能够滚动页面向下看,从某种意义上来说,长滚动页面和无限滚动的设计,如此地受…

南邮计算机专硕考研专业课,南京邮电大学(专业学位)计算机技术研究生考试科目和考研参考书目...

考研真题资料优惠价原价选择南京邮电大学(专业学位)计算机技术教材,也叫南京邮电大学(专业学位)计算机技术考研参考书、指定书目等等,是考验专业课复习过程中最重要的资料。考研是一种针对性很强的考试项目,参考书目由报考院校的研究生院制定…

ios 设置属性的center_【从0到1的Stata图表学习1】图例设置

导读规则:正文出现非加粗红色字体,对应Stata命令;正文中出现蓝色字体,对应往期链接;do文件中:"/**/"符号代表作者注释内容,帮助理解;"//"代表分节,便…

计算机网络的应用领域有那些,计算机网络应用领域

描述计算机网络应用领域一、计算机网络在现代企业中的应用计算机网络的发展和应用改变了传统企业的管理模式和经营模式。在现代企业中企业信息网络得到了广泛的应用。它是一种专门用于企业内部信息管理的计算机网络,覆盖企业生产经营管理的各个部门,在整…

ffmpeg rtsp转hls_Qt音视频开发24-ffmpeg音视频同步

## 一、前言用ffmpeg来做音视频同步,个人认为这个是ffmpeg基础处理中最难的一个,无数人就卡在这里,怎么也不准,本人也是尝试过网上各种demo,基本上都是渣渣,要么仅仅支持极其少量的视频文件比如收到的数据包…

1119. Pre- and Post-order Traversals (30)

友情提示:这题非常值得自己思考独立做出来,请反复确认后再往下拉 1119. Pre- and Post-order Traversals (30) 时间限制400 ms内存限制65536 kB代码长度限制16000 B判题程序Special作者CHEN, YueSuppose that all the keys in a binary tree are distinc…

单招计算机英语面试口语,单招面试英语自我介绍范文 自我介绍说什么

单招面试英语自我介绍范文 自我介绍说什么2018-03-11 10:50:17文/李男学好英语就意味着可能有更多的机会,单招面试中英语自我介绍也是很重要的。那么,英语自我介绍说什么呢?下面小编整理了一些英语自我介绍范文,供大家参考&#x…

angularjs directive scope变化为啥html,学习AngularJs:Directive指令用法(完整版)

本教程使用AngularJs版本:1.5.3摘要:Directive(指令)笔者认为是AngularJ非常强大而有有用的功能之一。它就相当于为我们写了公共的自定义DOM元素或CLASS属性或ATTR属性,并且它不只是单单如此,你还可以在它的基础上来操作scope、绑…

lisp遍历表中所有顶点_三十张图片让你彻底弄明白图的两种遍历方式:DFS和BFS...

1 引言遍历是指从某个节点出发,按照一定的的搜索路线,依次访问对数据结构中的全部节点,且每个节点仅访问一次。  在二叉树基础中,介绍了对于树的遍历。树的遍历是指从根节点出发,按照一定的访问规则,依次…

通用测试用例大全

为方便平时写测试用例,整理如下: 功能条件测试步骤测试数据预期结果备注搜索或查询 单独遍历各查询条件,测试按各查询条件是否都能够查询出相应的值. 查询出符合条件的记录 设置界面上所有查询条件进行查询,单击查询按钮后,测试执…

Net学习日记_ASP.Net_MVC_新语法笔记

01.新语法 本章中主要讲解.Net框架性语法。开发者可以使用新语法提高编程的效率以及代码的运行效率;其本质都是“语法糖”,由编译器在编译时转成原始语法。 1 自动属性 Auto-Implemented Properties 2 隐式类型 var 3 对象初始化器 与 集合初始化器 { …

Mybatis_接口编程

Mybatis参考使用文档:http://www.mybatis.org/mybatis-3/zh/index.html 1.项目结构 2.新增EmployeeMapper.java接口代码 package com.atguigu.mybatis.dao;import com.atguigu.mybatis.bean.Employee;public interface EmployeeMapper {public Employee getEmpById(…

iconv 判断字符编码_GBK 和 UTF8编码

相关学习内容:https://study.163.com/course/courseMain.htm?courseId1210747815&share2&shareId480000002227524​study.163.com9节课征服「字符编码」 - 网易云课堂9节课征服「字符编码」 - 网易云课堂​study.163.com首先来看一下常用的编码有哪些&…

完美粉红噪声及各种声波测试软件,粉红噪声的测试原理

粉红噪声。既然是噪声就绝对不是单纯的纯音,它是一种频率覆盖范围很宽的声音。低频能下降到接近0Hz(不包括0Hz)高频端能上到二十几千赫,而且它在等比例带宽内的能量是相等的(误差只不过0.1dB左右)。比如用1/3oct带通滤波器去计算分析,我们会发…

catia怎么进入装配_CATIA教程之常见去参方法

Catia去除参数常用于以下几种情形:1)模型比较大,通过邮件发送不方便,去除模型参数可以减小模型数据大小。2)有时候需要将模型给别人使用,不希望对方知道你的建模过程或者更改你的模型,为了保密或者保护自己&#xff0c…

beeline执行sql文件_MyBatis的SQL执行流程不清楚?看完这一篇就够了

前言MyBatis可能很多人都一直在用,但是MyBatis的SQL执行流程可能并不是所有人都清楚了,那么既然进来了,通读本文你将收获如下:1、Mapper接口和映射文件是如何进行绑定的2、MyBatis中SQL语句的执行流程3、自定义MyBatis中的参数设置…

Git 下载、安装与SSH配置

一、Git学习笔记(基于Github) Git简介 Git是目前流行的分布式版本管理系统。它拥有两套版本库,本地库和远程库,在不进行合并和删除之类的操作时这两套版本库互不影响。也因此其近乎所有的操作都是本地执行,所以在断网的…

windows7未能启动怎么修复计算机,win7系统提示windows无法启动这个硬件设备怎么办...

‍‍最近有很多深度技术win7用户在打开视频、光驱很有usb的时候,电脑出现了“图像处理设备由于其配置信息不完整或已损坏Windows 无法启动这个硬件设备(代码19)”的提示,这该怎么办呢?对于设备管理器中设备不正常的问题,下面由小编…