<span style="font-family: Arial, Helvetica, sans-serif; background-color: rgb(255, 255, 255);"> 好几天没写总结了,因为最近特别特别忙,各种驱动的代码都要我去改去测试,我主要最近主要是在搞驱动中数据,文件加密解密这块,就这块内容,便运用到了C语言最核心的两个知识点---数组和指针。</span>
加密数据和文件为什么要用到指针和数组?
首先你要打开一个文件,那么在标准C中,你可以定义一个文件描述符,标准C语言提供了fopen ,fread,fwrite,fseek等等的一些操作文件的接口,我们不妨来看看,这些接口到底是怎么使用的。
函数原型:FILE * fopen(const char * path,const char *mode);
返回值:文件顺利打开后,指向该流的文件指针就会被返回。如果文件打开失败则返回NULL,并把错误代码存在errno中。
一般而言,打开文件后会做一些文件读取或写入的动作,若打开文件失败,接下来的读写动作也无法顺利进行,所以一般在fopen()后作错误判断及处理。
mode有下列几种形态字符串:
r 以只读方式打开文件,该文件必须存在。
r+ 以可读写方式打开文件,该文件必须存在。
w 打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。
等等….
废话不多说,上代码:
#include<stdio.h>
//定义一个文件的路径
#define F_PATH "d:\\myfile\\file.dat"
int main(void)
{
FILE*fp=NULL;//需要注意,最好初始化为空,因为文件流也是指针
fp=fopen(F_PATH,"r");//以只读方式打开
if(NULL==fp)
{
return -1;//要返回错误代码
}
fclose(fp);//关闭文件描述符
fp=NULL;//需要指向空,否则会指向原打开文件地址
return 0;
}
打开方式总结:各种打开方式主要有三个方面的区别:
①打开是否为二进制文件,用“b”标识。
②读写的方式,有以下几种:只读、只写、读写、追加只写、追加读写这几种方式。
③对文件是否必须存在、以及存在时是清空还是追加会有不同的响应。
当然,我只是举了一个最简单的例子,如果你有兴趣,慢慢去尝试。
以下是linux系统编程的open 打开文件的操作:
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
int main()
{
int fd,size;
char s [ ]=”Linux Programmer!\n”,buffer[80];
fd=open(“/tmp/temp”,O_WRONLY|O_CREAT);
write(fd,s,sizeof(s));
close(fd);
fd=open(“/tmp/temp”,O_RDONLY);
size=read(fd,buffer,sizeof(buffer));
close(fd);
printf(“%s”,buffer);
}
在UNIX系统编程中,打开文件还可以用open这个函数,具体怎么用自己百度!!要养成看函数的习惯,看多了自然就会了,其实这么厚一本,6-7百页,等你会了10个,那6-7百页的东西就都会了,只是函数而已嘛,拿来用就行了,至于会不会用,那要问问你自己C语言学得怎么样,还有对操作系统的理解程度怎么样。
以上说的是打开文件的一些简单操作,当然还有读,写,偏移等等。
函数原型:
size_t fread(void *buffer, size_t size, size_t count, FILE *stream);
buffer
用于接收数据的内存地址
size
要读的每个数据项的字节数,单位是字节
count
要读count个数据项,每个数据项size个字节.
stream
输入流
返回值:实际读取的元素个数。如果返回值与count不相同,则可能文件结尾或发生错误。从ferror和feof获取错误信息或检测是否到达文件结尾。
写:size_t fwrite(const void* buffer, size_t size, size_tcount, FILE* stream);
注意:这个函数以二进制形式对文件进行操作,不局限于文本文件
返回值:返回实际写入的数据块数目
(1)buffer:是一个指针,对fwrite来说,是要获取数据的地址;
(2)size:要写入内容的单字节数;
(3)count:要进行写入size字节的数据项的个数;
(4)stream:目标文件指针;
(5)返回实际写入的数据项个数count。
偏移:
int fseek(FILE *stream, long offset, int fromwhere);
函数设置文件指针stream的位置。如果执行成功,stream将指向以fromwhere(偏移起始位置:文件头0(SEEK_SET),当前位置1(SEEK_CUR),文件尾2(SEEK_END))为基准,偏移offset(指针偏移量)个字节的位置。如果执行失败(比如offset超过文件自身大小),则不改变stream指向的位置。
返回值: 成功,返回0,失败返回-1,并设置errno的值,可以用perror()函数输出错误。
与此类似的UNIX接口有:open , read , write ,lseek,create ,opendir ,readdir,等等,慢慢学慢慢用,孰能生巧。
那么,为什么说这些知识点用到了加密解密上了,怎么用的?
比如,现在有一串密码.
I123W598DWADA88DASDAJ
这串密码需要进行加密,那么我们先用一个char型的数组将它保存起来。
Charbuffer[] = {‘ I’,1,2,3,’W’,5,9,8,’D’,’W’,’A’,’D’,’A’,8,8,’D’,’A’,’S’,’D’,’A’,’J’} ;
那如何加密?
我们可以用下面的方法来对这串数组进行简单的加密:
#include <stdio.h>
#define NR(x) (sizeof(x)/sizeof(x[0]))
unsigned char buffer[] = {'I',1,2,3,'W',5,9,8,'D','W','A','D','A',8,8,'D','A','S','D','A','J' } ;
//加密
void Crypt_Data(unsigned char buffer[]);
//解密
void Uncrypt_Data(unsigned char buffer[]);
int main(void)
{
static int i = 0 ;
//将数组加密
Crypt_Data(buffer);
//将数组解密
Uncrypt_Data(buffer);
//数组输出
for(i = 0 ; i < NR(buffer) ; i++){
if(buffer[i] < 60)
buffer[i] = buffer[i] + '0' ;
else
buffer[i] = buffer[i] ;
}
printf("%s\n",buffer);
return 0 ;
}
//加密
void Crypt_Data(unsigned char buffer[])
{
static int i = 0 ;
for(i = 0 ; i < NR(buffer) ; i++)
{
buffer[i] << 1 ; //将数组中每个数据按位左移1位
buffer[i] += 2 ; //左移完都加上2
buffer[i] += 3 ; //都加上2后都加上3
buffer[i] >> 4 ; //整体数据按位左移4位
}
}
//解密
void Uncrypt_Data(unsigned char buffer[])
{
static int i = 0 ;
for(i = 0 ; i < NR(buffer) ; i++)
{
buffer[i] >> 1 ; //按照加密的格式将数组解密出来
buffer[i] -= 2 ;
buffer[i] -= 3 ;
buffer[i] << 4 ;
}
}
运行结果:
<img src="https://img-blog.csdn.net/20160107214053180?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="" />
好像没和文件有什么关系对吧?哈哈,这就需要开动脑经啦。如果现在数据都在一个文件里,我
要将数据读出来,然后进行加密操作存在一个数组里去,接着将原来的文件的数据清空,然后再写到这个文件里去。思路就是这样子:
首先你先创建一个文件写上我说的‘ I’,1,2,3,’W’,5,9,8,’D’,’W’,’A’,’D’,’A’,8,8,’D’,’A’,’S’,’D’,’A’,’J’这串数据
接着你就可以编程进行操作了:
1 、 fopen :打开文件
2、fread :从文件里读数据存到一个buffer数组里
3、加密
4、fclose :关闭文件描述符
5、fopen以清空文件的模式接着打开这个文件。
6、lseek:偏移文件指针的位置
7、fwrite 将加密完的数据写入这个文件
8、加密完成
9、fclose:关闭文件描述符
加密操作大概就是这个样子,解密也是一样的,不用多说了。
同时,以上的在linux下开发也可以用open , read , write , lseek ,close这些接口来代替。
但是UNIX接口对于块数据效率会比较高,对于小的数据效率就低了,所以,在文件不大的情况下(小于一块),建议还是使用标准C的接口。
今天就到这里,往后遇到什么问题,再做总结