文件操作

1.文件分类


2.流概念

  • 流是一个动态的概念,可以将一个字节形象地比喻成一滴水,字节在设备、文件和程序之间的传输就是流,类似于水在管道中的传输,可以看出,流是对输入输出源的一种抽象,也是对传输信息的一种抽象。通过对输入输出源的抽象,屏蔽了设备之间的差异,使程序员能以一种通用的方式进行存储操作,通过对传输信息的抽象,使得所有信息都转化为字节流的形式传输,信息解读的过程与传输过程分离。

  • C语言中,I/O操作可以简单地看作是从程序移进或移出字节,这种搬运的过程便称为流(stream)。程序只需要关心是否正确地输出了字节数据,以及是否正确地输入了要读取字节数据,特定I/O设备的细节对程序员是隐藏的。

3.文件处理方法

4.文件操作API

4.1 文件api的分类

  • 文件读写api
    fgetc fputc 按照字符读写文件
    fputs fgets 按照行读写文件 (读写配置文件)
    fread fwirte 按照块读写文件 (大数据块迁移)
    fprintf 按照格式化进行读写文件
    fprintf(fp, "%s = %s\n", pKey, pValue);

  • 文件控制api
    文件是否结束
    文件指针的定位、跳转
    fseek(fp, 0L, SEEK_SET);
    把文件指针从0位置开始,移动到文件末尾
    fseek(fp, 0L, SEEK_END);
    获取文件长度;
    length = ftell(fp);

4.2 标准文件的读写

1.文件的打开fopen()

文件的打开操作表示将给用户指定的文件在内存分配一个FILE结构区,并将该结构的指针返回给用户程序,以后用户程序就可用此FILE指针来实现对指定文件的存取操作了。当使用打开函数时,必须给出文件名、文件操作方式(读、写或读写),如果该文件名不存在,就意味着建立(只对写文件而言,对读文件则出错),并将文件指针指向文件开头。若已有一个同名文件存在,则删除该文件,若无同名文件,则建立该文件,并将文件指针指向文件开头。

fopen(char *filename,char *type);

其中*filename是要打开文件的文件名指针,一般用双引号括起来的文件名表示,也可使用双反斜杠隔开的路径名。


而*type参数表示了对打开文件的操作方式。其可采用的操作方式如下:

方式含义
“r”打开,只读
“w”打开,文件指针指到头,只写
“a”打开,指向文件尾,在已存在文件中追加
“rb”打开一个二进制文件,只读
“wb”打开一个二进制文件,只写
“ab”打开一个二进制文件,进行追加
“r+”以读/写方式打开一个已存在的文件
“w+”以读/写方式建立一个新的文本文件
“a+”以读/写方式打开一个文件文件进行追加
“rb+”以读/写方式打开一个二进制文件
“wb+”以读/写方式建立一个新的二进制文件
“ab+”以读/写方式打开一个二进制文件进行追加

当用fopen成功的打开一个文件时,该函数将返回一个FILE指针,如果文件打开失败,将返回一个NULL指针。如想打开test文件,进行写:

FILE *fp;
if((fp=fopen("test","w"))==NULL)
{
printf("File cannot be    opened\n");
exit();
}
else
printf("File opened for writing\n");
                    ……
fclose(fp);

DOS操作系统对同时打开的文件数目是有限制的,缺省值为5,可以通过修改CONFIG.SYS文件改变这个设置。


2.关闭文件函数fclose()

文件操作完成后,必须要用fclose()函数进行关闭,这是因为对打开的文件进行写入时,若文件缓冲区的空间未被写入的内容填满,这些内容不会写到打开的文件中去而丢失。只有对打开的文件进行关闭操作时,停留在文件缓冲区的内容才能写到该文件中去,从而使文件完整。再者一旦关闭了文件,该文件对应的FILE结构将被释放,从而使关闭的文件得到保护,因为这时对该文件的存取操作将不会进行。

文件的关闭也意味着释放了该文件的缓冲区。

int fclose(FILE *stream);

它表示该函数将关闭FILE指针对应的文件,并返回一个整数值。若成功地关闭了文件,则返回一个0值,否则返回一个非0值。常用以下方法进行测试:

if(fclose(fp)!=0)
{
printf("File cannot be closed\n");
exit(1);
}
else
printf("File is now closed\n");

当打开多个文件进行操作,而又要同时关闭时,可采用fcloseall()函数,它将关闭所有在程序中打开的文件。
int fcloseall();
该函数将关闭所有已打开的文件,将各文件缓冲区未装满的内容写到相应的文件中去,接着释放这些缓冲区,并返回关闭文件的数目。如关闭了4个文件,则当执行:
n=fcloseall(); 时,n应为4。

3.文件的读写

(1).读写文件中字符的函数(一次只读写文件中的一个字符):

int fgetc(FILE *stream);
int fgetchar(void);
int fputc(int ch,FILE *stream);
int fputchar(int ch);
int getc(FILE *stream);
int putc(int ch,FILE *stream);

其中fgetc()函数将把由流指针指向的文件中的一个字符读出,例如:
ch=fgetc(fp);
将把流指针fp指向的文件中的一个字符读出,并赋给ch,当执行fgetc()函数时,若当时文件指针指到文件尾,即遇到文件结束标志EOF(其对应值为-1),该函数返回一个-1给ch,在程序中常用检查该函数返回值是否为-1来判断是否已读到文件尾,从而决定是否继续。

#include "stdio.h"
main()
{
FILE *fp;
ch ch;
if((fp=fopen("myfile.tex","r"))==NULL)
{
printf("file cannot be opened\n");
exit(1);
}
    while((ch=fgetc(fp))!=EOF) fputc(ch,stdout);
fclose(fp);
}

该程序以只读方式打开myfile.txt文件,在执行while循环时,文件指针每循环一次后移一个字符位置。用fgetc()函数将文件指针指定的字符读到ch变量中,然后用fputc()函数在屏幕上显示,当读到文件结束标志EOF时,便关闭该文件。
上面的程序用到了fputc()函数,该函数将字符变量ch的值写到流指针指定的文件中去,由于流指针用的是标准输出(显示器)的FILE指针stdout,故读出的字符将在显示器上显示。又比如:
putc(ch,fp);
该函数执行结构,将把ch表示的字符送到流指针fp指向的文件中去。

在TC:
* putc()等价于fputc(),getc()等价于fgetc();
* putchar(c)相当于fputc(c,stdout);
* getchar()相当于fgetc(stdin)。


注意,这里使用char ch,其实是不科学的,因为最后判断结束标志时,是看ch!=EOF,而EOF的值为-1,这显然和char是不能比较的。所以,某些使用,我们都定义成int ch


(2).读写文件中字符串的函数

char *fgets(char *string,int n,FILE *stream);
char *gets(char *s);
int fprintf(FILE *stream,char *format,variable-list);
int fputs(char *string,FILE *stream);
int fscanf(FILE *stream,char *format,variable-list);
  • 其中fgets()函数将把由流指针指定的文件中n-1个字符,读到由指针string指向的字符数组中去,例如:
    fgets(buffer,9,fp);
    将把fp指向的文件中的8个字符读到buffer内存区,buffer可以是定义的字符数组,也可以是动态分配的内存区。

注意,fgets()函数读到’\n’就停止,而不管是否达到数目要求。同时在读取字符串的最后加上’\0’。

fgets()函数执行完以后,返回一个指向该串的指针。如果读到文件尾或出错,则均返回一个空指针NULL,所以长用feof()函数来测定是否到了文件尾或者是ferror()函数来测试是否出错,例如下面的程序用fgets()函数读test.txt文件中的第一行并显示出来:

#include "stdio.h"
main()
{
FILE *fp;
char str[128];
if((fp=fopen("test.txt","r"))==NULL)
{
printf("cannot open file\n");
exit(1);
}
while(!feof(fp))
{
if(fgets(str,128,fp)!=NULL) printf("%s",str);
}
fclose(fp);
}
  • gets()函数执行时,只要未遇到换行符或文件结束标志,将一直读下去。因此读到什么时候为止,需要用户进行控制,否则可能造成存储区的溢出。

  • fputs()函数向指定文件写入一个由string指向的字符串,’\0’不写入文件。

  • fprintf()和fscanf()同printf()和scanf()函数类似,不同之处就是printf()函数是向显示器输出,fprintf()则是向流指针指向的文件输出;fscanf()是从文件输入。

下面程序是向文件test.dat里输入一些字符:

#include<stdio.h>
main()
{
    char *s = "That's good news";
    int i = 617;
    FILE *fp;
    fp = fopen("test.dat", "w");
    fputs("Your score of TOEFLis", fp);
    fputc(':', fp);
    fprintf(fp, "%d\n", i);
    fprintf(fp, "%s", s);
    fclose(fp);
}

用DOS的TYPE命令显示TEST.DAT的内容如下所示:
屏幕显示
Your score of TOEFL is: 617
That’s good news

下面的程序是把上面的文件test.dat里的内容在屏幕上显示出来:

#include<stdio.h>
main()
{
    char *s, m[20];
    int i;
    FILE    *fp;
    fp=fopen("test.dat", "r");    
    fgets(s, 24, fp);             
    printf("%s", s);
    fscanf(fp, "%d", &i);         
    printf("%d", i);
    putchar(fgetc(fp));           
    fgets(m, 17, fp);             
    puts(m);                      
    fclose(fp);
    getch();
}

运行后屏幕显示:
Your score of TOEFL is: 617
That’s good news

4.清除和设置文件缓冲区

(1).清除文件缓冲区函数:

int fflush(FILE *stream);
int flushall();
  • fflush()函数将清除由stream指向的文件缓冲区里的内容,常用于写完一些数据后,立即用该函数清除缓冲区,以免误操作时,破坏原来的数据。

  • flushall()将清除所有打开文件所对应的文件缓冲区。

(2).设置文件缓冲区函数

 void setbuf(FILE *stream,char *buf);
 void setvbuf(FILE *stream,char *buf,int type,unsigned size);

这两个函数将使得打开文件后,用户可建立自己的文件缓冲区,而不使用fopen()函数打开文件设定的默认缓冲区。

  • 对于setbuf()函数,buf指出的缓冲区长度由头文件stdio.h中定义的宏BUFSIZE的值决定,缺省值为512字节。当选定buf为空时,setbuf函数将使的文件I/O不带缓冲

  • 而对setvbuf函数,则由malloc函数来分配缓冲区。参数size指明了缓冲区的长度(必须大于0),而参数type则表示了缓冲的类型,其值可以取如下值:

type 值含义
_IOFBF文件全部缓冲,即缓冲区装满后,才能对文件读写
_IOLBF文件行缓冲,即缓冲区接收到一个换行符时,才能对文件读写
_IONBF文件不缓冲,此时忽略buf,size的值,直接读写文件,不再经过文件缓冲区缓冲

5.文件的随机读写函数

前面介绍的文件的字符/字符串读写,均是进行文件的顺序读写,即总是从文件的开头开始进行读写。这显然不能满足我们的要求,C语言提供了移动文件指针和随机读写的函数,它们是:

(1).移动文件指针函数:

    long ftell(FILE *stream);
    int rewind(FILE *stream);
    fseek(FILE *stream,long offset,int origin);
  • 函数ftell()用来得到文件指针离文件开头的偏移量。当返回值是-1时表示出错。
  • rewind()函数用于文件指针移到文件的开头,当移动成功时,返回0,否则返回一个非0值。
  • fseek()函数用于把文件指针以origin为起点移动offset个字节,其中origin指出的位置可有以下几种:

    origin数值代表的具体位置
    SEEK_SET0文件开头
    SEEK_CUR1文件指针当前位置
    SEEK_END2文件尾

