C语言中的异或是一种按位操作的计算,其计算原理为,操作数对应位上的值相同,则结果位上值为0,否则为1.异或的运算符号为^,于是真值表如下:
0^0=0
0^1=1
1^0=1
1^1=0
这样区分源数据和秘钥值,有如下四种可能:
0^0=0
0^0=0
0^1=1
1^1=0
1^0=1
1^0=1
1^1=0
0^1=1
以上是将一个值,用另一个值连续异或两次后的计算过程,可以看到,最终的值与原始值是相同的。这就是异或加密的基础原理。
异或加密是所有加密中最容易实施且代码量相对精简的一种加密方式,其原理为任意数据被同一个值(key)两次异或后,值不变。
这种特性,使得异或加密算法的加解密流程是完全相同的,也就是说加解密可以用同一个函数实现。
一、算法思路。
依次读入文件字符,并用key值对其异或,结果输入到目标文件中。
二、算法描述。
1 打开源文件及目标文件。
2 获取密钥值(key)。
3 读入一个字符。
4 对其进行异或计算。
5 结果写入目标文件。
6 重复3-5直到文件结尾。
7 关闭文件。
三、代码实现。
为使代码简洁易懂,输入文件设定为in.txt, 输出文件设定为out.txt, key值设定为0x5C。
假定所有文件及目录可读写。
#include <stdio.h>#define IN "in.txt"
#define OUT "out.txt"
#define KEY 0x5C
void scrambler(const char *in, const char *out, unsigned char key)
{
FILE *i, *o
int c
i = fopen(in, "rb")
o = fopen(out, "wb")//打开文件,因为假定可读写,所以对是否成功不做判断
while((c = fgetc(i))!=EOF)//读入字符直到文件结尾
{
c^=key//执行异或加密或解密
fputc(c,o)//写入文件
}
fclose(i)
fclose(o)
}
int main()
{
scrambler(IN,OUT,KEY)
return 0
}
但是异或加密也是有缺陷的。最大的缺陷是加密操作是二进制层面的,获取到的加密文件大部分将是不可读的乱码。这也是为什么在打开文件的时候要以二进制方式的原因。
鉴于此,在此不便给出测试样例及输入输出,可自行调试。
对于加密要求不高的完全可以自己定义规则来进行加密。这种加密是很简单很自由的,例如你在存文件的时候可以将文件中的每个字符都加上一个数,然后读取该文件的时候再每个字符相应地减去那个数,即可实现就简单的加密,这样你储存的文件看上去就是乱码了。只是这个规则太简单,规则你可以自己定,加密与解密对着来就行了。下面程序用异或操作对文件进行加密和解密
/******************设计思路******************/
//根据用户输入的加密/机密密码,
//每次都拿原文件和密码等长度的一个字符串和密码
//对应元素异或进行加密/解密
//另外因为是用异或方法,所以加密和解密就是同一个程序
//即按照同样的加密即是对文件的解密
#include
#include
#include
#include
#include
charfilename[256]//原文件
charpassword[256]//加密/解密密码
constcharfilenametemp[]="temp15435255435325432543.temp"//加密/解密中间文件
voidinputpass(char*pass)//密码输入以"******"显示
voidmain(){
FILE*fp//加密/解密的文件
FILE*fptemp//加密/解密过程临时文件
intpwdlen//密码长度
inti=0//计数器
charch=0//读入的字符
printf("请输入要加密/解密的文件名(全路径名):\n")
gets(filename)
if((fp=fopen(filename,"rb"))==NULL){
printf("找不到文件%s\n",filename)
exit(1)
}//if
printf("请输入要加密/解密的密码:\n")
inputpass(password)
pwdlen=strlen(password)
if(pwdlen==0){
printf("密码不能为空,加密/解密失败\n")
exit(1)
}//if
fptemp=fopen(filenametemp,"wb")//打开中间文件
while(1){
ch=fgetc(fp)//从原文件读入一个字符
if(feof(fp)){//已经读到文件尾
break//退出循环
}
ch^=password[i++]//对原字符和密码进行异或操作
fputc(ch,fptemp)//将异或结果写入中间文件
if(i==pwdlen){//使得原文件每和密码长度相同的固定长度异或加密
i=0
}
}//while
fclose(fp)//关闭打开原文件
fclose(fptemp)//关闭打开中间文件
remove(filename)//删除原文件
rename(filenametemp,filename)//将中间文件重命名为原文件
printf("加密/解密成功\n")//至此加密/解密成功
}
//密码输入以"******"显示
voidinputpass(char*pass){
inti=0
charc
while(isprint(c=getch())){
pass[i++]=c
//printf("*")
}
pass[i]='\0'
printf("\n")
}