usaco 4.3 Letter Game 搜索

 简单搜索题,几乎不需要优化,直接枚举一个或者两个单词,判断其组成字符的个数是否小于给定串对应的字符个数,满足则更新答案。搜索过程中如果需要单词里的字符不再给定串里,则提前跳出。

 

  1. /*
  2. PROG: lgame
  3. LANG: C++
  4. ID: heben991
  5. */
  6. #include <iostream>
  7. #include <algorithm>
  8. using namespace std;
  9. const int N = 60010, M = 12, L = 30;
  10. const int value[L]={2,5,4,4,1,6,5,5,1,7,6,3,5,2,3,5,7,2,1,2,4,6,6,7,5,7};
  11. char dict[N][M], *ans[N][2];
  12. int len, n, size, l[N];
  13. int num[L], h1[L],h2[L];
  14. int sum_value(int k)
  15. {
  16.     int i, sum=0;
  17.     for(i = 0; i < l[k]; ++i) sum += value[dict[k][i]-'a'];
  18.     return sum;
  19. }
  20. int main()
  21. {
  22.     int i, j, k, t, sum, v1, v2, ma=0;
  23.     char s[M],s1[M],s2[M],ch,*c;
  24.     freopen("lgame.dict","r",stdin);
  25.     freopen("lgame.out","w",stdout);
  26.     while(1)
  27.     {
  28.         gets(dict[n]);
  29.         if(dict[n][0]=='.')break;
  30.         l[n] = strlen(dict[n]);
  31.         ++n;
  32.     }
  33.     freopen("lgame.in","r",stdin);
  34.     gets(s);
  35.     len = strlen(s);
  36.     for(i = 0; i < len; ++i) num[s[i]-'a']++;
  37.     for(i = 0, sum = 0; i < len; ++i)
  38.         sum += value[s[i]-'a'];
  39.     for(i = 0; i < n; ++i)
  40.     {
  41.         if(l[i] > len) continue;
  42.         {
  43.             for(k = 0; k < l[i]; ++k) // < l[i], not < len
  44.             {
  45.                 if(num[dict[i][k]-'a']==0) break;  // -'a'
  46.             }
  47.             if(k < l[i]) continue;
  48.             for(k = 0; k < len; ++k) h1[s[k]-'a']=0;
  49.             for(k = 0; k < l[i]; ++k) h1[dict[i][k]-'a']++;
  50.             for(k = 0; k < len; ++k)
  51.                 if(num[s[k]-'a']<h1[s[k]-'a'])break;
  52.             v1 = sum_value(i);
  53.             if(k==len && l[i]<=len && v1 >= ma)
  54.             {
  55.                 if(v1>ma)  ma = v1, size = 0;
  56.                 ans[size++][0]=dict[i];
  57.                 // don't continue here
  58.             }
  59.         }
  60.         {
  61.             for(j = i; j < n; ++j)
  62.             if(l[j]+l[i]<=len)
  63.             {
  64.                 for(k = 0; k < l[j]; ++k)
  65.                 {
  66.                     if(num[dict[j][k]-'a']==0) break;;
  67.                 }
  68.                 if(k < l[j]) continue;
  69.                 for(k = 0; k < len; ++k) h2[s[k]-'a']=0;
  70.                 for(k = 0; k < l[j]; ++k) h2[dict[j][k]-'a']++;
  71.                 for(k = 0; k < len; ++k)
  72.                     if(num[s[k]-'a']<h1[s[k]-'a']+h2[s[k]-'a'])
  73.                     {
  74.                         break;
  75.                     }
  76.                 v2 = sum_value(j);
  77.                 if(k==len && v1+v2 >= ma)
  78.                 {
  79.                     if(v1+v2 > ma) ma = v1+v2, size=0;
  80.                     ans[size][0]=dict[i];
  81.                     ans[size++][1]=dict[j];
  82.                 }
  83.             }
  84.         }
  85.     }
  86.     printf("%d/n", ma);
  87.     for(i = 0; i < size; ++i)
  88.     {
  89.         printf("%s", ans[i][0]);
  90.         if(ans[i][1]) printf(" %s", ans[i][1]);
  91.         puts("");
  92.     }
  93.     return 0;
  94. }

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值