您的位置:首页 > 其它

古典密码之hill密码的加密与解密程序实现

2016-03-16 11:15 627 查看
*欢迎阅读小明哥的博客*这里主要介绍的是:古典密码之 hill密码加密解密过程的编程实现。
首先,请看对我对hill密码做的简单介绍。
hill密码是古典密码中多表代换密码部分的重要一环,以下的介绍节选自百度,想要深入了解的请查阅书籍补充相关知识。
原理:希尔密码(Hill Password)是运用基本矩阵论原理的替换密码,由Lester S. Hill在1929年发明。每个字母当作26进制数字:A=0, B=1, C=2... 一串字母当成n维向量,跟一个n×n的矩阵相乘,再将得出的结果模26。注意用作加密的矩阵(即密匙)在\mathbb_^n必须是可逆的,否则就不可能译码。只有矩阵的行列式和26互质,才是可逆的。
需要的知识储备:

1)线性代数基础知识.2) 初等数论基础知识.约定:
1)希尔密码常使用Z26字母表,在此贴中,我们也以Z26最为字母表进行讲解.在附带源码中有两种字母表选择.
2) 大家都知道最小的质数是2,1 既不是质数也不是合数. 在此我们定义1对任何质数的模逆为其本身.因为对于任意质数n,有: 1*1 % n = 1 的. 也应该是很好理解的.过程:1)加密:密文=明文*密钥矩阵 (注:明文要被分割成与密钥维数相同的一维行列式)
2)解密:明文=密文*密钥矩阵的逆 (注:要求与加密过程相同)加密解密过程如下图:

例:



加密过程:

解密:



对上述过程进行编程,主要的函数声明如下:
/*
*
*  头文件名称:hillcrypto.h
*  实现文件名称:hillcrypto.cpp
*  项目名称:多表代换密码之hill密码
*  作者:邹明
*  完成时间:2016.3.14
*
*/

#ifndef  __HILLCRTPTO_H__
#define __HILLCRTPTO_H__

#include<iostream>
using namespace std;

#include<assert.h>
#include <iomanip>

#define ROW 4  //密钥行数为4
#define COV 4   //密钥列数为4

void InputKeys(float keys[ROW][COV]);   //输入密钥
void InputWords(char *words);        //输入明文
void InputObwords(char *words);      //输入密文
void PopKeys(float keys[ROW][COV]);     //输出密钥
void Encryption(float keys[ROW][COV], char *words, char *crypto);   //明文加密
void Decode(float keys[ROW][COV], char *words, char *crypto);     //密文解密
bool Gauss(float A[ROW][COV], float B[ROW][COV], int n);          //高斯消去法求逆矩阵
void 	ObMatrix(float a[ROW][COV], float b[ROW][COV], int n);      //求密钥逆矩阵
void menu();   //菜单

#endif

函数实现过程中的主函数实现以及菜单函数实现如下:
/*   实现文件名称:hillcrypto.cpp    */
#include"hillcrypto.h"

int main()
{
menu();   //菜单+选择
system("pause");
return 0;
}

