본문 바로가기
프로그램/C언어 1000제

C언어 204제] 암호 깨기 II(Crypt Kicker II)

by 건티 2024. 12. 26.
728x90

출처 : 반크_백제역사 유적지구 및 이스탐블역사 유적지구

 

문제]

텍스트를 암호화하는 방법 중에 보안상 취약하긴 하지만 흔하게 쓰이는 방법으로 알파벳 글자를 다른 글자로 바꾸는 방법이 있다. 즉 알파벳의 각 글자가 다른 글자로 치환된다. 암호화된 것을 다시 원래대로 되돌릴 수 있으려면 두 개의 서로 다른 글자가 같은 글자로 치환되지 않아야 한다.
알려진 평문 공력법(known plain text attack)이라는 강력한 암호 분석 방법이 있다. 알려진 평문 공격법은 상대방이 암호화했다는 것을 알고 있는 구문이나 문장을 바탕으로 암호화된 텍스트를 관찰해서 인코딩 방법을 유추하는 방법이다.
여러 줄의 텍스트가 주어졌을 때 같은 케이스에서는 모든 줄에서 같은 치환 방법을 사용한다고 가정하고 그 중 한 줄에 the quick brown fox jumps over the lazy dog라는 평문을 암호화한 것이라는 점을 이용해서 암호화된 텍스트를 복호화하라.

입력
첫번째 줄에는 양의 정수 하나만 들어있는데, 이 정수는 테스트 케이스의 개수를 나타낸다. 그 다음 줄은 빈 줄이다. 서로 다른 테스트 케이스는 빈 줄로 구분된다.
각 케이스는 열 줄로 구성되는데, 앞에서 설명한 방법에 따라 암호화도니 텍스트다. 암호화된 내용은 소문자와 스페이스만으로 구성되며 길이는 최대 80문자로 제한된다. 입력되는 텍스트는 100줄로 제한된다.

출력
각 테스트 케이스에 대해 각 줄을 복호화하여 표준 출력으로 출력한다. 가능한 암호화 방법이 두 개 이상있으면 그 중 아무 결과나 출력해도 된다. 복호화할 수 없는 경우에는 다음과 같은 결과를 출력한다.
  No solution.
서로 다른 케이스 사이에는 빈 줄을 한 개 집어넣는다.

입력 예
1

vtz ud xnm xugm itr pyy jttk gmv xt otgm xt xnm puk ti xnm fprxq
xnm ceuob lrtzv ita hegfd tsmr xnm ypwq ktj
frtjrpgguvj otvxmdxd prm iev prmvx xnmq

출력 예
now is the time for all good men to come to the aid of the party
the quick brown fox jumps over the lazy dog
programming contests are fun arent they



출처]
Programming Challenges 알고리즘 트레이닝 북(한빛미디어) : 문제 20 암호 깨기 II(Crypt Kicker II) p100

 

 

참고풀이]

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

int main()
{
   static char Org[]="the quick brown fox jumps over the lazy dog";
   static char Line[100][100];
   static char map[256], inv[256];
   int cases, lines;
   int i, j, possible;
   int chk;
   char ch;

   scanf("%d",&cases);
   gets(Line[0]);
   gets(Line[0]);

   while(cases-- > 0)
   {
      chk=1;
      lines=0;
      while(gets(Line[lines]) && Line[lines][0])
         lines++;
      possible=0;
      for(i=0; i<lines; i++)
      {
         if(strlen(Org) != strlen(Line[i]))
            continue;
         for(ch='a';ch<='z';ch++)
         {
            map[ch]='\0';
            inv[ch]='\0';
         }
         map[' ']=' ';
         for(j=0;Line[i][j];j++)
            if(map[Line[i][j]] && map[Line[i][j]] != Org[j] || 
               inv[Org[j]] && inv[Org[j]] != Line[i][j])
            { 
               chk=0; 
               break; 
            }
            else
            {
               map[Line[i][j]] = Org[j];
               inv[Org[j]] = Line[i][j];
            }
         if(chk)
         {
            possible=1;
            break;
         }
      }

      if(possible)
      {
         for(i=0; i<lines;i++)
         {
            for(j=0;Line[i][j];j++)
               putchar(map[Line[i][j]]);
            putchar('\n');
         }
      }
      else
         printf("No solution.\n");
      if(cases>0) putchar('\n');
   }
   return 0;
}

 

참고풀이 결과]

 

 

 

 

 

대한민국의 아름다운 영토, 독도

 

반응형

댓글