二进制I/O

数据写入到文件效率最高的是用二进制形式写入,二进制输出避免了在数值转换为字符串过程中,所涉及的开销和精度损失,但而精致并非人眼所能阅读,所以这个技巧只有当数据被另一个程序按顺序读取才能使用。

/*二进制读和写函数
buffer是指向用于保存数据的内存位置指针
size 是缓冲区中每个元素的字节数
count 是读取或写入的元素数量
stream是读写流*/
size_t fread(void *buffer, size_t size, size_t count, FILE *steam);
size_t fwrite(void *buffer, size_t size, size_t count, FILE *steam);

buffer被解释为一个或多个值的数组,count参数指定数组中有多少值,所以读取标量,count值应该为1。函数返回值是实际读取或者写入的元素(非字节)数目。如果遇到文件结尾或者错误,那么这个数字可能比请求的元素数目要小。

下面的代码先二进制写入浮点数到文件,然后再读出:

#include <stdio.h>
#include <string.h> int main(){
float array[2] = {1.1, 2.2};
float temp[2]; FILE *file = fopen("D:\\C\\writeBin.txt", "wb");
if(file == NULL){
printf("open error");
}
//二进制写入两个浮点数
if(fwrite(array, sizeof(float), 2, file) != 2){
printf("写入失败\n");
}
fclose(file); //二进制打开
file = fopen("D:\\C\\writeBin.txt", "rb");
//二进制读入两个浮点数
if(fread(temp, sizeof(float), 2, file) == 2){
printf("temp: %.2f %.2f\n", temp[0], temp[1]);
} return 0;
}

运行:

文件打开是二进制不可见的,所以乱码

刷新的定位函数:

fflush迫使一个输出流的缓冲区内的数据进行物理写入,不管它是否已经写满,原型如下:

int fflush(FILE *stream);

当需要把输出缓冲立即写入时可以使用这个函数

正常情况下,数据以线性的方式写入,后写入的数据在文件中是在之前写入的数据之后的,C语言支持随机访问I/O,可以任意顺序访问文件的不同位置。通过在读写前确定到文件中需要的位置来实现。下面两个函数:

//返回流当前位置,也就是下一个读取或者写入开始的位置,距离文件起始位置的偏移量
long ftell(FILE *stream);

二进制流中,返回值就是距离开始的字节数,文本流中,这个值表示一个位置,但它不一定是距离文件开始的字节数,因为系统会对行尾字符进行翻译转换,但是它的值都可以传递给fseek,作为距离开始位置偏移量。

//fseek函数允许你在一个流中定位,将改变下一个读入和写入操作的位置。
int fseek(FILE *stream, long offset, int from);

定位到起始位置之前是错误的,定位到文件尾之后写入将扩展这个文件,读取将导致“到达文件尾”信息。

#include <stdio.h>

int main()
{
float array[2] = {1.1, 2.2};
FILE *file = fopen("D:\\C\\writeBin.txt", "wb"); fwrite(array, sizeof(float), 5, file);
//偏移到seek_end之后,将扩展文件
fseek(file, 1000, 0);
fwrite(array, sizeof(float), 1, file); fclose(file);
return 0;
}

SEEK_SET,流起始位置

SEEK_CUR,流当前位置

SEEK_END,流尾部结束位置

对于二进制流,从SEEK_END进行定位可能不被支持,应该避免

对于文本流,因为会执行行末字符映射,所以文本文件的字节数可能和程序写入的字节数不同,所以无法通过SEET_CUR和SEEK_END准确定位,所以用这个两个位置时offset应该为0,如果是从SEEK_SET,offset必须是ftell的返回偏移位置。

如果使用fseek,将导致三个副作用,

1.行尾指示字符被清除

2.ungetc把字符返回流中,那么这个被退回的字符将被丢掉,因为在定位操作后,它不再是下一个字符了。

3.定位允许从写模式切换到读取模式

#include <stdio.h>

int main()
{
float array[2] = {1.1, 2.2};
float temp[1];
FILE *file = fopen("D:\\C\\writeBin.txt", "wb");
fwrite(array, sizeof(float), 5, file);
fseek(file, 1000, 0);
//更改写模式到读模式
fread(temp, sizeof(float), 1, file);
//打印1.1
printf("%lf\n", temp[1]);
fclose(file);
return 0;
}

运行:

//将读写指针设置为指定流的起始位置,同时清除流错误提示标志。
void rewind(FILE *stream);

fgetpos和tfsetpos分别是ftell和fseek的替代,但是不是标准定义的。

改变缓冲方式

当流打开后,没有进行任何其他操作时,可以通过下面的函数改变留的缓冲方式

//另设一个数组,用于对流进行缓冲
void setbuf(FILE *stream, char *buf);

为流自行指定一个缓冲,大小为BUFSIZ,定义在stdio.h中,可以防止I/O函数库为他动态分配缓冲,如果使用NULL调用,则关闭所有缓冲方式

int setvbuf(FILE *stream, char *buf, int mode, size_t size);

mode参数用于设置缓冲模式,_IONBF指定不缓冲,__IOLBF,行缓冲,每遇到换行符,缓冲刷新,__IOFBF指定一个完全缓冲流。

流错误函数

//遇到文件尾,返回真,可以通过fseek和rewind清除
int feof(FILE *stream)
//报告流错误状态,如果出现错误返回真
int ferror(FILE *stream);
//对流错误标志重置
void clearerr(FILE *stream);

