zzulioj 1215: 破解简单密码

问题

1215: 破解简单密码
时间限制: 1 Sec  内存限制: 128 MB
提交: 865  解决: 347

密码是我们生活中非常重要的东东,我们的
那么一点不能说的秘密就全靠它了。哇哈哈. 
接下来渊子要在密码之上再加一套密码,
虽然简单但也安全。 
假设老王原来一个BBS上的密码为zvbo941987,
为了方便记忆,他通过一种算法把这个密码变换成YUANzi1987,
这个密码是他的名字和出生年份,怎么忘都忘不了,
而且可以明目张胆地放在显眼的地方而不被别人知道真正的密码。 
他是这么变换的,
大家都知道手机上的字母: 
1--1, abc--2, def--3, ghi--4, jkl--5, mno--6, pqrs--7, tuv--8 wxyz--9, 0--0,
就这么简单,老王把密码中出现的小写字母都变成对应的数字,
数字和其他的符号都不做变换,声明:密码中没有空格,
而密码中出现的大写字母则边成小写之后往后移一位,
如:X,先边成小写,再往后移一位,不就是y了嘛,简单吧。
记住,z往后移是a哦。

输入
输入包括多个测试数据。输入是一个明文,
密码长度不超过100个字符,输入直到文件结尾。
输出
输出渊子真正的密文。

样例输入 Copy
YUANzi1987
样例输出 Copy
zvbo941987

分析

小白代码

// 出现的小写字母都变成对应的数字,
// 数字和其他的符号都不做变换,
// 密码中出现的大写字母则边成小写之后往后移一位,
// 如:X,先边成小写,再往后移一位,不就是y了
// 出现的小写字母都变成对应的数字,
// 数字和其他的符号都不做变换,
// 密码中出现的大写字母则边成小写之后往后移一位,
// 如:X,先边成小写,再往后移一位,不就是y了
#include <iostream>
#include <string>
#include <map>
using namespace std;

map<char, int> chMap { // 例如,'a' 对应 2, 通过 chMap['a'] 可以得到 2。
        {'a', 2}, {'b', 2}, {'c', 2},
        {'d', 3}, {'e', 3}, {'f', 3},
        {'g', 4}, {'h', 4}, {'i', 4},
        {'j', 5}, {'k', 5}, {'l', 5},
        {'m', 6}, {'n', 6}, {'o', 6},
        {'p', 7}, {'q', 7}, {'r', 7}, {'s', 7},
        {'t', 8}, {'u', 8}, {'v', 8},
        {'w', 9}, {'x', 9}, {'y', 9}, {'z', 9}
};

int main() {
    string str;
    string res;
    while (cin >> str) {
        res = ""; // 记得初始化,防止上次实例结果累加到这一次
        for (auto mem : str) {
    
            if (isupper(mem)) { // 大写字母的转换          
                res += mem - 'A' + 'a' + 1 > 'z' ? 'a' : mem - 'A' + 'a' + 1;
                // 通过三目运算操作判断 z+1 的特殊情况
            }
            else if (islower(mem)) { // 小写字母的转换
                res += to_string(chMap[mem]); // 从map获得对应字母的值
            }
            else res += mem;
        }
        cout << res << endl;
    }
    return 0;
}

大佬的码

分析

分大写字母、小写字母、其他、三种情况,
其中大写字母向后移一位要考虑z的特殊性,
因此采用“环”的思想,将a到z视作在025循环的26个字符,
第i个字符,后进一位后用(i+1%26表示,
在+97转化为ascll字符
/*
分大写字母、小写字母、其他、三种情况,
其中大写字母向后移一位要考虑z的特殊性,
因此采用“环”的思想,将a到z视作在0到25循环的26个字符,
第i个字符,后进一位后用(i+1)%26表示,
在+97转化为ascll字符
*/

#include<bits/stdc++.h>
using namespace std;

char n[1000];

int main()
{

    while(~scanf("%s",&n));//将一行连续字符存入n

	for (int i = 0; n[i] != '\0'; i++)
	{
		if (n[i] >= 'A' && n[i] <= 'Z')
		{
			n[i] = tolower(n[i]);						//大写字母变小写
			cout << (char)((n[i] - 96) % 26 + 97); //字母向后移一位
		}
		else if (n[i] >= 'a' && n[i] <= 'o')
		{
			cout << ((int)n[i] - 97) / 3 + 2; //在o之前都是每三个一组,可以合并化简代码
		}
		else if(n[i]>='p'&&n[i]<='s'){cout<<7;}//从p开始出现4个一组所以单独判断
		else if(n[i]>='t'&&n[i]<='v'){cout<<8;}
		else if(n[i]>='w'&&n[i]<='z'){cout<<9;}
		else{cout<<n[i];}
	}
}

小结

Only Hard Work!
Go get it!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值