1、对于内核而言,所有打开文件都由文件描述符引用。文件描述符是一个非负整数,当打开一个现存文件或创建一个新文件时,用open或creat返回的文件描述符标识该文件,将其作为参数传递给read或write。
按照惯例,UNIX shell使用文件描述符0与进程的标准输入相结合,文件描述符1与标准输入相结合,文件描述符2与标准错误输出相结合。 STDUIN_FILENO、STDOUT_FILENO、STDERR_FILENO,这几个宏代替了0,1,2这几个数。
#include#include #include #include #include #include #include int main() { int fd; char readBuf[128]; int n_read = read(0,readBuf,5); int n_write = write(1,readBuf,strlen(readBuf)); return 0; }
2、文件描述符,这个数字再一个进程中表示一个特定的含义,当我们open一个文件时,操作系统在内存中构建了一些数据来表示这个动态文件,然后返回给应用程序一个数字作为文件描述符,这个数字就和我们内存中维护这个动态文件的这些数据结构绑定上了,义后我们应用程序如果要操作这个动态文件,只需要文件描述符区分。
3、文件描述符的作用域就是当前进程,出了这个进程,文件描述符就没有意义了。
open函数打开文件,打开成功返回一个文件描述符,打开失败,返回-1
静态文件与动态文件1、在Linux中要操作一个文件,一般是先open打开一个文件,得到文件描述符,然后对文件进行读写或其他操作,最后是close关闭文件。
2、一定要关闭文件,否则会造成文件的损坏。
3、文件平时是存放在块设备中的文件系统文件中的,我们把这种文件叫做静态文件,当我们去open打开一个文件时,linux内核做的操作包括:内核在进程中建立一个打开文件的数据结构,记录下我们打开的这个文件;内核在内存中申请一段内存,并将静态文件的内容从块设备中读取到内核中特定地址管理存放(叫动态文件)。
4、打开文件以后,以后对这个文件的读写操作,都是针对内存中的这一份动态文件的,而并不是针对静态文件。当我们对动态文件进行读写以后,内存中动态文件和块设备中的静态文件就不同步了,当我们close关闭文件时,close内部内核将内存中的动态文件的内容去更新块设备中的静态文件。
5、为什么不直接对块设备操作
块设备本身读写非常不灵活,是按块读写的,而内存是按字节单位操作的,而且可以随即操作,很灵活。
练习:实现cp指令cp src.c des.c 有三个参数,
int main(int argc,char **argv)
int argc是main带的参数个数,第二个参数const char *argv[]或者char **argv或者char argv[][]是具体的参数。
1、C语言参数:./a.out
2/思路:a、打开原文件;
b、将原文件内容读到buf;
c、打开/创建目标文件;
d、将buf写入到目标文件;
e、close两个文件
#include练习2:配置文件的修改#include #include #include #include #include #include int main(int argc,char **argv) { int fdSrc; int fdDes; char readBuf = NULL; if(argc != 3){ printf("pararm errorn"); exit(-1); } fdSrc = open(argv[1],O_RDWR); int n_read = read(fdSrc,readBuf,size); fdDes = open(argv[2],O_RDWR|O_CREAT|O_TRUNC,0600); int n_write = write(fdDes,readBuf,strlen(readBuf)); close(fdSrc); close(fdDes); return 0; }
#include写结构体/结构体数组到文件#include #include #include #include #include #include int main(int argc,char **argv) { int fdSrc; char *readBuf = NULL; if(argc != 2){ printf("pararm errorn"); exit(-1); } fdSrc = open(argv[1],O_RDWR|O_TRUNC); int size = lseek(fdSrc,0,SEEK_END); lseek(fdSrc,0,SEEK_SET); readBuf=(char*)malloc(sizeof(char)*size + 8); int n_read = read(fdSrc,readBuf,1024); char *p = strstr(readBuf,"LENG="); if(p==NULL){ printf("not foundn"); exit(-1); } p = p+strlen("LENG="); *p = '5'; lseek(fdSrc,0,SEEK_SET); int n_write = write(fdSrc,readBuf,strlen(readBuf)); close(fdSrc); return 0; }
#include标准C库打开/创建/读写文件#include #include #include #include #include #include struct Test { int a; char c; }; int main() { int fd; struct Test data[2] = {{100,'a'},{101,'b'}}; struct Test data2[2]; fd = open("./file1",O_RDWR|O_CREAT,0600); int n_write = write(fd,&data,sizeof(struct Test)*2); lseek(fd,0,SEEK_SET); int n_read = read(fd,&data2,sizeof(struct Test)*2); printf("read %d, %cn",data2[0].a,data2[0].c); printf("read %d, %cn",data2[1].a,data2[1].c); return 0; }
与Linux的差不多
#include标准C库写入结构体到文件#include int main() { //FILE *fopen(const char *path, const char *mode); FILE *fp; char *str = "qieerbushe"; char readBuf[128] = {0}; fp = fopen("./1.txt","w+"); //size_t fwrite(const void *ptr, size_t size, size_t nmemb,FILE *stream); fwrite(str,sizeof(char),strlen(str),fp); //int fseek(FILE *stream, long offset, int whence); fseek(fp,0,SEEK_SET); //size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream); fread(readBuf,sizeof(char),strlen(str),fp); printf("read data: %sn",readBuf); fclose(fp); return 0; }
#include#include #include struct Test { int a; char c; }; int main() { FILE *fp; struct Test data = {100,'a'}; struct Test data2; fp = fopen("./file1","w+"); int n_write = fwrite(&data,sizeof(struct Test),1,fp); fseek(fp,0,SEEK_SET); int n_read = fread(&data2,sizeof(struct Test),1,fp); printf("read %d, %cn",data2.a,data2.c); fclose(fp); return 0; }



