zoukankan      html  css  js  c++  java
  • Trie

    uva1401 这题说的是给出一个由S个不同单词组成的字典和一个长字符串.把这个字符串分解成若干个单词的连接(单词可以重复使用),有多少种方法? 比如有4 个单词 a b cd ab 则abcd有两种分解方法 a+b+cd 和 ab+cd   解法 可以用递推dp[i] 表示从第i个字符开始的字符串可以有多少种的方案组成用暴力的方法去查找果断是不行的 我们可以用字典树去优化这个算法使得这个查找过程变得又花了好多,然后通过dp得到解

    #include <iostream>
    #include <cstdio>
    #include <string.h>
    #include <vector>
    using namespace std;
    const int maxn = 4000*100+10;
    const int maxvlen = 300000+10;
    const int mod=20071027;
    char S[maxn],word[4005][105];
    int len[40005],dp[maxvlen];
    struct Trie{
        int ch[maxn][26];
        int val[maxn];
        int sz;
         void  clear() {sz=1; memset(ch[0],0,sizeof(ch[0]));}
        int idx(char c){   return c-'a'; }
        void insert(char *s ,int v)
        {
              int u=0, n = strlen(s);
              for(int i=0; i<n; ++i){
                     int c = idx(s[i]);
              if(ch[u][c]==0){
                  memset(ch[sz],0,sizeof(ch[sz]));
                  val[sz]=0;
                  ch[u][c]=sz++;
              }
              u=ch[u][c];
            }
            val[u]=v;
        }
        void find_prefixes(char *s, int len, vector<int> &ans)
        {
              int u=0;
              for(int i=0; i<len; ++i) {
                 if(s[i]=='
    ') break;
                 int c = idx(s[i]);
                 if(ch[u][c]==0)break;
                 u=ch[u][c];
                 if(val[u] !=0 )
                 ans.push_back(val[u]);
              }
        }
    }tree;
    int main()
    {
        int n,cas=0,L;
        while(scanf("%s%d",S,&n)==2)
            {
               tree.clear();
               L = strlen(S);
               for( int i = 1; i <= n; ++ i ){
                   scanf("%s",word[i]);
                   len[i]=strlen(word[i]);
                   tree.insert(word[i],i);
                }
                memset(dp,0,sizeof(dp));
                dp[L]=1;
                for(int i=L-1; i>=0; --i)
                    {
                          vector<int> E;
                          tree.find_prefixes(S+i,L-i,E);
                          for(int k=0; k<E.size() ; ++k)
                          {
                               int d = E[k];
                               dp[i]=(dp[i]+dp[ i + len[ d ] ])%mod;
                          }
                    }
                   printf("Case %d: %d
    ", ++cas, dp[0]);
            }
        return 0;
    }
    View Code

  • 相关阅读:
    C#手写日志(txt格式)
    dedecms:解析Robots.txt 协议标准
    dedecms列表页有图调用缩略图无图留空的方法
    简单的随机数实现
    单页面定时跳转的办法小结
    css3通过scale()实现放大功能、通过rotate()实现旋转功能
    rem自适应布局小结001
    Java的BIO、NIO、AIO
    Java实现静态代理、动态代理
    博弈论基础
  • 原文地址:https://www.cnblogs.com/Opaser/p/3862730.html
Copyright © 2011-2022 走看看