void menu()
{
float keys[ROW][COV] = { 8, 6, 9, 5, 6, 9, 5, 10, 5, 8, 4, 9, 10, 6, 11, 4 };  //加密矩阵(默认密钥)
float obkeys[ROW][COV] = { 0 };  //解密矩阵 (密钥逆矩阵)
char words[100] = { 0 };
char crypto[100] = { 0 };
char obwords[100] = { 0 };
bool flag = true;   //菜单选择
bool chose = false;    //密钥选择
char cn = 0;
while (flag)
{
int n = 0;
cout << endl;
cout << "\t~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~" << endl;
cout << "\t\t\t1.输入密钥" << endl;
cout << "\t\t\t2.明文加密" << endl;
cout << "\t\t\t3.密文解密" << endl;
cout << "\t\t\t4.退出" << endl << endl;
cout << "\t~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~" << endl;
cout << "请选择->:";
cin >> n;
switch (n)
{
case 1:
system("cls");
cout << "默认密钥为:";
PopKeys(keys);
cout << "请问您要重新输入密钥? y/n" << endl << "请选择->:";
cin >> cn;
if ((cn == 'y') || (cn == 'Y'))
{
InputKeys(keys);   //输入密钥
}
else if ((cn == 'n') || (cn == 'N'))
{
cout << "感谢您选择使用默认密钥!" << endl;
}
else
cout << "输入有误,请重新选择!" << endl;
system("pause");
break;
case 2:
system("cls");
InputWords(words);  //输入明文
Encryption(keys, words, crypto);  //加密
cout << "密文是->:" << crypto << endl;
system("pause");
break;
case 3:
system("cls");
InputObwords(crypto);  //输入密文
ObMatrix(keys, obkeys, COV);  //计算解密矩阵
Decode(obkeys, obwords, crypto);  //解密
cout << "明文是->:" << obwords << endl;
system("pause");
break;
case 4:
system("cls");
cout << endl << endl << endl;
cout << setw(15) << "谢谢使用!" << endl;
flag = false;
system("pause");
break;
default:
cout << "选择有误,请重新选择!" << endl;
system("pause");
break;
}
}
}
输入明文函数和输入密文函数:
void InputWords(char *words)  //输入明文
{
assert(words);
cout << "请输入明文:";
char *start = words;
int flag = 1;
getchar();
while (flag)
{
*words = getchar();
words++;
if (*(words - 1) == '\n')
{
*words = '\0';
flag = 0;
}
}
words = start;
while (*start)
{
if (('A' <= *start) && (*start <= 'Z'))
{
*words = *start;
words++;
}
else if (('a' <= *start) && (*start <= 'z'))
{
*words = *start - 32;
words++;
}
start++;
}
*words = '\0';
cout << "输入成功 !" << endl;
}

void InputObwords(char *words)  //输入密文
{
assert(words);
cout << "请输入密文:";
char *start = words;
int flag = 1;
getchar();
while (flag)
{
*words = getchar();
words++;
if (*(words - 1) == '\n')
{
*words = '\0';
flag = 0;
}
}
words = start;
while (*start)
{
if (('A' <= *start) && (*start <= 'Z'))
{
*words = *start;
words++;
}
else if (('a' <= *start) && (*start <= 'z'))
{
*words = *start - 32;
words++;
}
start++;
}
*words = '\0';
cout << "输入成功 !" << endl;
}
输入密钥与输出密钥函数:
void InputKeys(float keys[ROW][COV])  //输入密钥
{
cout << "请输入密钥:" << endl;
for (size_t i = 0; i < ROW; i++)
{
cout << "请输入第" << i << "行密钥("<<ROW<<"个数):";
for (size_t j = 0; j < COV; j++)
{
cin >> keys[i][j];
}
}
cout << "输入成功 !" << endl;
}

void PopKeys(float keys[ROW][COV])  //输出密钥
{
cout << "密钥为:" << endl;
for (size_t i = 0; i < ROW; i++)
{
for (size_t j = 0; j < COV; j++)
{
cout << keys[i][j] << "   ";
}
cout << endl;
}
}
加密函数:
void Encryption(float keys[ROW][COV], char *words, char *crypto)  //加密函数
{
assert(words);
int len = strlen(words);
char *start = words;

while (len > 0)
{
int matrix[ROW] = { 0 };
for (int i = 0; i < ROW; i++)
{
if (*start)
matrix[i] = *start - 'A';
else
matrix[i] = 0;
start++;
}
len -= ROW;
int cry[ROW] = { 0 };
for (int i = 0; i < ROW; i++)
{
int temp = 0;
for (int j = 0; j < COV; j++)
{
temp = matrix[j] * keys[j][i] + temp;
}
cry[i] = temp % 26;
*crypto = 'A' + cry[i];  //计算密文
crypto++;
}
}
}
解密函数,以及求逆矩阵函数:
void Decode(float obkeys[ROW][COV], char *words, char *crypto)//解密函数
{
assert(crypto);
int len = strlen(crypto);
char *start = crypto;
while (len > 0)
{
int matrix[ROW] = { 0 };
for (int i = 0; i < ROW; i++)
{
if (*start)
matrix[i] = *start - 'A';
else
matrix[i] = 0;
start++;
}
len -= ROW;
int cry[ROW] = { 0 };
for (int i = 0; i < ROW; i++)
{
int temp = 0;
for (int j = 0; j < COV; j++)
{
temp = matrix[j] * obkeys[j][i] + temp;
}
cry[i] = temp % 26;
*words = 'A' + cry[i];  //计算明文
words++;
}
}
}

