RSA算法C语言实现

一、源文件三个rsa.h , rsa.c, main.c

//rsa.h

#include <stdio.h>

#define MAX_NUM 63001

#define MAX_PRIME 251

//! 返回代码

#define OK 100

#define ERROR_NOEACHPRIME 101

#define ERROR_NOPUBLICKEY 102

#define ERROR_GENERROR 103

unsigned int MakePrivatedKeyd( unsigned int uiP, unsigned int uiQ );

unsigned int GetPrivateKeyd( unsigned int iWhich );

unsigned int MakePairkey( unsigned int uiP, unsigned int uiQ, unsigned int uiD );

unsigned int GetPairKey( unsigned int &d, unsigned int &e );

void rsa_encrypt( int n, int e, char *mw, int iLength, int *&cw );

void rsa_decrypt( int n, int d, int *&cw, int cLength, char *mw );

void outputkey();

//rsa.c

#include "rsa.h"

//! 保存私钥d集合

struct pKeyset

{

unsigned int set[ MAX_NUM ];

unsigned int size;

}pset;

//! 保存公、私钥对

struct pPairkey

{

unsigned int d;

unsigned int e;

unsigned int n;

}pairkey;

// 名称:isPrime

// 功能:判断两个数是否互质

//参数:m: 数a; n: 数b

// 返回:m、n互质返回true; 否则返回false

bool isPrime( unsigned int m, unsigned int n )

{

unsigned int i=0;

bool Flag = true;

if( m<2 || n<2 )

return false;

unsigned int tem = ( m > n ) ? n : m;

for( i=2; i<=tem && Flag; i++ )

{

bool mFlag = true;

bool nFlag = true;

if( m % i == 0 )

mFlag = false;

if( n % i == 0 )

nFlag = false;

if( !mFlag && !nFlag )

Flag = false;

}

if( Flag )

return true;

else

return false;

}

// 名称:MakePrivatedKeyd

// 功能:由素数Q、Q生成私钥d

//参数:uiP: 素数P; uiQ: 素数Q

// 返回:私钥d

unsigned int MakePrivatedKeyd( unsigned int uiP, unsigned int uiQ )

{

unsigned int i=0;

//! 得到所有与z互质的数( 私钥d的集合 )

unsigned int z = ( uiP -1 ) * ( uiQ -1 );

pset.size = 0;

for( i=0; i<z; i++ )

{

if( isPrime( i, z ) )

{

pset.set[ pset.size++ ] = i;

}

}

return pset.size;

}

// 名称:MakePairKey

// 功能:生成RSA公、私钥对

//参数:uiP: 素数P; uiQ: 素数Q; uiD: 私钥d

// 返回:错误代码

unsigned int MakePairkey( unsigned int uiP, unsigned int uiQ, unsigned int uiD )

{

bool bFlag = true;

unsigned int i = 0, e;

unsigned int z = ( uiP-1 ) * ( uiQ-1 );

unsigned int d = pset.set[uiD];

//d=uiD;

if( !isPrime( z, d ) )

return ERROR_NOEACHPRIME;

for( i=2; i<z; i++ )

{

if( (i*d)%z == 1 )

{

e = i;

bFlag = false;

}

}

if( bFlag )

return ERROR_NOPUBLICKEY;

if( (d*e)%z != 1 )

ERROR_GENERROR;

pairkey.d = d;

pairkey.e = e;

pairkey.n = uiP * uiQ;

return OK;

}

// 名称:GetPairKey

// 功能:对外提供接口,获得公、私钥对

//参数:uiP: 素数P; uiQ: 素数Q; uiD: 私钥d

// 返回:

unsigned int GetPairKey( unsigned int &d, unsigned int &e )

{

d = pairkey.d;

e = pairkey.e;

return pairkey.n;

}

// 名称:GetPrivateKeyd

// 功能:对外提供接口,由用户选择ID得以私钥d

//参数:iWhich: 用户选择私钥d的ID

// 返回:私钥d值

unsigned int GetPrivateKeyd( unsigned int iWhich )

{

if( pset.size >= iWhich )

return pset.set[ iWhich ];

else

return 0;

}

// 名称:rsa_encrypt

// 功能:RSA加密运算

//参数:n: 公钥n; e: 公钥e; mw: 加密明文; iLength: 明文长度; cw: 密文输出

// 返回:无

void rsa_encrypt( int n, int e, char *mw, int mLength, int *&cw )

{

int i=0, j=0;

__int64 temInt = 0;

for( i=0; i<mLength; i++ )

{

temInt = mw[i];

if( e!=0 )

{

for( j=1; j<e; j++ )

{

temInt = ( temInt * mw[i] ) % n;

}

}

else

{

temInt = 1;

}

cw[i] = (int)temInt;

}

}

// 名称:rsa_decrypt

// 功能:RSA解密运算

//参数:n: 私钥n; d: 私钥d; cw: 密文; cLength: 密文长度; mw: 明文输出

// 返回:无

void rsa_decrypt( int n, int d, int *&cw, int cLength, char *mw )

{

int i=0, j=-1;

__int64 temInt = 0;

for( i=0; i<cLength/4; ++i )

{

mw[i] = 0;

temInt = cw[i];

if( d != 0 )

{

for( j=1; j<d; j++ )

{

temInt = (__int64)( temInt * cw[i] ) % n;

}

}

else

{

temInt = 1;

}

mw[i] = (char)temInt;

}

}

void outputkey()

{

printf("PublicKey(e,n): (%d,%d)\n",pairkey.e,pairkey.n);

printf("PrivateKey(d,n): (%d,%d)\n",pairkey.d,pairkey.n);

}

//main.c

// 工程:RSA

// 功能:RSA加、解密文件

//作者:jlcss|ExpNIS

#include <stdio.h>

#include <afxwin.h>

#include <math.h>

#include "rsa.h"

#define DECRYPT_FILE "RSA加密密文.txt"

#define ENCRYPT_FILE "RSA解密明文.txt"

//! 约束文件最大2M

#define MAX_FILE 1024*1024*2

// 名称:usage

// 功能:帮助信息

//参数:应用程序名称

// 返回:提示信息

void Usage( const char *appname )

{

printf( "\n\tusage:rsa -k 素数P 素数Q\n" );

printf( "\tusage: rsa -e 明文文件 公钥e 公钥n\n" );

printf( "\tusage: rsa -d 密文文件 私钥d 私钥n\n" );

}

// 名称:IsNumber

// 功能:判断数字字符数组

//参数:strNumber:字符数组

// 返回:数字字组数组返回true,否则返回false;

bool IsNumber( const char *strNumber )

{

unsigned int i;

if( !strNumber )

return false;

for ( i = 0 ; i < strlen(strNumber) ; i++ )

{

if ( strNumber[i] < '0' || strNumber[i] > '9' )

return false;

}

return true;

}

// 名称:IsPrimeNumber

// 功能:判断素数

//参数:num: 输入整数

// 返回:素数返回true,否则返回false;

bool IsPrimeNumber( unsigned int num )

{

unsigned int i;

if( num <= 1 )

return false;

unsigned int sqr = (unsigned int)sqrt((double)num);

for( i = 2; i <= sqr; i++ )

{

if( num % i == 0 )

return false;

}

return true;

}

// 名称:FileIn

// 功能:读取磁盘文件到内存

//参数:strFile:文件名称;inBuff:指向文件内容缓冲区

// 返回:实际读取内容大小(字节)

int FileIn( const char *strFile, unsigned char *&inBuff )

{

int iFileLen=0, iBuffLen=0;

//! 打开密文文件

CFile file( strFile, CFile::modeRead );

iFileLen = ( int )file.GetLength();

if( iFileLen>MAX_FILE )

{

printf( "文件长度不能大于 %dM,!\n", MAX_FILE/(1024*1024) );

goto out;

}

iBuffLen = iFileLen;

inBuff = new unsigned char[iBuffLen];

if( !inBuff )

goto out;

ZeroMemory( inBuff, iBuffLen );

file.Read( inBuff, iFileLen );

file.Close();

out:

return iBuffLen;

}

