所以这就是问题,我正在为我的游戏阅读一个关卡文件,在linux下工作正常:
@0
@12
200x200 version 3
@16
973 blocks
@989
@993
18 zones
但是在Windows下我得到以下结果:
@0
@212
200x200 version 3
@216
973 blocks
@1200
@1204
18 zones
呃?窗口的ftell统计数据偏移量为200?读取文件显然会产生相同的数据,但是fread使用(?)ftell的值来确定文件中可以读取的字节数。所以当我在文件末尾阅读时遇到问题:
@1425
zone#9 2x3 @ 66/9
@1425
zone#10 2x3 @ 66/9
@1425
zone#11 2x3 @ 66/9
@1425
zone#12 2x3 @ 66/9
@1425
zone#13 2x3 @ 66/9
@1425
zone#14 2x3 @ 66/9
etc.
这是相应的代码(由于所有的调试打印,目前有点难看。):
void fread_all(void *ptr, size_t size, size_t count, FILE *stream) {
fread(ptr, size, count, stream);
printf("@%ld\n", ftell(stream));
}
bool map_load(struct Map *map, const char *file_name) {
FILE *fp = fopen(file_name, "r");
if (fp != NULL) {
fseek(fp, 0, SEEK_SET);
printf("@%ld\n", ftell(fp));
// Header
int *header = (int*)calloc(sizeof(int), 3);
fread_all(header, sizeof(int), 3, fp);
printf("%dx%d version %d\n", header[0], header[1], header[2]);
map->pos_x = 0;
map->pos_y = 0;
map->map_x = 0;
map->map_y = 0;
map->size_x = header[0];
map->size_y = header[1];
map_zones_remove(map);
free(header);
// Blocks
unsigned int *block_size = (unsigned int*)malloc(sizeof(unsigned int));
fread_all(block_size, sizeof(int), 1, fp);
printf("%d blocks\n", *block_size);
unsigned char *block_data = (unsigned char*)calloc(sizeof(unsigned char), *block_size);
fread_all(block_data, sizeof(unsigned char), *block_size, fp);
unsigned char *tmp = map->blocks;
map->blocks = rle_decode(block_data, *block_size);
free(tmp);
free(block_size);
free(block_data);
// Zones
int *zone_count = (int*)malloc(sizeof(int));
fread_all(zone_count, sizeof(int), 1, fp);
printf("%d zones\n", *zone_count);
int *d = (int*)calloc(sizeof(int), 6);
for(int i = 0, l = *zone_count; i < l; i++) {
fread_all(d, sizeof(int), 6, fp);
map_zone_create(map, d[0], d[1], d[2], d[3], d[4], d[5]);
printf("zone#%d %dx%d @ %d/%d\n", i, d[2], d[3], d[0], d[1]);
}
map_platforms_create(map);
free(zone_count);
free(d);
fclose(fp);
return true;
}
return false;
}
我真的不知道这里发生了什么。编译器是Visual Studio 10和Linux下的GCC 4.4。
答案 0 :(得分:22)
以二进制模式打开文件:
FILE *fp = fopen(file_name, "rb");
在文本模式下,可能会进行翻译以匹配例如依赖于操作系统的编码。换行到C库的。
答案 1 :(得分:5)
ftell
), fseek
和"rb"
将仅作为字节偏移量工作。否则,您只能"r"
fseek
之前已归还的内容; ftell
的结果不会是字节偏移。
二进制模式在窗口上有所不同,其中文本模式将两个字符回车符,换行序列映射到单个换行符。 Linux上不需要映射。
答案 2 :(得分:0)
您不应该使用ftell
和fseek
来确定文件的大小,因为它可能是不可靠性的来源: