• P1666 前缀单词


    P1666 前缀单词

    题目描述

    一组单词是安全的,当且仅当不存在一个单词是另一个单词的前缀,这样才能保证数据不容易被误解。现在你手上有一个单词集合S,你需要计算有多少个子集是安全的。

    注意空集永远是安全的。

    输入输出格式

    输入格式:

    第一行一个数n,表示集合的大小,以下n行。每行一个由’a’……’z’构成的字符串。

    【数据规模】

    对30%的数据,满足1≤n≤10;

    对于100%的数据,满足1≤n≤50;字符串长度≤50,没有两个字符串是完全相同的。

    输出格式:

    安全子集的个数。

    输入输出样例

    输入样例#1: 复制
    3
    hello
    hell
    hi
    输出样例#1: 复制
    6

    枚举小状态来找状态转移方程

    洛谷题解

    我们预处理一个f[i][j]表示第i个单词与第j个单词是否能共存,

    然后考虑一个dp[i]表示在前i个单词中,必须包含第i个单词的子集个数,首先dp[i]=1(只包含自己一个元素的一个子集方案数)

    那么如果前面存在一个j<i,并且f[i][j]=1(即i与j可以共存),那么j所有的子集方案数加上一个i元素就形成了对应新的方案,那么状态就由j转移到i了,最后,直接累计dp[1....n]即可,当然最后还要算上空集哟。

    参考代码:

     1 #include<algorithm>
     2 #include<iostream>
     3 #include<string>
     4 #define REP(i,a,b) for (register int i=(a);i<=(b);i++)
     5 using namespace std;
     6 const int N=61;
     7 string a[N];
     8 long long f[N][N],dp[N];
     9 int n;
    10 inline bool calc(int i,int j){
    11     if (a[i].size()>a[j].size())swap(i,j);
    12     return a[j].find(a[i])!=0;
    13 }
    14 int main(){
    15     ios::sync_with_stdio(false);
    16     cin>>n;
    17     REP(i,1,n)cin>>a[i];
    18     sort(a+1,a+n+1);
    19     REP(i,1,n){
    20         dp[i]=1;
    21         REP(j,1,n)f[i][j]=calc(i,j);
    22     }
    23     REP(i,1,n)REP(j,i,n)dp[j]+=f[i][j]?dp[i]:0;
    24     long long ret=0;
    25     REP(i,1,n)ret+=dp[i];
    26     cout<<ret+1;
    27     return 0;
    28 }
  • 相关阅读:
    使用iOS网络请求
    Invalid RNPermission 'ios.permission.xxx'. should be one of: ( )
    React Native 报错 Error: spawn EACCES 权限
    React Native 适配Android物理返回键,实现连续两次点击退出
    图解:平衡二叉树,AVL树
    NOIP 骗分技巧
    P1004 方格取数
    5. 最长回文子串
    全链路压测自动化实践
    深度学习在美团配送ETA预估中的探索与实践
  • 原文地址:https://www.cnblogs.com/Renyi-Fan/p/7741762.html
Copyright © 2020-2023  润新知