08:Vigenère密码
总时间限制:
1000ms
内存限制:
65536kB
描述
16世纪法国外交家Blaise de Vigenère设计了一种多表密码加密算法——Vigenère密码。Vigenère密码的加密解密算法简单易用,且破译难度比较高,曾在美国南北战争中为南军所广泛使用。
在密码学中,我们称需要加密的信息为明文,用M表示;称加密后的信息为密文,用C表示;而密钥是一种参数,是将明文转换为密文或将密文转换为明文的算法中输入的数据,记为k。 在Vigenère密码中,密钥k是一个字母串,k=k1k2…kn。当明文M=m1m2…mn时,得到的密文C=c1c2…cn,其中ci=mi®ki,运算®的规则如下表所示:
![](https://i-blog.csdnimg.cn/blog_migrate/35132061a721993c7551304424d9c7ce.png)
Vigenère加密在操作时需要注意:
1. ®运算忽略参与运算的字母的大小写,并保持字母在明文M中的大小写形式;
2. 当明文M的长度大于密钥k的长度时,将密钥k重复使用。
例如,明文M=Helloworld,密钥k=abc时,密文C=Hfnlpyosnd。
明文 | H | e | l | l | o | w | o | r | l | d |
密钥 | a | b | c | a | b | c | a | b | c | a |
密文 | H | f | n | l | p | y | o | s | n | d |
输入
输入共2行。
第一行为一个字符串,表示密钥k,长度不超过100,其中仅包含大小写字母。第二行为一个字符串,表示经加密后的密文,长度不超过1000,其中仅包含大小写字母。
对于100%的数据,输入的密钥的长度不超过100,输入的密文的长度不超过1000,且都仅包含英文字母。
输出
输出共1行,一个字符串,表示输入密钥和密文所对应的明文。
样例输入
CompleteVictory Yvqgpxaimmklongnzfwpvxmniytm
样例输出
Wherethereisawillthereisaway
来源
NOIP2012复赛 提高组 第一题
- 查看
- 提交
- 统计
- 提问
-
#include<bits/stdc++.h> using namespace std; string k,c; int lk,lc,i,j; int main() { cin>>k; cin>>c; lk=k.length(); lc=c.length(); i=0;j=0; while (i!=lc) { int bz=0,pd,u=0; char sc; if (c[i]>='a'&&c[i]<='z') { if (k[j]>='a'&&k[j]<='z') bz=(int)k[j]-'a'; else if (k[j]>='A'&&k[j]<='Z') bz=(int)k[j]-'A'; pd=c[i]-'a'; if (pd>=bz) sc=(char)c[i]-bz,u=1; else bz-=(int)c[i]-'a'+1; } else { if (k[j]>='a'&&k[j]<='z') bz=(int)k[j]-'a'; else if (k[j]>='A'&&k[j]<='Z') bz=(int)k[j]-'A'; pd=c[i]-'A'; if (pd>=bz) sc=(char)c[i]-bz,u=1; else bz-=(int)c[i]-'A'+1; } if (u==1) cout<<sc; else { if (c[i]>='a'&&c[i]<='z') sc=(char)'z'-bz; else sc=(char)'Z'-bz; cout<<sc; } i++; j++; if (j==lk) j=0; } return 0; }