临时文件和文件操纵

//创建一个文件,wb+打开,程序关闭后自动删除,不会与已存在文件同名
FILE *tmpfile(void);
//删除文件,成功返回0
int remove(char const *filename);
//重命名,成功返回0
int rename(char const*oldname, char const * newname);

重命名后删除文件:

#include <stdio.h>

int main()
{
printf("%d\n", rename("writeBin.txt", "newName"));
printf("%d\n", remove("newName"));
return 0;
}

运行

C和指针 第十五章 二进制I/O的更多相关文章

  1. C和指针 第十五章 文件I/O

    stdio.h中包含了声明FILE结构 struct _iobuf { char *_ptr; //文件输入的下一个位置 int _cnt; //当前缓冲区的相对位置 char *_base; //指 ...

  2. C和指针 第十五章 习题

    15.8 十六进制倾印码 #include <stdio.h> #include <stdlib.h> #include <string.h> #include & ...

  3. C和指针 第十五章 输入输出缓冲

    对于C,所有的I/O操作都只是简单的从程序移进或移出字节,这种字节流便成为流(stream),我们需要关心的只是创建正确的输出字节数据,以及正确的输入读取数据,特定的I/O设备细节都是对程序隐藏的. ...

  4. C和指针 第十五章 错误报告perror和exit

    15.1 错误报告 perror 任何一种程序都存在出错的可能,包括系统的函数库,当出现错误时,系统提示发生错误,标准库函数在一个外部整型变量中保存错误代码,然后把错误代码传给用户程序,提示错误原因. ...

  5. C++ Primer Plus学习:第十五章

    第十五章 友元.异常和其他 友元 友元类 表 0-1 class Tv { public: friend class Remote; } Remote类可以使用Tv的数据成员,Remote类在Tv类后 ...

  6. 【C++】《C++ Primer 》第十五章

    第十五章 面向对象程序设计 一.OOP:概述 面向对象程序设计(OOP)的核心思想是数据抽象.继承和动态绑定. 通过使用数据抽象,可以将类的接口和实现分离. 使用继承,可以定义相似的类型并对其相似关系 ...

  7. 15第十五章UDF用户自定义函数(转载)

    15第十五章UDF用户自定义函数 待补上 原文链接 本文由豆约翰博客备份专家远程一键发布

  8. 《Linux命令行与shell脚本编程大全》 第十五章 学习笔记

    第十五章:控制脚本 处理信号 重温Linux信号 信号 名称 描述 1 HUP 挂起 2 INT 中断 3 QUIT 结束运行 9 KILL 无条件终止 11 SEGV 段错误 15 TERM 尽可能 ...

  9. CSS3秘笈复习:十三章&十四章&十五章&十六章&十七章

    第十三章 1.在使用浮动时,源代码的顺序非常重要.浮动元素的HTML必须处在要包围它的元素的HTML之前. 2.清楚浮动: (1).在外围div的底部添加一个清除元素:clear属性可以防止元素包围浮 ...

随机推荐

  1. github常用操作

    1.创建一个新的repository: $cd ~/hello-world        //到hello-world目录,本地目录名与repository的名字不一定相同 $git init    ...

  2. js特效,加速度,图标跳动

    看到一个在地图上的特效,就是标注当前位置之后,图标一直在跳动,那效果看着比较得劲,就自己写了个图标跳动的jsjs代码: setTimeout("jump()",5); var t= ...

  3. [LeetCode] Valid Sudoku 验证数独

    Determine if a Sudoku is valid, according to: Sudoku Puzzles - The Rules. The Sudoku board could be ...

  4. jQuery Ajax 实例 ($.ajax、$.post、$.get)

    jQuery Ajax 实例 ($.ajax.$.post.$.get) 转 Jquery在异步提交方面封装的很好,直接用AJAX非常麻烦,Jquery大大简化了我们的操作,不用考虑浏览器的诧异了. ...

  5. 分享一些自己的学习过程和学习方法(来自daimajia)

    每天,都会有人在微博上私信我,问我关于学习和成长的问题.这种问题我一般都不会回复某个j,毕竟每个人的情况不一样,每个人对待事物的性格也不一样,我不能夸下海口的说,你看某本书几个月就能如何如何,我能做的 ...

  6. 微信小程序-关于重定向问题

    关于微信的重定向问题 wx.redirectTo(OBJECT) 参数 参数 类型 必填 说明 url 串 是 需要跳转的应用内页面的路径 成功 功能 否 接口调用成功的回调函数 失败 功能 否 接口 ...

  7. String 字符串

    多行字符串: http://php.net/manual/zh/language.types.string.php#language.types.string.syntax.nowdoc DEMO: ...

  8. UBER的故事

    今天分享一个很好的视频,19分钟,这个视频讲了过去几年一家伟大公司的成长的思路,这个演讲的PPT很棒,演讲者的语速.语调.表情等也非常适合大家在日常工作中学习.   链接:http://v.youku ...

  9. Genymotion安装问题

    今天安装虚拟机各种报错,网上收了各种资料,然而并不能解决问题.啥也没干整整一天总算找到解决方案解决.     解决方法︰ 删除所有以前的虚拟框适配器 转到设备管理器中,单击"操作" ...

  10. emmet的使用

    http://blog.wpjam.com/m/emmet-grammar/ 使用 Emmet 生成 HTML 的语法详解 开源程序 浏览:21537 2013年05月09日 文章目录[隐藏] 生成 ...