// 名称:FileOut

// 功能:加/解密结果输出到当前目录磁盘文件中

//参数:strOut指向输出字符缓冲区,输出大小len,strFile为输出文件

// 返回:无

void FileOut( const void *strOut, int len, const char *strFile )

{

//! 输出到文件

CFile outfile( strFile , CFile::modeCreate | CFile::modeWrite );

outfile.Write( strOut , len );

outfile.Close();

}

// 名称:CheckParse

// 功能:校验应用程序入口参数

//参数:argc等于main主函数argc参数,argv指向main主函数argv参数

// 返回:若参数合法返回true,否则返回false

//备注:简单的入口参数校验

bool CheckParse( int argc, char** argv )

{

bool bRes = false;

if( argc != 4 && argc != 5 )

goto out;

if( argc == 4 && argv[1][1] == 'k' )

{

//! 生成公、私钥对

if( !IsNumber( argv[2] ) ||

!IsNumber( argv[3] ) ||

atoi( argv[2] ) > MAX_PRIME ||

atoi( argv[3] ) > MAX_PRIME )

goto out;

}

else if( (argc == 5) && (argv[1][1] == 'e' || argv[1][1] == 'd') )

{

//! 加密、解密操作

if( !IsNumber( argv[3] ) ||

!IsNumber( argv[4] ) ||

atoi( argv[3] ) > MAX_NUM ||

atoi( argv[4] ) > MAX_NUM )

goto out;

}

else

Usage(*argv);

bRes = true;

out:

return bRes;

}

// 名称:kOption1

// 功能:程序k选项操作:由素数P、Q生成私钥d集合

//参数:uiP: 程序入口参数P; uiQ: 程序入口参数Q

// 返回:执行正确返回生成私钥数目,否则返回0

unsigned int kOption1( unsigned int uiP, unsigned int uiQ )

{

unsigned int uiRes = 0;

if( !IsPrimeNumber( uiP ) )

{

printf( "P输入错误,P必须为(0, %d]素数", MAX_PRIME );

return uiRes;

}

if( !IsPrimeNumber( uiQ ) )

{

printf( "Q输入错误,Q必须为(0, %d]素数", MAX_PRIME );

return uiRes;

}

if( uiP == uiQ )

{

printf( "素数P与素数Q相同,很容易根据公钥n开平方得出素数P和Q,这种加密不安全,请更换素数!\n" );

return uiRes;

}

printf( "正在生成私钥d集合......\n" );

uiRes = MakePrivatedKeyd( uiP, uiQ );

return uiRes;

}

//! 程序主函数

int main( int argc, char **argv )

{

unsigned int p , q , d , n , e;//two prime p & q, public key(n, e) , private key(n , d)

CheckParse(argc,argv );

d=4828; //uid

if(argc == 4)

{

p = atoi( argv[2] );

q = atoi( argv[3] );

MakePrivatedKeyd(p, q);

MakePairkey(p, q, d );

outputkey();

}

else if(argc == 5)

{

char FileName[20];

strcpy(FileName, argv[2]);

int len;

if(argv[1][1] == 'e' )

{

unsigned char *inBuffer=(unsigned char *)malloc(MAX_FILE); //输入缓冲区

int *cw=(int *)malloc(MAX_FILE);

len = FileIn(FileName , inBuffer);

e = atoi(argv[3]);

n = atoi(argv[4]);

rsa_encrypt( n, e, (char *)inBuffer, len, cw );

FileOut( cw, 4*len, DECRYPT_FILE );

}

else if(argv[1][1] == 'd')

{

char *Buffer=(char *)malloc(MAX_FILE); //输入缓冲区

int *cw=(int *)malloc(MAX_FILE);

len = FileIn(FileName, (unsigned char *&)cw);

d = atoi(argv[3]);

n = atoi(argv[4]);

rsa_decrypt( n, d, cw, len, Buffer );

FileOut( Buffer, len/4, ENCRYPT_FILE );

}

}

return 0;

}

二、运行方法及结果


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容