例如:
fseek(fp,10L,0);
把文件指针从文件开头移到第10字节处,由于offset参数要求是长整型数,故其数后带L

fseek(fp,-15L,2);
把文件指针从文件尾向前移动15字节。


(2).文件随机读写函数

int fread(void *ptr,int size,int nitems,FILE *stream);
int fwrite(void *ptr,int size,int nitems,FILE *stream);
  • fread()函数从流指针指定的文件中读取nitems个数据项,每个数据项的长度为size个字节,读取的nitems数据项存入由ptr指针指向的内存缓冲区中,

在执行fread()函数时,文件指针随着读取的字节数而向后移动,最后移动结束的位置等于实际读出的字节数。该函数执行结束后,将返回实际读出的数据项数,这个数据项数不一定等于设置的nitems,因为若文件中没有足够的数据项,或读中间出错,都会导致返回的数据项数少于设置的nitems。当返回数不等于nitems时,可以用feof()ferror()函数进行检查。

  • fwrite()函数从ptr指向的缓冲区中取出长度为size字节的nitems个数据项,写入到流指针stream指向的文件中,执行该操作后,文件指针将向后移动,移动的字节数等于写入文件的字节数目。该函数操作完成后,也将返回写入的数据项数。

4.3非标准文件的读写

这类函数最早用于UNIX操作系统,ANSI标准未定义,但有时也经常用到,DOS 3.0以上版本支持这些函数。它们的头文件为io.h。
由于我们不常用这些函数,所以在这里就简单说一下。

1.文件的打开和关闭

  • open()函数的作用是打开文件,其调用格式为:
    int open(char *filename, int access);
    该函数表示按access的要求打开名为filename的文件,返回值为文件描述字(符),其中access有两部分内容:
    基本模式和修饰符, 两者用” |”方式连接。修饰符可以有多个, 但基本模式只能有一个。

access的规定

基本模式含义
O_RDONLY只读
O_WRONLY只写
O_RDWR读写
修饰符含 义
O_APPEND文件指针指向末尾
O_CREAT文件不存在时创建文件, 属性按基本模式属性
O_TRUNC若文件存在, 将其长度缩为0, 属性不变
O_BINARY打开一个二进制文件
O_TEXT打开一个文字文件

open()函数打开成功, 返回值就是文件描述字的值(非负值), 否则返回-1。

  • close()函数的作用是关闭由open()函数打开的文件, 其调用格式为:
    int close(int handle);
    该函数关闭文件描述字handle相连的文件。

2.读写函数

  • read的调用
int read(int handle, void *buf, int count);

read()函数从handle(文件描述字)相连的文件中,读取count个字节放到buf所指的缓冲区中,

返回值为实际所读字节数, 返回-1表示出错。返回0 表示文件结束。

  • write()函数的调用格式为:
int write(int handle, void *buf, int count);

write()函数把count个字节从buf指向的缓冲区写入与handle相连的文件中, 返回值为实际写入的字节数。

3.随机定位函数

  • lseek()函数的调用格式为:
    int lseek(int handle, long offset, int fromwhere);
    该函数对与handle相连的文件位置指针进行定位,功能和用法与fseek()函数相同。

  • tell()函数的调用格式为:
    long tell(int handle);
    该函数返回与handle相连的文件位置指针相距文件开头的距离,功能和用法与ftell()相同


5.相关代码

5.1 单个字符操作

#define _CRT_SECURE_NO_WARNINGS


#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main(void)
{
    FILE *fp = NULL;

    char * filename1 = "c:/2.txt";

    fp = fopen(filename1,"w+");
    char a[27] = "abcdefghijklmnopqrstuvwxyz";
    int i = 0;




    if (fp == NULL)
    {
        printf("func fopen fialed!\n");
        return -1;
    }

    for (i = 0; i < strlen(a);i++)
    {
        fputc(a[i],fp);
    }
    fclose(fp);
    //在读写文件之前要确保文件指针的位置合法且文件有内容
    fp = fopen(filename1, "r+");
    while (!feof(fp))
    {
        fputc(fgetc(fp),stdout);
    }
    printf("\nopen success\n");

    fclose(fp);

    system("PAUSE");
    return 1;
}