void 	ObMatrix( float a[ROW][COV], float b[ROW][COV], int n)  //求逆矩阵函数
{
int i, j;         //定义矩阵的行列式
if (Gauss(a, b, n))
{
cout << "该方阵的逆矩阵为: \n";
for (i = 0; i < n; i++)
{
cout << setw(4);
for (j = 0; j < n; j++)
{
int temp =b[i][j]/ 1;
float num = b[i][j] - temp;
if (fabs(num) < 0.50)
b[i][j] = (int)temp;
else
b[i][j] = temp + (int)(num * 2);
cout << b[i][j] << setw(10);
}
cout << endl;
}
}
cout << "逆矩阵(mod26):" << endl;
for (int i = 0; i < ROW; i++)
{
cout << setw(4);
for (int j = 0; j < COV; j++)
{
if (b[i][j] >= 0)
{
b[i][j] = (int)b[i][j] % 26;
}
else
{
b[i][j] = 26 + (int)b[i][j] % 26;
}
cout << b[i][j] << setw(6);
}
cout << endl;
}

}

bool Gauss(float A[ROW][COV], float B[ROW][COV], int n) //高斯消去法
{
int i, j, k;
float max, temp;
float t[ROW][COV];                //临时矩阵
//将A矩阵存放在临时矩阵t

中
for (i = 0; i < n; i++)
{
for (j = 0; j < n; j++)
{
t[i][j] = A[i][j];
}
}
//初始化B矩阵为单位阵
for (i = 0; i < n; i++)
{
for (j = 0; j < n; j++)
{
B[i][j] = (i == j) ? (int)1 : 0;
}
}
for (i = 0; i < n; i++)
{
//寻找主元
max = t[i][i];
k = i;
for (j = i + 1; j < n; j++)
{
if (fabs(t[j][i]) > fabs(max))
{
max = t[j][i];
k = j;
}
}
//如果主元所在行不是第i行,进行行交换
if (k != i)
{
for (j = 0; j < n; j++)
{
temp = t[i][j];
t[i][j] = t[k][j];
t[k][j] = temp;
//B伴随交换
temp = B[i][j];
B[i][j] = B[k][j];
B[k][j] = temp;
}
}
//判断主元是否为0, 若是, 则矩阵A不是满秩矩阵,不存在逆矩阵
if (t[i][i] == 0)
{
cout << "There is no inverse matrix!";
return false;
}
//消去A的第i列除去i行以外的各行元素
temp = t[i][i];
for (j = 0; j < n; j++)
{
t[i][j] = t[i][j] / temp;        //主对角线上的元素变为1
B[i][j] = B[i][j] / temp;        //伴随计算
}
for (j = 0; j < n; j++)        //第0行->第n行
{
if (j != i)                //不是第i行
{
temp = t[j][i];
for (k = 0; k < n; k++)        //第j行元素 - i行元素*j列i行元素
{
t[j][k] = t[j][k] - t[i][k] * temp;
B[j][k] = B[j][k] - B[i][k] * temp;
}
}
}
}
return true;
}
程序运行结果:


选择:1

选择:y


选择:n


选择 2.明文加密:

选择 3.密文解密:

选择 4.退出:


限于目前的水平,仅写出上述函数,欢迎大家阅读,加密解密函数,我只写了hill密码的一部分,上述代码还有很多需要完善的地方,欢迎各位大神指点,恳请各位批评指正,谢谢

本文出自 “分享中进步” 博客,请务必保留此出处http://xmwen1.blog.51cto.com/10730069/1751672
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: