题目大意:

定义一个串:只含有 '( )','[ ]','{ }',3种(6个)字符。

定义 SS 串:

  1. 空串是SS表达式。
  2. 若A是SS表达式,且A串中不含有中括号和大括号,则(A)是SS表达式。
  3. 若A是SS表达式,且A串中不含有大括号,则[A]是SS表达式。
  4. 若A是SS表达式,则{A}是SS表达式。

    定义SS串深度:
  5. 空串深度为0.
  6. 若A可以写成*A'*,其中A‘为SS串,*为任意括号,则\(D(A)=D(A’)+1\)。
  7. 若A可以写成BC的形式,其中B、C均是SS串,则\(D(A)=max\{D(B),D(C) \}\)。

    求由l1个对括号,l2对中括号,l3对大括号,深度为d 构成的SS串的个数。

题解:这是一道字符串上的计数类 dp 问题,一般对于字符串计数类问题都先把字符串划分成若干个独立的部分,即:划分子问题,再进行求解。首先是状态的选择,\(dp[d][i][j][k]\) 表示深度不超过 d,且由 i,j,k 个对应括号构成的SS串的个数,之所以选择深度不超过 d,是因为若选择深度恰好为 d,将很难从子状态转移到当前状态,或者说,要考虑的情况也比较多。转移到状态转移如下:



在看题解时,看到了另一种比较优秀的解释:对于每一个括号序列可以看成是一棵树的 dfs 序列(类似 dfs 序),树的最大深度是 d,求计数。

记忆化搜搜版代码如下

#include <cstdio>
#include <iostream>
#include <algorithm>
#include <memory.h>
using namespace std;
const int mod=11380; int dp[31][11][11][11],l1,l2,l3,d; int dfs(int dep,int x,int y,int z){
int &ans=dp[dep][x][y][z];
if(dep<0)return 0;
if(!dep){
if(x+y+z==0)return 1;
else return 0;
}
if(x+y+z==0)return 1;
if(ans>=0)return ans;
int cnt=0;
for(int i=0;i<x;i++)
for(int j=0;j<=y;j++)
for(int k=0;k<=z;k++)
cnt=(cnt+(long long)dfs(dep-1,i,j,k)*dfs(dep,x-1-i,y-j,z-k))%mod;
for(int j=0;j<y;j++)
for(int k=0;k<=z;k++)
cnt=(cnt+(long long)dfs(dep-1,0,j,k)*dfs(dep,x,y-1-j,z-k))%mod;
for(int k=0;k<z;k++)cnt=(cnt+(long long)dfs(dep-1,0,0,k)*dfs(dep,x,y,z-1-k))%mod;
return ans=cnt;
} int main(){
scanf("%d%d%d%d",&l1,&l2,&l3,&d);
memset(dp,-1,sizeof(dp));
printf("%d\n",(dfs(d,l1,l2,l3)-dfs(d-1,l1,l2,l3)+mod)%mod);
return 0;
}

迭代版代码如下

#include <cmath>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std; const int P = 11380;
const int M = 35;
const int N = 15;
int f[N][N][N][M]; int fun(int a, int b, int c, int d) {
if (a + b + c == 0) return 1;
int tmp = 0;
for (int i = 0; i < c; i++)
tmp = (tmp + f[a][b][c - i - 1][d] * f[0][0][i][d - 1]) % P;
for (int i = 0; i < b; i++)
for (int j = 0; j <= c; j++)
tmp = (tmp + f[a][b - i - 1][c - j][d] * f[0][i][j][d - 1]) % P;
for (int i = 0; i < a; i++)
for (int j = 0; j <= b; j++)
for (int k = 0; k <= c; k++)
tmp = (tmp + f[a - i - 1][b - j][c - k][d] * f[i][j][k][d - 1]) % P;
return tmp;
} int main() {
int l1, l2, l3, d;
cin >> l1 >> l2 >> l3 >> d;
f[0][0][0][0] = 1;
for (int i = 0; i <= l1; i++)
for (int j = 0; j <= l2; j++)
for (int k = 0; k <= l3; k++)
for (int l = 1; l <= d; l++)
f[i][j][k][l] = fun(i, j, k, l);
if (d) f[l1][l2][l3][d] = (f[l1][l2][l3][d] - f[l1][l2][l3][d - 1] + P) % P;
cout << f[l1][l2][l3][d] << endl;
return 0;
}

【POJ1187】陨石的秘密的更多相关文章

  1. [POJ1187] 陨石的秘密

    问题描述 公元11380年,一颗巨大的陨石坠落在南极.于是,灾难降临了,地球上出现了一系列反常的现象.当人们焦急万分的时候,一支中国科学家组成的南极考察队赶到了出事地点.经过一番侦察,科学家们发现陨石 ...

  2. Genotype&&陨石的秘密

    Genotype: Genotype 是一个有限的基因序列.它是由大写的英文字母A-Z组成,不同的字母表示不同种类的基因.一个基因可以分化成为一对新的基因.这种分化被一个定义的规则集合所控制.每个分化 ...

  3. 题解 【POJ1187】 陨石的秘密

    解析 考虑到数据范围,其实我们可以用记搜. 设\(f[a][b][c][d]\)表示还剩\(a\)个'{}',\(b\)个"[]",\(c\)个"()",深度\ ...

  4. poj[1187][Noi 01]陨石的秘密

    Description 公元11380年,一颗巨大的陨石坠落在南极.于是,灾难降临了,地球上出现了一系列反常的现象.当人们焦急万分的时候,一支中国科学家组成的南极考察队赶到了出事地点.经过一番侦察,科 ...

  5. POJ 1187 陨石的秘密 (线性DP)

    题意: 公元11380年,一颗巨大的陨石坠落在南极.于是,灾难降临了,地球上出现了一系列反常的现象.当人们焦急万分的时候,一支中国科学家组成的南极考察队赶到了出事地点.经过一番侦察,科学家们发现陨石上 ...

  6. AcWing 317. 陨石的秘密

    1 -> {} 2 -> [] 3 -> () \(f[d][a][b][c]\) 表示 \([i * 2 - 1, j * 2]\) 这段区间 深度为 d \(1\) 有 \(a\ ...

  7. 常规DP专题练习

    POJ2279 Mr. Young's Picture Permutations 题意 Language:Default Mr. Young's Picture Permutations Time L ...

  8. 别人整理的DP大全(转)

    动态规划 动态规划 容易: , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , ...

  9. dp题目列表

    此文转载别人,希望自己能够做完这些题目! 1.POJ动态规划题目列表 容易:1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 11 ...

随机推荐

  1. easyui datagrid remoteSort的实现 Controllers编写动态的Lambda表达式 IQueryable OrderBy扩展

    EF 结合easy-ui datagrid 实现页面端排序 EF动态编写排序Lambda表达式 1.前端页面 var mainListHeight = $(window).height() - 20; ...

  2. sql 某字段存储另一个表的多个id值并以逗号分隔,现根据id去中文并拼接同样以逗号分隔

    首先介绍用到的两个函数 charindex(要查找的表达式1,表达式2),返回值为表达式1在表达式2中的下标,未找到则返回0.(sql的下标是从1开始的),例如 select charindex('s ...

  3. Iptables防火墙规则使用梳理

    iptables是组成Linux平台下的包过滤防火墙,与大多数的Linux软件一样,这个包过滤防火墙是免费的,它可以代替昂贵的商业防火墙解决方案,完成封包过滤.封包重定向和网络地址转换(NAT)等功能 ...

  4. "Linux内核分析"第七周

    可执行程序的装载 张文俊+原创作品转载请注明出处+<Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.预 ...

  5. 第七组团队项目——专业课程资源共享平台——需求分析&原型设计

    一.项目目标.定位需求: (1)目标:在教师.学生之间建立一个综合的.全面的.快捷的.高效的免费课程和学习资源共享.交流与推荐的开放性平台,实现多维和动态的推荐与分类检索服务. (2)定位:学生与教师 ...

  6. Sprint 冲刺第三阶段第6-10天

    这几天一直都在整理我们之前的内容,检查会不会有细节问题.例如界面跳转.颜色等. 因为一直没办法找到guitub存放位置.于是在这里存放一些主代码. MainActivity.java package ...

  7. 『编程题全队』Beta 阶段冲刺博客一

    1.提供当天站立式会议照片一张 2.每个人的工作 (有work item 的ID) (1) 昨天已完成的工作 孙志威: 1.讨论并制定了Beta阶段的计划 孙慧君: 1.Beta阶段任务的认领 黄华林 ...

  8. sublime text3修改默认配置文件是失败的解决方法

    如果你修改sublime text3的默认配置文件Preferences.sublime-settings失败,现实的错误信息如下图: 其实根据提示信息就好找问题出在哪里了:权限 要想成功的修改默认配 ...

  9. JavaScript 作用域链与闭包

    作用域链 在某个作用域访问某个变量或者函数时,会首先在自己的局部环境作用域中搜寻变量或者函数,如果本地局部环境作用域中有该变量或者函数,则就直接使用找到的这个变量值或者函数:如果本地局部环境作用域中没 ...

  10. Ubuntu 服务器指南

    https://help.ubuntu.com/lts/serverguide/   Jabber Instant Messaging Server https://help.ubuntu.com/l ...