-2

我正在尝试用 C 编写一个简单的加密程序。我的目标是将“jim”(可以是任何单词)翻译为 10、9、13。然后加上 1,2,3(因为单词有 3 个字母)并得到 11,12,16 然后再翻译成文本,然后在屏幕上写 klp。我有 2 个算法,一个用于加密,第二个用于返回第一个。它是工作,但是对于这些话,当我启动第一个算法然后第二个算法(逆)时,出现问题“编译器可执行通过”。“compileo vou euecrqable throrgh”我检查了几次算法,我不知道我忽略了哪里

PS:如果你想运行名为 unencrypted.txt 的 make txt 文件并在其中写入单词,那么第一个算法然后是第二个

int main() {
    int g,z,o,c,l,i,j,k,*D;

    char alfabe[26]={'a','b','c','d','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','t','u','v','w','x','y','z'};
    FILE *fp1;
    FILE *mat;
    char word[20];

    fp1 = fopen("unencrypted.txt","r+");
    do {
        g = fscanf(fp1,"%s",word);
        if (g != EOF) {
            mat=fopen("encrypted.txt","a+");
            c=strlen(word);
            D=(int*)calloc(c,sizeof(int));
            for(i=0;i<c;i++) {
                for(j=0;j<26;j++) {
                    if(word[i]==alfabe[j]) {
                         D[i]=(((j+1)+(i+1))%26);
                         break;
                    }
                }
            }
        }

        for(z=0;z<c;z++){
              o=D[z];
              word[z]=alfabe[o-1];
        }

        for(k=0;k<c;k++) {
           fprintf(mat,"%c",word[k]);
        }
        fprintf(mat," ");
        fclose(mat);

    } while (g != EOF);
    fclose(fp1);
}

int main() {
    int g,z,o,c,l,i,j,k,*D;

    char alfabe[26]={'a','b','c','d','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','t','u','v','w','x','y','z'};
    FILE *fp1;
    FILE *mat;
    char word[20];

    fp1 = fopen("encrypted.txt","a+");
    do {
        g = fscanf(fp1,"%s",word);
        if (g != EOF) {
            mat=fopen("unencrypted1.txt","a+") ;
            c=strlen(word);
            D=(int*)calloc(c,sizeof(int));
            for(i=0;i<c;i++) {
                for(j=0;j<26;j++) {
                    if(word[i]==alfabe[j]) {
                        if(0>((j+1)-(i+1))){
                            D[i]=((j+1)-(i+1)+26);
                        } else D[i]=((j+1)-(i+1));
                        break;
                    }
                }
            }
        }

        for(z=0;z<c;z++){
            o=D[z];
            word[z]=alfabe[o-1];
        }

        for(k=0;k<c;k++) {
           fprintf(mat,"%c",word[k]);
        }
        fprintf(mat," ");
        fclose(mat);

    } while (g != EOF);
    fclose(fp1);
}
4

1 回答 1

0

首先,在您的加密例程中,您可以有未定义的行为:

word[z]=alfabe[o-1];

访问不应该 if 的内存,o == 0如果单词中字符的基于 1 的索引和字母表中基于 1 的索引的总和是 26 的倍数,则会发生这种情况。

为避免这种情况,您可以设置word[z] = alfabe[(o+25)%26],或者更简单地,+1在前一个循环中省略一个 s 并使用alfabe[o].

的访问alfabe[-1],如果发生,是一种未定义的行为,不太可能导致分段错误或类似的剧烈事情,它可能只会导致在加密字中写入意外字节。

在解密中,你有类似的问题,

if(0>((j+1)-(i+1))){
    D[i]=((j+1)-(i+1)+26);
} else D[i]=((j+1)-(i+1));

如果单词足够长,甚至j - i + 26可能是否定的(也就是说,对于自然英语文本来说极不可能 - 在许多其他语言中也是如此)。

再一次,它可能会发生D[i]设置为 0,在这种情况下

for(z=0;z<c;z++){
    o=D[z];
    word[z]=alfabe[o-1];
}

将再次访问alfabe[-1]并导致未定义的行为。

另请注意,如果您的输入包含除小写字母和空格(被 丢弃fscanf)之外的任何内容,则数组的相应条目D包含该内存位置发生的任何字节,再次导致未定义的行为。

如果您的输入仅包含小写字母(和空格),您可以通过设置在字母表中找到索引比使用循环更容易

j = word[i] - 'a';

为避免这些alfabe[-1]问题,我建议使用基于 0 的索引。

于 2012-05-09T22:01:01.307 回答