BZOJ 1055 区间DP

1055: [HAOI2008]玩具取名

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 1144  Solved: 668
[Submit][Status][Discuss]

Description

某人有一套玩具,并想法给玩具命名。首先他选择WING四个字母中的任意一个字母作为玩具的基本名字。然后他会根据自己的喜好,将名字中任意一个字母用“WING”中任意两个字母代替,使得自己的名字能够扩充得很长。现在,他想请你猜猜某一个很长的名字,最初可能是由哪几个字母变形过来的。

Input

第一行四个整数W、I、N、G。表示每一个字母能由几种两个字母所替代。接下来W行,每行两个字母,表示W可以用这两个字母替代。接下来I行,每行两个字母,表示I可以用这两个字母替代。接下来N行,每行两个字母,表示N可以用这两个字母替代。接下来G行,每行两个字母,表示G可以用这两个字母替代。最后一行一个长度不超过Len的字符串。表示这个玩具的名字。

Output

一行字符串,该名字可能由哪些字母变形而得到。(按照WING的顺序输出)如果给的名字不能由任何一个字母变形而得到则输出“The name is wrong!”

Sample Input

1 1 1 1
II
WW
WW
IG
IIII

Sample Output

IN

HINT

W可以变成II所以IIII可以缩成WW IN均能变成WW所以WW又可以缩成I或者N 所以最终答案应该按照“WING”的顺序输出IN

[数据范围]

100%数据满足Len<=200,W、I、N、G<=16

题解:暴力区间DP

#include<cstdio>
#include<iostream>
#include<cstring>
#include<vector>
using namespace std;
#define maxn 201
vector<int > mp[maxn];
int a,b,c,d,sum;
char ch[3],s[201];
int dp[201][201][101];
char next[400];
int str[555];
void stt(char a[])
{
    sum=0;
    if(a[0]==‘W‘)
    {
        sum+=1;
    }else if(a[0]==‘I‘)sum+=2;else if(a[0]==‘N‘)sum+=3;else if(a[0]=‘G‘)sum+=4;
    sum*=10;
        if(a[1]==‘W‘)
    {
        sum+=1;
    }else if(a[1]==‘I‘)sum+=2;else if(a[1]==‘N‘)sum+=3;else if(a[1]=‘G‘)sum+=4;
}
int  main()
{

   next[1]=‘W‘;str[‘W‘]=1;
   next[2]=‘I‘;str[‘I‘]=2;
   next[3]=‘N‘;str[‘N‘]=3;
   next[4]=‘G‘;str[‘G‘]=4;
     scanf("%d%d%d%d",&a,&b,&c,&d);
     for(int i=1;i<=a;i++)
     {
         scanf("%s",ch);
         stt(ch);
         //cout<<sum<<endl;
         mp[sum].push_back(1);
     }
     for(int i=1;i<=b;i++)
     {
         scanf("%s",ch);
         stt(ch);
         mp[sum].push_back(2);
     }
      for(int i=1;i<=c;i++)
     {
         scanf("%s",ch);
         stt(ch);
        mp[sum].push_back(3);
     }
      for(int i=1;i<=d;i++)
     {
         scanf("%s",ch);
         stt(ch);
         mp[sum].push_back(4);
     }
     scanf("%s",s);
     memset(dp,0,sizeof(dp));
     for(int i=0;i<strlen(s);i++){
        dp[i][i][str[s[i]]]=1;
     }
     int n=strlen(s);
    for(int l=2;l<=n;l++)
    {
        for(int i=0;i<n-l+1;i++)
        {
            for(int k=i;k<i+l-1;k++)
            {
                for(int j=1;j<=4;j++)
                    for(int h=1;h<=4;h++)
                {
                    if(dp[i][k][j]&&dp[k+1][i+l-1][h])
                    {
                        for(int f=0;f<mp[j*10+h].size();f++)
                        {
                            dp[i][i+l-1][mp[j*10+h][f]]=1;
                        }
                    }
                }
            }
        }
    }
    int flag=0;
    for(int i=1;i<=4;i++)
        if(dp[0][n-1][i])cout<<next[i],flag=1;
        if(!flag)cout<<"The name is wrong!";
    cout<<endl;
    return 0;
}

代码君

时间: 2024-08-29 14:19:04

