閱讀769 返回首頁    go 阿裏雲 go 技術社區[雲棲]


2011藍橋杯【初賽試題】程序設計題三

一種Playfair密碼變種加密方法如下:首先選擇一個密鑰單詞(稱為pair)(字母不重複,且都為小寫字母),然後與字母表中其他字母一起填入至一個5x5的方陣中,填入方法如下:
1.首先按行填入密鑰串。
2.緊接其後,按字母序按行填入不在密鑰串中的字母。
3.由於方陣中隻有25個位置,最後剩下的那個字母則不需變換。
如果密鑰為youandme,則該方陣如下:
y o u a n
d m e b c
f g h i j
k l p q r
s t v w x
在加密一對字母時,如am,在方陣中找到以這兩個字母為頂點的矩形(紅色字體):
y o u a n
d m e b c
f g h i j
k l p q r
s t v w x
這對字母的加密字母為該矩形的另一對頂點,如本例中為ob。
請設計程序,使用上述方法對輸入串進行加密,並輸出加密後的串。
另外有如下規定:
1、一對一對取字母,如果最後隻剩下一個字母,則不變換,直接放入加密串中;
2、如果一對字母中的兩個字母相同,則不變換,直接放入加密串中;
3、如果一對字母中有一個字母不在正方形中,則不變換,直接放入加密串中;
4、如果字母對出現在方陣中的同一行或同一列,如df或hi,則隻需簡單對調這兩個字母,即變換為fd或ih;
5、如果在正方形中能夠找到以字母對為頂點的矩形,假如字母對為am,則該矩形的另一對頂點字母中,與a同行的字母應在前麵,在上例中應是ob;同樣若待變換的字母對為ta,則變換後的字母對應為wo;
6、本程序中輸入串均為小寫字母,並不含標點、空格或其它字符。
解密方法與加密相同,即對加密後的字符串再加密,將得到原始串。
要求輸入形式如下:
從控製台輸入兩行字符串,第一行為密鑰單詞(長度小於等於25),第二行為待加密字符串(長度小於等於50),兩行字符串末尾都有一個回車換行符,並且兩行字符串均為小寫字母,不含其它字符。
在標準輸出上輸出加密後的字符串。
例如,若輸入:
youandme
welcometohangzhou
則表示輸入的密鑰單詞為youandme,形成的正方形如上所示;待加密字符串為welcometohangzhou。在正方形中可以找到以第一對字母we為頂點的矩形,對應另一對頂點字母為vb,因此加密後為vb,同理可找到與字母對lc,et,oh,ho對應的頂點字母對。而字母對om位於上述正方形中的同一列,所以直接以顛倒這兩個字母來加密,即為mo,字母對an同理。字母對gz中的z不在上述正方形中,因此原樣放到加密串中。最後剩一個字母u也原樣輸出。
因此輸出的結果為:
Vbrmmomvugnagzguu

 

 

簡單的二維字符串處理問題,不過有點繁瑣

 ps:寫了好久,惡心啊.....

#include<stdio.h>
#include<string.h>
char a[2000],b[2000],c[2000];
char str[25][25];
char english[27]="abcdefghijklmnopqrstuvwxyz";
int flag[27];
int main()
{
    int i,j,n,m,k,p,flagx,flagy;
    gets(a);//輸入密鑰單詞 
    gets(b);//輸入待加密字符串 
    n=strlen(a);//分別求長度 
    m=strlen(b);
    memset(flag,0,sizeof(flag));
    k=0;
    for(i=0;i<5;i++)
    for(j=0;j<5;j++)
    {
      str[i][j]=a[k++];
      if(k-1==n)
      {
         flagx=i;//記錄密鑰單詞填充到了5*5的矩陣的哪一行哪一列為止 
         flagy=j;
         break;
      }
    }
    for(i=0;i<26;i++)//尋找不存在在密鑰單詞中的字母(標記法) 
    {
      for(j=0;j<k;j++)
      if(a[j]==english[i])
      flag[i]=1;
    }
    p=0;
    for(i=flagx;i<5;i++)//在剩下的矩陣空間中填充 存在在密鑰單詞中的字母
    {
       if(i==flagx)
       j=flagy;
       else
       j=0;
       for(;j<5;j++)
       {
          while(flag[p]==1)
          {p++;}
          str[i][j]=english[p++];
       }
    }
    /*for(i=0;i<5;i++)//測試 
    {
      for(j=0;j<5;j++)
      {
         printf("%c ",str[i][j]);                   
      }
      puts("");
    }*/
    for(i=0;i<m;i++)//看看哪一個單詞不在5*5的矩陣中(下次遇到直接輸出) 
    { 
       k=0;
       for(j=0;j<5;j++)
       for(p=0;p<5;p++)
       {
          if(b[i]==str[j][p])
          k=1;
       }
       if(k==0)
       {k=i;break;}
    }
    int x1,y1,x2,y2,v=0,flag1=0,flag2=0;
    for(i=0;i<m;i++)//開始對待加密字符串進行加密 
    {
       if(i==m-1)//如果是最後一個字母,這直接輸出並結束循環 
       {
         c[v++]=b[m-1];
         break;
       }
       if(b[i]==b[i+1])//如果字符對相等,直接輸出 
       {
         c[v++]=b[i];
         c[v++]=b[i+1];
         i++;
         continue; 
       }
       for(j=0;j<5;j++)//尋找在矩陣中兩個字母組成的矩形的對角坐標 
       for(p=0;p<5;p++)
       {
          if(str[j][p]==b[i])
          {x1=j;y1=p;flag1=1;}
          if(str[j][p]==b[i+1])
          {x2=j;y2=p;flag2=1;}
       }
       if(k==i||k==i+1)//字符對其中有一個是上一次查出來的不存在在 5*5的矩陣中的,直接輸出 
       {
          c[v++]=b[i];
          c[v++]=b[i+1];
          i++;
          continue;
       }
       if(x1==x2||y1==y2)//同行同列直接輸出 
       {
          c[v++]=b[i+1];
          c[v++]=b[i]; 
          i++;
          continue;              
       }
       //剩下的就是能組成矩形的了,別忘了先輸出第一個字符那一行的字母 
       c[v++]=str[x1][y2];
       c[v++]=str[x2][y1];
       i++;    
    }
    printf("%s\n",c);//輸出答案 
    return 0;
} 


最後更新:2017-04-03 12:55:32

  上一篇:go C# 打開指定路徑下文件
  下一篇:go WIKIOI-1341 與3和5無關的數