5.2按照行的方式进行操作

#define _CRT_SECURE_NO_WARNINGS


#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main(void)
{
    FILE *fp = NULL;

    char * filename1 = "c:/22.txt";
    char a[30] = "abcdefghi\njklmnopq\nrstuvwxyz\n";
    char bufer[1024];
    int i = 0;



    fp = fopen(filename1, "w+");
    if (fp == NULL)
    {
        printf("func fopen fialed!\n");
        return -1;
    }


    fputs(a,fp);//不会把'\0'写入到文件中

    fclose(fp);
    //在读写文件之前要确保文件指针的位置合法且文件有内容
    fp = fopen(filename1, "r+");
    while (!feof(fp))
    {
        if (fgets(bufer, 1024, fp) != NULL)//c函数库会一行一行的拷贝数据到bufer指针所指的内存空间并且变成C风格的字符串,
        {                   //会自动追加行结束符,而且会把\n也拷贝到缓冲区
            printf("%s", bufer);
        }
    }
    printf("\nopen success\n");

    fclose(fp);

    system("PAUSE");
    return 1;
}

对于fgets的使用要格外小心,因为读取文件容易出错返回空指针,只有当返回非空指针的时候才认为是读取正确的,所以使用下面的方式确保每次操作的都是正确读取到的数据:

    while (!feof(fp))
    {
        if (fgets(bufer, 1024, fp) != NULL)//c函数库会一行一行的拷贝数据到bufer指针所指的内存空间并且变成C风格的字符串,
        {                   //会自动追加行结束符,而且会把\n也拷贝到缓冲区
            printf("%s", bufer);
        }
    }

5.3按照块的方式操作文件

#define _CRT_SECURE_NO_WARNINGS


#include <stdio.h>
#include <stdlib.h>
#include <string.h>

typedef struct _teacher
{
    char name[64];
    int age;

}Teacher;

int main(void)
{
    FILE *fp = NULL;

    char * filename1 = "c:/23.dat";

    Teacher tArray[3];
    Teacher testArray[3];
    int i = 0;

    for (i = 0; i < 3; i++)
    {
        sprintf(tArray[i].name,"%d%d%d",i+1,i+1,i+1);
        tArray[i].age = i + 31;
    }


    fp = fopen(filename1, "wb");//二进制写的方式打开
    if (fp == NULL)
    {
        printf("func fopen fialed!\n");
        return -1;
    }

    for (i = 0; i < 3; i++)
    {
        fwrite(&tArray[i], sizeof(Teacher), 1, fp);
    }
    fclose(fp);




    fp = fopen(filename1, "rb");//二进制读的方式打开
    if (fp == NULL)
    {
        printf("func fopen fialed!\n");
        return -1;
    }

    for (i = 0; i < 3; i++)
    {
        if(fread(&testArray[i], sizeof(Teacher), 1, fp)>0)//只有在正确读取以后才可以使用读取到的数据
            printf("name:%s,age:%d\n", testArray[i].name, testArray[i].age);
    }

    fclose(fp);

    system("PAUSE");
    return 1;
}

6.printf补充资料

