CS50 Vigenere - 奇怪的图案

Posted

技术标签:

【中文标题】CS50 Vigenere - 奇怪的图案【英文标题】:CS50 Vigenere - Strange pattern 【发布时间】:2015-11-01 06:48:48 【问题描述】:

我在使用 Vigenere 时遇到了一点问题,需要一些帮助。

/*
This program is a Vigenere Cipher.
I am Daniel of Asguard.
*/

#include <cs50.h>
#include <ctype.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main(int argc, string argv[])

string cipher; //this is a placeholder for the ciphered message.
char * key = argv[1];
int i = 0;


if (argc != 2) //this is meant to trigger if you don't enter the right call. So 

    printf("Please enter the cipher key when you call the program, such as './CaesarCipher 7'.\n"); //
    return 1;


if (!isalpha(key[i])) //this is meant to trigger if you don't enter the right call. So 

    printf("Please only enter a word, no numerical numbers please."); //
    return 1;


do

    //printf("Please enter the message you would like to have converted, please. \n");
    cipher =  GetString(); 

while (cipher == NULL);

for (int i = 0, k = 0, n = strlen(cipher); i < n; i++, k++) //this is so the code knows to change only the characters in the sting cipher.
    
        if (k >= strlen(key))
        
            k = 0;
        
            
            if (isupper(cipher[i]))
                
                    //cipher[i] = 'A' + (((cipher[i] - 'A') + (key[k]) - 'A')  % 26);
                    cipher[i] = ((key[k] - 65) + (cipher[i] - 65)) % 26;
                    printf("%s\n", cipher);
                
            else (islower(cipher[i]));
                
                    //cipher[i] = 'a' + (((cipher[i] - 'a') + (key[i]) - 'a') % 26);
                    cipher[i] = ((key[k] - 97) + (cipher[i] - 97)) % 26;
                    printf("%s\n", cipher);
                
        
    
printf("%s\n", cipher);
return 0;

当我这样做时,我的结果会得到奇怪的字符:⎽c▒⎺e┼├⎼▒┤└e⎼@☃de5▮:·/┬⎺⎼┐⎽⎻▒ce/⎻⎽e├2 $ └▒ ┐e ┴☃±e┼e⎼e 完成后我终端中的所有字母。

BaZ 的结果最终看起来像这样:

注意事项 注意事项 注意事项 注意事项 注意事项 注意事项 注意事项 注意事项 注意事项 注意事项 f 注意 f 注意 注意 特 特 特 ├e e

【问题讨论】:

【参考方案1】:

根据https://en.wikipedia.org/wiki/Vigen%C3%A8re_cipher 中的示例,下面的代码按预期工作

#include <ctype.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main (void)
 
    const char  *cipher = "ATTACKATDAWN";
    const char  *key = "LEMON";
    char *newkey;
    char *p, *q;
    int i =0;
    int col, row;
    if (strlen (key) < strlen (cipher))
    
printf ("key %s \n", key);
printf ("cipher \t%s \n", cipher);
newkey = malloc ( strlen (cipher) +1);
strcpy ( newkey, key);
p = (char *) (newkey + strlen (newkey)) ;
q = (char *) key; 
i =strlen (key);
while (i < strlen (cipher) )

        i++;
        if (*q == 0)
    q = (char *) key;
        *p = *q;
        p++;
        q++;

    *p = 0;
    printf ("newk \t%s \n", newkey);
    
    p = (char *) newkey;
    q= (char *) cipher;
    int a[1] ;
    a[1] =0;
    for (i =0  ; i < strlen(newkey); i++)
    
row = *p -65;
col = *q -65;
if (col+row > 26)
        a [0] = 65 + col+row -26;
else
        a [0] = 65 +col+row;
printf ("%s",(char *)  &a[0]);
p++;
q++;
    
    printf ("\n");
    free( newkey);
/*L X F O P V E F R N H R*/
return 0;

【讨论】:

【参考方案2】:

上线:

cipher[i] = ((key[k] - 65) + (cipher[i] - 65)) % 26;

您忘记在结果中加回 65(A 的 ASCII 码)。因此,cipher[i] 将是一个从 0 到 25 的字符,所有这些都是不可打印的控制代码(特别是 null / 0 字符被 C 字符串函数视为字符串结束标记)。

行:

cipher[i] = ((key[k] - 97) + (cipher[i] - 97)) % 26;

有同样的错误。

此外,在这两行中,您都假设密钥字符与您正在加密的消息字符具有相同的大小写。如果不是这种情况,您的加密结果将不正确。为了解决这个问题,我建议在主加密循环之前将整个密钥转换为所有大写(或小写)

您还可以更进一步,在循环之前将键转换为 0 到 25 之间的数字。但是,执行此操作后,您将无法在密钥上使用 strlen(),因为 strlen() 会查找字符串中第一个空 (= 0) 字符的位置。相反,您必须在转换密钥之前运行strlen(),并将其结果保存在变量中以备后用。实际上,在任何情况下这都是一个有用的优化(你也应该对消息进行优化)。

【讨论】:

以上是关于CS50 Vigenere - 奇怪的图案的主要内容,如果未能解决你的问题,请参考以下文章

CS50 Vigenere - 输出不正确

为啥 Vigenere 失败 (CS50x)?

cs50 的 Vigenere 密码程序错误

我的 CS50 Vigenere 密码程序有啥问题?

我的 CS50 Vigenere 代码有啥问题?

关于 CS50 pset2 vigenere