fopen函数以‘rb’模式 和 ‘r’ 模式打开文件的不同

在使用C语言基于OpenGL着色器编程时,读取glsl文件出现乱码问题。原因在于文件打开模式错误,使用'r'模式而非'rb'模式。'r'模式为文本读取,可能导致末尾出现垃圾值和换行符转换,而'rb'模式作为二进制读取则能避免这些问题,确保ftell()和fread()返回值一致。

交互式计算机图形学——基于opengl着色器的自顶向下方法的附录中,给出了读取glsl的函数:

char* readShaderSource(const char *path){
	FILE *fp = fopen(path,"r");

	if(fp==nullptr)
	  return nullptr;

	fseek(fp,0,SEEK_END);
	int size =  ftell(fp);
	fseek(fp,0,SEEK_SET);
	char *buf = new char[size+1];
	fread(buf,1,size,fp);	
	buf[size + 1] = '\0';
	cout<<buf;
	fclose(fp);

	return buf;
}

然而在用这个函数的过程中,opengl时不时会报错,“ERROR: 0:58: '<' : syntax error syntax error”。

在控制台输出读取的内容,发现在正常的glsl代码末尾有一段乱码。


原因何在?打开的方式不对。书上使用的是“r”,即文本读取模式,应该使用“rb”模式,也就是二进制读取模式打开文件。

乱码是多余的垃圾值(初始化时赋的初始值)。ftell()返回值比fread()返回值大。fread返回实际得到的字符数,正常情况文本文件中\r\n读取后变成了\n,这就会导致读取的字符串小于实际文本文件中的长度。当用rb取代r时,即用二进制模式时,ftell()和fread()返回的值相同。


参考帖子:http://www.debugease.com/vc/2167645.html


在使用 `fopen` 函数时,选择合适的文件打开模式是确保程序正确读写文件内容的关键。每种模式字符串对应不同的行为,适用于不同的使用场景。 ### 读写模式的选择 - `"r"`:以只读方式打开文件,该文件必须存在,否则函数返回 `NULL` [^4]。 - `"r+"`:以可读写方式打开文件文件必须存在,读写操作从文件开头开始 [^4]。 - `"w"`:以只写方式打开文件,如果文件已存在,其内容会被清空;如果文件不存在,则会创建新文件 [^4]。 - `"w+"`:以可读写方式打开文件,如果文件存在,内容会被清空;如果文件不存在,则创建新文件 [^4]。 - `"a"`:以追加方式打开文件,只允许在文件末尾写入数据。如果文件不存在,则创建新文件。读取操作可以从文件开头开始 [^4]。 - `"a+"`:以追加读取方式打开文件,允许在文件末尾写入数据,也允许读取文件内容。如果文件不存在,则创建新文件 。 ### 二进制模式的选择 - `"rb"`:以二进制只读方式打开文件文件必须存在 [^4]。 - `"rb+"` 或 `"r+b"`:以二进制读写方式打开文件,允许从文件开头读取写入数据,文件必须存在 。 - `"wb"`:以二进制只写方式打开文件,如果文件已存在,其内容会被清空;如果文件不存在,则创建新文件 [^4]。 - `"wb+"` 或 `"w+b"`:以二进制读写方式打开文件,如果文件存在,内容会被清空;如果文件不存在,则创建新文件 [^4]。 - `"ab"`:以二进制追加方式打开文件,只允许在文件末尾写入数据。如果文件不存在,则创建新文件 。 - `"ab+"` 或 `"a+b"`:以二进制追加读取方式打开文件,允许在文件末尾写入数据,也允许读取文件内容。如果文件不存在,则创建新文件 。 ### 模式字符串的差异 `"r+b"` `"w+b"` 的主要区别在于对文件内容的处理方式。`"r+b"` 模式要求文件必须存在,并且不会清空文件内容,读写操作从文件开头开始;而 `"w+b"` 模式会清空文件内容或创建新文件,适合需要覆盖文件的场景 [^1]。 ### 示例代码 以下是一个使用 `"w+b"` 模式打开二进制文件并进行读写操作的示例代码: ```c #include <stdio.h> int main() { FILE *fp = fopen("example.bin", "w+b"); if (fp == NULL) { printf("文件打开失败\n"); return 1; } int data = 0x12345678; fwrite(&data, sizeof(int), 1, fp); rewind(fp); int read_data; fread(&read_data, sizeof(int), 1, fp); printf("读取到的数据: 0x%x\n", read_data); fclose(fp); fp = NULL; return 0; } ``` ### 注意事项 - 在使用 `fopen` 函数时,应始终检查返回值是否为 `NULL`,以确保文件成功打开 [^4]。 - 文件操作完成后,必须调用 `fclose` 关闭文件,并将文件指针设置为 `NULL`,防止内存泄漏或游离指针问题 。 - 在 POSIX 系统(如 Linux)中,模式字符串中的 `b` 字符(表示二进制模式)会被忽略,但为了代码的可移植性清晰性,仍建议显式包含 `b` [^3]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值