“格式描述串”是由一系列的”格式转换说明符号”组成,格式转换说明符号的描述形式如下:
% [+][-] 0 m[.n] [输出精度] <形式字母>
(1)形式字母:制定输出格式,如表
d:十进制整型数
i:十进制整型数
x:十六进制整型数
o:八进制整型数
u:无符号十进制整型数
c:单个字符;
s:字符串
e:指数形式的浮点数
f:小数形式的浮点数
g:e和f中比较短的一种
p:显示变量所在的内存地址
n:它不是向printf()传递格式化信息,而是令printf()把自己已经输出的字符总数放到相应变元指
的整形变量中
%:符号%本身;
(2):输出精度如果形式字母是d,x,o.u,则可以指定如下两类精度
l:long型输出精度
h:short型输出精度
默认时为int型精度
如:long x=123454578;printf(“%d”,x);
如果形式字母为e,f,g的时候,则指定l的 时候为double精度,不指定为float精度;
(3):m[.n]指定输出长度,如果输出的是实例,则m表示该项输出占用字符位置的总长度,n表示小数部分的字

符长度,如float x=4.56;printf(“%7.4f”,x);
(4)0:指定不被使用的空位置填写0,入股不指定使用0,则不使用的位置为空白.该项仅仅对树枝输出时才可

以指定,对字符串输出不用指定.例如
int x=234;
printf(“%05d”,x);//00234
printf(“%5d”,x);//**234
(5)[+][-]:指定输出位置,如果指定+或者缺省时为右对齐,如果为”-“的时候为左对齐;

7.配置文件

  • 头文件
#ifndef __CFG_OP_H__
#define __CFG_OP_H__

#ifdef __cplusplus
extern "C" {
#endif

int GetCfgItem(const char * filename, char *pKey, char * pValue, int * pValueLen);

int SetCfgItem(const char * filename, char *pItemName, char * pItemValue, int  pItemValueLen);



#ifdef __cplusplus
}
#endif


#endif
  • 实现文件
#define _CRT_SECURE_NO_WARNINGS


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "cfg_op.h"

#define MAX_LINE  2048


int trimeSpace(char *src, char *dst,int *len)
{
    int ret = 0;
    int head = 0, tail = 0;
    char *tmpStr = src;

    if (NULL == src || dst == NULL)
    {
        printf("%s is error!(NULL == src || dst == NULL)\n",__FUNCTION__);
        return ret = -1;
    }

    while (*(tmpStr + head) != '\0' && isspace(*(tmpStr + head)))
    {
        head++;
    }

    tail = strlen(src) - 1;

    while (*(tmpStr + tail) != '\0' && isspace(*(tmpStr + tail)))
    {
        tail--;
    }

    strncpy(dst,tmpStr+head,tail-head+1);
    dst[tail - head + 1] = '\0';
    *len = tail - head +1;

    return ret;
}

int GetCfgItem(const char * filename, char *pKey, char * pValue, int * pValueLen)
{
    int ret = 0;
    FILE *fp = NULL;
    char lineBuffer[MAX_LINE];
    char *pTemp = NULL;

    fp = fopen(filename,"r");
    if (NULL == fp)
    {
        ret = -1;
        return ret;
    }

    while (!feof(fp))
    {
        memset(lineBuffer, 0, sizeof(lineBuffer));
        fgets(lineBuffer, MAX_LINE, fp);
        //printf("lineBuffer:%s\n",lineBuffer);

        pTemp = strstr(lineBuffer,pKey);
        if (NULL == pTemp)
        {
            continue;
        }

        pTemp = strchr(lineBuffer, '=');
        if (pTemp == NULL)
        {
            continue;
        }

        pTemp += 1;

        trimeSpace(pTemp, pValue, pValueLen);
        break;
    }

    return 0;
}

