我希望你是我独家记忆

一段永远封存的记忆,随风而去
posts - 263, comments - 31, trackbacks - 0, articles - 3
   :: 首页 :: 新随笔 ::  :: 聚合  :: 管理

PKU ——1506——(同ZJU2003)

Posted on 2008-08-19 19:49 Hero 阅读(211) 评论(0)  编辑 收藏 引用 所属分类: 代码如诗--ACM

 

  1 //1506 Accepted 572K 250MS C++ 3223B 
  2 
  3 //求出传递闭包后,a对应的密码的outdeg应该为(inn-1),b->(inn-2)
  4 //然后用pwd2word[]建立映射
  5 
  6 //以后注意特殊数据--比如要解密的单词中根本没有加密的字母
  7 
  8 #include <stdio.h>
  9 #include <stdlib.h>
 10 #include <string.h>
 11 
 12 const int size = 300 ;
 13 
 14 int edge[size][size] ;
 15 int indeg[size] ;
 16 int outdeg[size] ;
 17 
 18 int text2pwd[size] ;//文本到对应密码的转换
 19 int pwd2text[size] ;//密码到对应文本的转换
 20 
 21 char fword[300000] ; char sword[300000] ;
 22 int flen = 0 ; int slen = 0 ;
 23 
 24 int inn, inp, testnum ;
 25 
 26 bool can = true ;
 27 
 28 void input()
 29 {
 30     memset( edge, 0sizeof(edge) ) ;
 31 
 32     scanf( "%d %d"&inn, &inp ) ;
 33 
 34     scanf( "%s", fword ) ;
 35     forint i=2; i<=inp; i++ ) {
 36         scanf( "%s", sword ) ;
 37         flen = strlen( fword ) ; slen = strlen( sword ) ;
 38         int k = 0 ;    for( k=0; fword[k]==sword[k]&&k<slen&&k<flen; k++ ) ;//找到第一个不相同的字母
 39         if( k < flen && k < slen ) {
 40             edge[fword[k]-'a'+1][sword[k]-'a'+1= 1 ;//建图
 41         }
 42         strcpy( fword, sword ) ;
 43     }
 44     //scanf( "%s", fword ) ;//最恶心的数据输入(At the next line is encrypted message.)
 45     getchar() ;//注意输入的一个text不是一个字符串
 46     gets( fword ) ;//别忘了getchar() ;
 47 }
 48 
 49 void f_indeg()
 50 {
 51     memset( indeg, 0sizeof(indeg) ) ;
 52     forint sn=1; sn<=inn; sn++ ) {
 53         forint en=1; en<=inn; en++ ) {
 54             if( edge[en][sn] ) indeg[sn]++ ;
 55         }
 56         edge[sn][sn] = 0 ;
 57     }//构建indeg[]入度
 58 }
 59 
 60 void f_outdeg()
 61 {
 62     memset( outdeg, 0sizeof(outdeg) ) ;
 63     forint sn=1; sn<=inn; sn++ ) {
 64         forint en=1; en<=inn; en++ ) {
 65             if( sn!=en&&edge[sn][en] ) outdeg[sn]++ ;
 66         }
 67     }
 68 }
 69 
 70 void process()
 71 {
 72 
 73     bool haspwd = false ;
 74     flen = strlen( fword ) ;
 75     forint i=0; i<flen; i++) {//判断是否真的加密了--没有也过了
 76         if( fword[i]>='a'&&fword[i]<'a'+inn ) { haspwd = true ; break ; }
 77     }
 78     if!haspwd ) { printf( "%s\n", fword ) ; return ; }
 79 
 80     forint k=1; k<=inn; k++ ) {//求传递闭包
 81         forint i=1; i<=inn; i++ ) {
 82             forint j=1; j<=inn; j++ ) {
 83                 if( edge[i][k] && edge[k][j] ) {
 84                     edge[i][j] = 1 ; 
 85                 }
 86             }
 87         }
 88     }
 89 
 90     can = true ;//假定可以解密
 91     forint i=1; i<=inn; i++ )    {//判断是否存在环
 92         if( edge[i][i] ) { can = false ; break ; }
 93     }
 94     forint i=1; i<=inn; i++ ) {//判断是否存在孤立点
 95         if!can )    break ;
 96         forint j=1; j<=inn; j++ ) {
 97             if( i == j )    continue ;
 98             if0==edge[i][j]&&0==edge[j][i] )
 99             { can = false ; break ; }
100         }
101     }
102 
103     if!can ) { printf( "Message cannot be decrypted.\n" ) ; return ; }
104 
105     f_outdeg() ;
106 
107     memset( text2pwd, -1sizeof(text2pwd) ) ;
108     memset( pwd2text, -1sizeof(pwd2text) ) ;
109 
110     forint i=1; i<=inn; i++ ) {//建立文本和密文的相互映射
111         int cnt = 0 ; int num = 0 ;
112         forint k=1; k<=inn; k++ ) if( (inn-i)==outdeg[k] ) { cnt++ ; num=k ; }
113         if1 == cnt ) { text2pwd[i] = num ; pwd2text[num] = i ; }
114     }
115 
116     flen = strlen( fword ) ; 
117     forint i=0; i<flen; i++ )
118     {
119         if( fword[i]>='a'&&fword[i]<'a'+inn )
120         {
121             int curnode = fword[i] - 'a' + 1 ;
122             if( pwd2text[curnode]<0 ) { can = falsebreak ; }
123             fword[i] = pwd2text[curnode] + 'a' - 1 ;
124         }
125     }
126 
127     if( can )    printf( "%s\n", fword ) ;
128     else    printf( "Message cannot be decrypted.\n" ) ;
129 }
130 
131 
132 int main()
133 {
134     //while( scanf( "%d", &testnum ) != EOF )
135     scanf( "%d"&testnum ) ;
136     {
137         forint ct=1; ct<=testnum; ct++ )
138         {
139             input() ;
140 
141             process() ;
142 
143             //output() ;
144         }//for(ct)
145     }//while
146 
147     return 0 ;
148 }
149 

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理