BZOJ 1055 区间DP的相关文章

bzoj 1996 区间dp

1996: [Hnoi2010]chorus 合唱队 Time Limit: 4 Sec  Memory Limit: 64 MBSubmit: 1727  Solved: 1115[Submit][Status][Discuss] Description Input Output Sample Input 4 1701 1702 1703 1704 Sample Output 8 HINT 要想知道[l,r]的初始队形的方案数,如果我们知道[l,r-1]和[l+1,r]有几种初始方案的话似乎就

BZOJ 1068 (区间DP)

题意:字符串的压缩,f[l][r][0]代表还没M,f[l][r][1]代表有M. 1 #include<cstdio> 2 #include<cmath> 3 #include<cstring> 4 #include<algorithm> 5 #include<iostream> 6 int f[55][55][2],n; 7 char s[555]; 8 bool ok(int l,int r){ 9 int len=(r-l+1)/2; 1

bzoj 1055 [HAOI2008]玩具取名(区间DP)

1055: [HAOI2008]玩具取名 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1258  Solved: 729[Submit][Status][Discuss] Description 某 人有一套玩具,并想法给玩具命名.首先他选择WING四个字母中的任意一个字母作为玩具的基本名字.然后他会根据自己的喜好,将名字中任意一个字母用 “WING”中任意两个字母代替,使得自己的名字能够扩充得很长.现在,他想请你猜猜某一个很长的名字,最初可能

BZOJ 1055 玩具取名(区间DP)

很显然的区间DP,定义dp[i][j][k], 如果dp[i][j][k]=1表示字符串[i,j]可以组成k字符. # include <cstdio> # include <cstring> # include <cstdlib> # include <iostream> # include <vector> # include <queue> # include <stack> # include <map>

bzoj 1055 [HAOI2008]玩具取名 区间dp

题面 题目传送门 解法 直接区间dp即可 时间复杂度:\(O(16n^3)\) 代码 #include <bits/stdc++.h> #define N 210 using namespace std; struct Node { int x, y; } a[5][N]; int s[5], p[5][5][5]; bool f[N][N][5], vis[N][N][5]; int num(char ch) { if (ch == 'W') return 1; if (ch == 'I')

BZOJ 1090 字符串折叠(区间DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1090 题意:字符串AAAAAAAAAABABABCCD的最短折叠为9(A)3(AB)CCD,注意数字的长度和圆括号都算最后长度.求一种折叠方式使得总长度最小. 思路:f[L][R]=min(R-L+1,f[L][i]+f[i+1][R]),另外若[L,R]能由[i+1,R]重复若干次,则也可用折叠后的长度更新f[L][R]. char s[N]; int f[N][N],n; int

[BZOJ 1068] [SCOI2007] 压缩 【区间 DP 】

题目链接:BZOJ - 1068 题目分析 这种区间 DP 之前就做过类似的,也是字符串压缩问题,不过这道题稍微复杂一些. 需要注意如果某一段是 S1S1 重复,那么可以变成 M + Solve(S1) + R ,不过这个 Solve(S1) 中不能在中间有 M ,否则后面的 R 向前找到的 M 就不再是开头的 M 了. 代码 #include <iostream> #include <cstdio> #include <cstring> #include <al

[BZOJ 1260][CQOI2007]涂色paint 题解(区间DP)

[BZOJ 1260][CQOI2007]涂色paint Description 假设你有一条长度为5的木版,初始时没有涂过任何颜色.你希望把它的5个单位长度分别涂上红.绿.蓝.绿.红色,用一个长度为5的字符串表示这个目标:RGBGR. 每次你可以把一段连续的木版涂成一个给定的颜色,后涂的颜色覆盖先涂的颜色.例如第一次把木版涂成RRRRR,第二次涂成RGGGR,第三次涂成RGBGR,达到目标. 用尽量少的涂色次数达到目标. Input 输入仅一行,包含一个长度为n的字符串,即涂色目标.字符串中的

[BZOJ 1652][USACO 06FEB]Treats for the Cows 题解(区间DP)

[BZOJ 1652][USACO 06FEB]Treats for the Cows Description FJ has purchased N (1 <= N <= 2000) yummy treats for the cows who get money for giving vast amounts of milk. FJ sells one treat per day and wants to maximize the money he receives over a given