为什么文件指针的作用与 C 语言中的普通指针不同?

问题描述

这是我的 C 代码

#include"stdio.h"
#include"stdlib.h"
#include"string.h"
#include"limits.h"

int main(int argc,char **argv){
    int Array[2]={100,150};
    int *pArray=Array;
    int *qArray=pArray;
    qArray++;

    printf("pArray address:%p\n",pArray);
    printf("qArray address:%p\n",qArray);

    if(pArray!=qArray){
        printf("pArray not equals to qArray\n\n");
    }

    FILE *fp1=NULL,*fp2=NULL,*fp3=NULL,*fp4=NULL,*fp5=NULL;

    fp1=fopen("test.bin","rb+");
    if(fp1==NULL){
        return 1;
    }

    fp2=fp1;
    fp3=fp1;
    fp4=fp1;
    fp5=fp1;

    fseek(fp2,100l,SEEK_CUR);
    fseek(fp3,200l,SEEK_CUR);
    fseek(fp4,300l,SEEK_CUR);
    fseek(fp5,400l,SEEK_CUR);

    printf("fp1 position:%lu\n",ftell(fp1));
    printf("fp2 position:%lu\n",ftell(fp2));
    printf("fp3 position:%lu\n",ftell(fp3));
    printf("fp4 position:%lu\n",ftell(fp4));
    printf("fp5 position:%lu\n",ftell(fp5));

    if(fp1==fp5){
        printf("fp1 equals to fp5.\n");
    }

    fclose(fp1);
    return 0;
}

这是输出

$ ./program 
pArray address:0x7ffe5e6780a0
qArray address:0x7ffe5e6780a4
pArray not equals to qArray

fp1 position:1000
fp2 position:1000
fp3 position:1000
fp4 position:1000
fp5 position:1000
fp1 equals to fp5.

我得出一个结论:移动普通指针不会影响与其共享地址的指针,但是使用fseek函数移动文件指针会影响。

为什么文件指针的作用与普通指针不同?

解决方法

A FILE * 不是指向文件的指针;它是一个指向 FILE 的指针。根据 C 2018 7.21.1 2,FILE 是:

……一种能够记录控制流所需的所有信息的对象类型,包括其文件位置指示符、指向其关联缓冲区的指针(如果有)、一个错误指示符,用于记录发生读/写错误,并且文件结束指示符记录是否已到达文件末尾......

因此 fp2=fp1; 使 fp2 指向用于控制文件的同一个对象,因此对 fp1 指向的对象所做的任何更改都会出现在由fp2 反之亦然,因为它们是同一个对象。

qArray++ 改变指针 qArrayfseek(fp2,100l,SEEK_CUR); 更改 fp2 指向的对象(及其控制的流)。类似地,如果您调用某个例程 ChangeAnElementInArray(qArray,index,value),则会更改数组中的一个元素;它不会改变 qArray 指针。

此外,FILE * 不是文件中的位置。它只是一个指向用于控制流的数据结构的指针。