int SetCfgItem(const char * filename, char *pItemName, char * pItemValue, int  pItemValueLen)
{
    int ret = 0;
    FILE *fp = NULL;
    int length = 0;
    char file_bufer[1024 * 8] = {0};
    char linebuf[MAX_LINE] = { 0 };
    char *pTemp = NULL;
    char iTag = 0;

    if (filename == NULL || pItemName == NULL || pItemValue == NULL)
    {
        ret = -1;
        printf("%s is error!(filename == NULL || pItemName == NULL || pItemValue == NULL)\n", __FUNCTION__);
        return ret;
    }

    fp = fopen(filename, "r+");
    if (NULL == fp)
    {
        printf("can not open the file with r+%s\n",filename);
        fp = fopen(filename, "w+t");

        if (fp == NULL)
        {
            ret = -2;
            printf("%s is error!open failed!\n", __FUNCTION__);
            return ret;
        }
    }


    fseek(fp,0L,SEEK_END);
    length = ftell(fp);
    fseek(fp, 0L, SEEK_SET);

    if (length > 8 * 1024)
    {
        ret = -3;
        printf("This file is too large!\n");
        fclose(fp);
        return ret;
    }

    while (!feof(fp))
    {
        memset(linebuf,0,sizeof(linebuf));
        pTemp = fgets(linebuf,MAX_LINE,fp);
        if (NULL == pTemp)
        {
            printf("fgets is failed!\n");
            break;
        }

        pTemp = strstr(linebuf, pItemName);
        if (NULL == pTemp)
        {
            strcat(file_bufer,linebuf);
            continue;
        }
        else
        {
            sprintf(linebuf, "%s=%s", pItemName, pItemValue);
            strcat(file_bufer, linebuf);
            iTag = 1;
        }
    }

    if (0 == iTag)
    {
        fprintf(fp, "%s = %s\r\n", pItemName,pItemValue);
    }
    else{
        if (NULL != fp)
        {
            fclose(fp);
            fp = NULL;
        }

        fp = fopen(filename,"w+t");
        if (fp == NULL)
        {
            ret = -2;
            printf("%s is error!open failed!\n", __FUNCTION__);
            return ret;
        }

        fputs(file_bufer, fp);

    }
    if (NULL != fp)
    {
        fclose(fp);
        fp = NULL;
    }
    return ret;
}
  • 测试文件
#define _CRT_SECURE_NO_WARNINGS


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "cfg_op.h"

#define CFGNAME "c:/lzj.txt"

void ShowMenu()
{
    printf("======================\n");
    printf("1.测试写配置文件\n");
    printf("2.测试读配置文件\n");
    printf("0.退出程序\n");
    printf("======================\n");
}


int TestSetCfg()
{
    char key[1024];
    char value[1024];
    int ret = 0;

    printf("\n请键入key:");
    scanf("%s",key);

    printf("\n请键入value:");
    scanf("%s", value);

    ret = SetCfgItem(CFGNAME, key, value, strlen(value));

    if (ret != 0)
    {
        printf("%s is error\n", __FUNCTION__);
        return ret;
    }

    printf("你的输入是:%s=%s\n", key, value);
    return ret;
}

int TestGetCfg()
{
    char key[1024];
    char value[1024];
    int ret = 0;
    int len = 0;

    printf("\n请键入key:");
    scanf("%s", key);

    ret = GetCfgItem(CFGNAME, key, value, &len);

    if (ret != 0)
    {
        printf("%s is error\n", __FUNCTION__);
        return ret;
    }

    printf("%s=%s\nlen:%d\n",key,value,len);
    return ret;
}
int main(void)
{
    int chose;
    for (;;)
    {
        /*显示一个菜单*/
        ShowMenu();

        scanf("%d",&chose);
        switch (chose)
        {
        case 1:
            TestSetCfg();
            break;
        case 2:
            TestGetCfg();
            break;
        case 0:
            exit(0);
            break;

        default:
            exit(0);
            break;
        }
    }

    system("PAUSE");
    return 1;
}


8.文件加解密

示例代码:

#define _CRT_SECURE_NO_WARNINGS


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "des.h"


