欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

C语言基础及指针⑧文件IO

程序员文章站 2022-11-05 12:46:31
在结构体与指针中 , 我们了解到结构体与java中的类相似 , 也是一种自定义类型数据结构 , 也学习了结构的各种用法 , 以及简单的应用 。 在编写应用程序的时候 , 文件io操作是不可避免的 ,...

在结构体与指针中 , 我们了解到结构体与java中的类相似 , 也是一种自定义类型数据结构 , 也学习了结构的各种用法 , 以及简单的应用 。

在编写应用程序的时候 , 文件io操作是不可避免的 , 小到日志本地化收集 , 大到数据格式化存储 , 都需要使用文件io , 来操作文件流进行数据处理 。在使用java开发的时候 , 我们有file类和filereader,filewriter类来搭配使用 , 也有fileinputstream,fileoutputstream和bufferedinputstream,bufferedoutputstream金牌组合 。使得java中的文件io很方便 , 下面我们就来看看简单的java文件io示例:

     // 读取文件中的字符
    private static void readstring() throws exception{
        file _file = new file("e:"+separetor+"a.txt") ;
        if (!_file.exists()) {
            boolean createstatuts = _file.createnewfile() ;
            if (createstatuts) {
                system.out.println("创建了一个新文件 ,并且创建成功了");
            }else {
                system.out.println("创建新文件失败");
            }
        }

        // 创建输入流
        inputstream fileinputstream = new fileinputstream(_file) ;
        byte[] bytes = new byte[1024] ;
        int len = -1;
        stringbuffer buffer = new stringbuffer() ;
        while ((len = fileinputstream.read(bytes)) != -1) {
            buffer.append(new string(bytes,0,len)) ;
        }
        // 关闭输入流
        fileinputstream.close() ;

        system.out.println(buffer.tostring());
    }

    // 将字符串写入文件
    private static void writestring() throws exception{
        file _file = new file("e:"+separetor+"a.txt") ;
        if (!_file.exists()) {
            boolean createstatuts = _file.createnewfile() ;
            if (createstatuts) {
                system.out.println("创建了一个新文件 ,并且创建成功了");
            }else {
                system.out.println("创建新文件失败");
            }
        }

        // 创建输出流
        outputstream fileoutputstream = new fileoutputstream(_file) ;
        string info = "《op 青空 pure rouge 君吻》《君吻 その目に映るもの piano》" ;
        fileoutputstream.write(info.getbytes()) ;
        // 关闭输出流
        fileoutputstream.close() ;

    }

文件io操作步骤:

1.创建一个file对象(需要操作的文件)
2.构建输入输出流
3.创建缓冲区 , 缓存读写数据 (将流数据读入到内存或写入到磁盘)
3.关闭流 (关闭文件流)

语言都是相通的 , 在c语言中文件io的操作也是如上述几步 , 下面我们就一起来看看:

/*读取文本文件*/
void readtextfile() {
    char* path = "c:\\users\\zeno\\documents\\visual studio 2015\\projects\\hello_c\\hello_c\\structpointer.c";

    // 打开文件
    file* fp = fopen(path, "r");
    if (fp == null)
    {
        printf("打开文件失败\n");
        return;
    }
    // 字符缓冲区 , 每次读一个字符串 , 都会缓存到字符数组中
    char buffer[1024];
    while (fgets(buffer, 1024, fp)) {
        printf("%s", buffer);
    }

    // 关闭文件流
    fclose(fp);

/*写入文本文件*/
void writetextfile() {
    char* path = "e:\\document\\write.txt";

    char* content = "如果 爱情是一场花火 ,一闪即逝的花火,我也要去追求\n如果 爱情是一场花火 ,一闪即逝的花火,我也要去追求\n";

    // 打开文件
    /*
        打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。
    */
    file* fp = fopen(path, "w");
    if (fp == null)
    {
        printf("打开文件失败\n");
        return;
    }
    // 写入文件
    fputs(content, fp);

    // 关闭文件流
    fclose(fp);
}
}

首先使用fopen函数得到一个文件指针 , 操作符r为读取文件流 , 构建了一个buffer数据缓冲区 , 通过fgets函数循环读取文件数据 , fclose函数关闭文件流 。在操作文件io的时候 , 最重要的函数 , 莫过于fopen函数了 , 首先我们来看一下fopen函数的定义:

_acrtimp file* __cdecl fopen(
    _in_z_ char const* _filename,
    _in_z_ char const* _mode
    );

我们发现fopen函数 , 需要传入文件全路径名称 , 还有一个_mode , 这个是文件操作模式 , c语言中文件操作主要依靠操作模式来辨别是输入流还是输出流的 。
下面列举一些常用的操作模式:

mode有下列几种形态字符串:

r 打开只读文件,该文件必须存在。

r+ 打开可读写的文件,该文件必须存在。

w 打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。

w+ 打开可读写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件。

a 以附加的方式打开只写文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留。

a+ 以附加方式打开可读写的文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾后,即文件原先的内容会被保留。

值得注意的是 , 上述操作模式是针对文本文件的 , 如果要操作二进制文件 , 则需要在上述操作符后面加上b , 如rb,wb,ab , 等等 。

不论是文本文件的操作还是字符文件的操作 , 都是 , 打开文件 , 创建缓冲区 , 读写文件 。

二进制文件读写

/*读写二进制文件 -- 复制文件*/
void filecopy() {
    char* currentpath = "e:\\android_pdf\\研磨设计模式.pdf";
    char* destpath = "e:\\android_pdf\\研磨设计模式_new.pdf";

    // 打开文件
    file* currentfile_p = fopen(currentpath, "rb");
    file* destfile_p = fopen(destpath, "wb");

    // 先读取再写入
    int buffer[1024]; // 数据缓冲区
    int len; // 每次读取数据的长度
    while ((len = fread(buffer,sizeof(int),1024,currentfile_p)) != eof)
    {
        // 将缓冲区里的内容写入到文件中
        fwrite(buffer, sizeof(int), len, destfile_p);
    }

    // 关闭流
    fclose(destfile_p);
    fclose(currentfile_p);
}

读写二进制和读写文本文件没多少区别 , 最大的区别就是fopen函数中的模式的不同 , 文本文件是r,w , 二进制文件是rb,wb 。

了解了文件io的基本操作 , 我们使用文件io流写一个解密的小程序。我们知道 , 所有的文件都是以二进制存储的 , 我们看的文本文件, 图片文件 , 视频文件 , 都是以二进制存储在磁盘上的 , 那么 , 我们可以将文件读取出来 , 进行二进制运算 , 就可以将文件加密解密了 。下面我们通过^异或运算来进行文件的加密解密 , 异或运算的规则如下:

0 ^ 1 得 1
1 ^ 1 得 0
0 ^ 0 得 0
1 ^ 0 得 1

相同为0 不同为1 , 例如 , 我们将4这个数加密 , 异或的数是5 , 下面我们来看看运算:

4的二进制是:0100
5的二进制是:0101
异或运算结果(加密):4 ^ 5  == 0001    
异或运算结果(解密): 0001 ^ 0101 == 0100   
由上述可见 , ^一次为加密 , 再^一次就是解密

代码示例如下:

文件加密

/*加密文件*/
void encryptfile() {

    // 待加密文件路径
    char* normal_path = "e:\\poto\\xj.jpg";
    // 加密后文件路径
    char* encrypt_path = "e:\\poto\\xj_encrypt.jpg";

    //打开文件
    file* normal_fp = fopen(normal_path, "rb");
    file* encrypt_fp = fopen(encrypt_path, "wb");
    // 读文件
    int buffer;
    while ((buffer = fgetc(normal_fp)) != eof) {
        // 写入文件
        fputc(buffer ^ 8, encrypt_fp);
    }

    printf("文件加密成功\n");

    // 关闭流
    fclose(encrypt_fp);
    fclose(normal_fp);
}

文件解密

/*文件解密*/
void decryptfile() {
    // 加密文件路径
    char* encrypt_path = "e:\\poto\\xj_encrypt.jpg";
    // 解密文件路径
    char* decrypt_path = "e:\\poto\\xj_deencrypt.jpg";

    // 打开文件
    file* encrypt_fp = fopen(encrypt_path, "rb");
    file* decrypt_fp = fopen(decrypt_path, "wb");

    // 读取文件
    int buffer;
    while ((buffer = fgetc(encrypt_fp)) != eof) {
        // 写文件
        fputc(buffer ^ 8, decrypt_fp);
    }

    printf("文件解密成功\n");

    // 关闭流
    fclose(decrypt_fp);
    fclose(encrypt_fp);
}

了解了文件加密的原理 , 我们也可以推导出其他形式的加密 , 如带密码的文件加密解密 , 混合文件加密解密等等 。

不知不觉c语言基础系列已经写了快十篇了 , 也快告一段落了 , 有了这些基础知识 , 我们就可以去分析分析jni.h头文件了 , 下一个系列是jni开发系列 , 我们学c语言就是为了能和java打交道 , 那么下个系列我们就来学习c与java的桥梁 , jni (java native interface)技术 。