C语言中文件的读写实际过程

C语言中文件的读写实际过程,第1张

RB和R本质上是二进制数据流,但用于文本的二进制数据的含义是ASCII或其他内部代码。与Rb/WB的一个重要区别是,R/W的文本换行格式在不同的系统上是不同的(DOS/windows下的回车字符Cr('\R')+换行字符LF('\n')、UNIX/Linux下的LF和MAC下的Cr),这导致文件 *** 作的某些细节不同(如fputs和fprintf对字符串结尾的处理)。

具体的读写实现由库函数处理,WB/RB模式一般采用缓冲区读写,最终实现对外部内存文件或其他输入/输出流的抽象(包括在内存中建立缓冲区)在此基础上,w/R模式处理了上述一些差异。通常,缓冲区的 *** 作取决于库函数的实现,用户看不到库函数,因为它是不必要的(无法保证不同平台的实施是一致的)。

对于标准输入设备,如键盘,它被抽象为标准输入流stdin。除了抽象过程本身(由底层系统实现)之外,实现方法与从外部存储文件读取没有本质区别。

Context提供了领个方法来打开该应用程序的数据文件夹中的文件I/O流,具体如下:

FileInputStream openFileInput(String name)

打开应用程序的数据文件夹下的name文件对应的数据流

FileOutputSream openFileOutput(String name, int mode)

打开应用程序数据文件夹下的name文件对应的输出流,mode指定文件打开的模式,该模式主要有一下4种:

①MODE_PRIVATE(该文件只能被当前应用程序读写)

②MODE_APPEND(以追加方式打开,可以在文件中追加内容)

③MODE_WORLD_READABLE(该文件内容可以被其他应用程序读取)

④MODE_WORLD_WRITEABLE(该文件内容可以被其他应用程序度,写)

   读取文件:

假设name为要打开的.文件名字

FileInputStream f = openFileInput(name)byte[] buf = new byte[1024]int hasRead = 0StringBuilder sb = new StringBuilder("")while((hasRead = f.read(buf)>0)){ sb.append(new String(buf, 0 , hasRead))}f.close()

   获取文件内容字符串:

sb.toString()

写文件:

假设要写入的字符串为content FileOutputStream f = openFileOutput(name,MODE_APPEND)PrintStream temp = new PrintStream(f)temp.println(content)temp.close()

注:应用程序的数据文件默认保存在/data/datea/files目录下,使用openFileInput和openFileOutput方法来打开文件输入流,输出流时,打开的都是应用程序的数据文件夹里的文件,也就是说是在手机内存中的文件,而不是SD卡中的文件。

内核中读写文件

1.filp_open()在kernel中可以打开文件,其原形如下:

Struct file* filp_open(const char* filename, int open_mode, int mode)该函数返回strcut file*结构指针,供后继函数 *** 作使用,该返回值用IS_ERR()来检验其有效性。

2. 读写文件(vfs_read/vfs_write)

kernel中文件的读写 *** 作可以使用vfs_read()和vfs_write,在使用这两个函数前需要说明一下get_fs()和 set_fs()这两个函数。

vfs_read() vfs_write()两函数的原形如下:

ssize_t vfs_read(struct file* filp, char __user* buffer, size_t len, loff_t* pos)

ssize_t vfs_write(struct file* filp, const char __user* buffer, size_t len, loff_t* pos)

注意这两个函数的第二个参数buffer,前面都有__user修饰符,这就要求这两个buffer指针都应该指向用空的内存,如果对该参数传递kernel空间的指针,这两个函数都会返回失败-EFAULT。但在Kernel中,我们一般不容易生成用户空间的指针,或者不方便独立使用用户空间内存。要使这两个读写函数使用kernel空间的buffer指针也能正确工作,需要使用set_fs()函数或宏(set_fs()可能是宏定义),如果为函数,其原形如下:

void set_fs(mm_segment_t fs)

该函数的作用是改变kernel对内存地址检查的处理方式,其实该函数的参数fs只有两个取值:USER_DS,KERNEL_DS,分别代表用户空间和内核空间,默认情况下,kernel取值为USER_DS,即对用户空间地址检查并做变换。那么要在这种对内存地址做检查变换的函数中使用内核空间地址,就需要使用set_fs(KERNEL_DS)进行设置。get_fs()一般也可能是宏定义,它的作用是取得当前的设置,这两个函数的一般用法为:


欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/tougao/11549952.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-16
下一篇 2023-05-16

发表评论

登录后才能评论

评论列表(0条)

保存