int FileSymEnc(const char *pfile1,const char *pfile2)
{
    int ret = 0, tmplen1 = 0, tmplen2 = 0;
    FILE *fp1=NULL, *fp2=NULL;
    unsigned char buffer[4096] = {0};
    unsigned char crypt[4096] = { 0 };
    int cryptlen = 0;


    fp1 = fopen(pfile1,"rb");
    if (NULL == fp1)
    {
        goto END;
    }


    fp2 = fopen(pfile2, "wb");
    if (NULL == fp2)
    {
        goto END;
    }


    while (!feof(fp1))
    {
        tmplen1 = fread(buffer, 1, 4096, fp1);

        ret = DesEnc(buffer,tmplen1,crypt,&cryptlen);
        if (ret != 0)
        {
            ret = -3;
            printf("加密出错\n");
            goto END;
        }



        if (feof(fp1))
        {
            break;
        }
        if (tmplen1 < 4096)
        {
            break;
        }
        tmplen2 = fwrite(crypt, 1, cryptlen, fp2);
        if (cryptlen != tmplen2)
        {
            ret = -2;
            printf("请检查磁盘是否已满\n");
            goto END;
        }
    }


    tmplen2 = fwrite(crypt, 1, cryptlen, fp2);
    if (cryptlen != tmplen2)
    {
        ret = -2;
        printf("请检查磁盘是否已满\n");
        goto END;
    }
END:
    if (fp1 != NULL)
    {
        fclose(fp1);
    }
    if (fp2 != NULL)
    {
        fclose(fp2);
    }
    return ret;
}

int FileRecovery(const char *pfile1, const char *pfile2)
{
    int ret = 0, tmplen1 = 0, tmplen2 = 0;
    FILE *fp1 = NULL, *fp2 = NULL;
    unsigned char buffer[4096] = { 0 };
    unsigned char crypt[4096] = { 0 };
    int cryptlen = 0;


    fp1 = fopen(pfile1, "rb");
    if (NULL == fp1)
    {
        goto END;
    }


    fp2 = fopen(pfile2, "wb");
    if (NULL == fp2)
    {
        goto END;
    }


    while (!feof(fp1))
    {
        tmplen1 = fread(buffer, 1, 4096, fp1);

        ret = DesDec(buffer, tmplen1, crypt, &cryptlen);
        if (ret != 0)
        {
            ret = -3;
            printf("解密出错\n");
            goto END;
        }



        if (feof(fp1))
        {
            break;
        }
        if (tmplen1 < 4096)
        {
            break;
        }
        tmplen2 = fwrite(crypt, 1, cryptlen, fp2);
        if (cryptlen != tmplen2)
        {
            ret = -2;
            printf("请检查磁盘是否已满\n");
            goto END;
        }
    }


    tmplen2 = fwrite(crypt, 1, cryptlen, fp2);
    if (cryptlen != tmplen2)
    {
        ret = -2;
        printf("请检查磁盘是否已满\n");
        goto END;
    }
END:
    if (fp1 != NULL)
    {
        fclose(fp1);
    }
    if (fp2 != NULL)
    {
        fclose(fp2);
    }
    return ret;
}

int main(void)
{
    int ret = 0;
    const char * file1 = "c:/22.txt";
    const char * file2 = "c:/22enc.txt";
    const char * file3 = "c:/22dec.txt";

    ret = FileSymEnc(file1,file2);
    if (ret != 0)
    {
        ret = -2;
        printf("%s failed!\n",__FUNCTION__);
    }

    ret = FileRecovery(file2, file3);
    if (ret != 0)
    {
        ret = -2;
        printf("%s failed!\n", __FUNCTION__);
    }

    system("PAUSE");
    return 1;
}

其中的加密解密接口在其他文件实现,属于开源软件,声明为:

int DesEnc(
        unsigned char *pInData,
        int            nInDataLen,
        unsigned char *pOutData,
        int           *pOutDataLen);

//用户使用函数des解密
int DesDec(
       unsigned char *pInData,
       int            nInDataLen,
       unsigned char *pOutData,
       int           *pOutDataLen);

秘钥定义为:

#define  USER_PASSWORD_KEY "abcd1234"

注意:C语言中函数的返回值一定要明确给出,否则得到的是未知值

Logo

CSDN联合极客时间,共同打造面向开发者的精品内容学习社区,助力成长!

更多推荐