【题解】HDU4689 Derangement(有技巧的计数DP)
【题解】HDU4689 Derangement(有技巧的计数DP)
呵呵没告诉我多测组数,然后\(n\le 20,7000\mathrm{ms}\)我写了个状压上去T了
题目大意:
要你求错排的方案数,但要求\(i\)位上的数比\(i\)大/小。大小关系用正负号告诉你,读入一个字符串。
\(O(n2^n)\)
设\(dp(s)\)表示已经放了\(|s|\)个数进去,放的数占满了\(s\)中的位置的方案数
转移太显然直接贴代码
//@winlere
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define DE(s) cerr<<(#s)<<"="<<(s)<<endl;
#define lowbit(x) ((x)&-(x))
using namespace std; typedef long long ll;
inline int qr(){
register int ret=0,f=0;
register char c=getchar();
while(!isdigit(c))f|=c==45,c=getchar();
while(isdigit(c)) ret=ret*10+c-48,c=getchar();
return f?-ret:ret;
}
const int maxn=20;
int n,U;
ll dp[1<<maxn];
char c[maxn];
int main(){
while(~scanf("%s",c)){
n=strlen(c);
dp[0]=1;
U=(1<<n)-1;
for(int t=1;t<=U;++t){
dp[t]=0;
int cnt=0;
for(int g=t;g;g-=lowbit(g)) ++cnt;
for(int i=0,g=t;i<n&&g;++i){
if(g>>i&1){
if(c[i]=='+'&&cnt>i+1) dp[t]+=dp[t^(1<<i)];
if(c[i]=='-'&&cnt<i+1) dp[t]+=dp[t^(1<<i)];
g^=1<<i;
}
}
}
printf("%lld\n",dp[U]);
}
return 0;
}
过不了 别想了
\(O(n^2)\)
考虑+号是一个后缀性的匹配,-号是一个前缀型的匹配。也就是说我们不可能直接把数给选好,要在后面再进行选择。这启发我可以设这样的状态:
\(dp(i,j)\)表示已经考虑前\(i\)个符号,但是需要从后面拉来\(j\)个\(>i\)数来凑齐前面的\(“+”\)。
当前是负号:
当前位置上的数拿来匹配前面的+
\[\]\[
\]\[\]\[
\]当前位置上的数拿来匹配前面一个+
\[\]\[
\]\[\]\[\]
//@winlere
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define DE(s) cerr<<(#s)<<"="<<(s)<<endl
#define lowbit(x) ((x)&-(x))
using namespace std; typedef long long ll;
inline int qr(){
register int ret=0,f=0;
register char c=getchar();
while(!isdigit(c))f|=c==45,c=getchar();
while(isdigit(c)) ret=ret*10+c-48,c=getchar();
return f?-ret:ret;
}
const int maxn=25;
int n,U;
ll dp[maxn][maxn];
char c[maxn];
int main(){
while(~scanf("%s",c+1)){
n=strlen(c+1);
memset(dp,0,sizeof dp);
dp[0][0]=1;
int cnt_minus=0,cnt_plus=0;
for(int t=1;t<=n;++t){
if(c[t]=='-') {
for(int i=0;i<=cnt_plus;++i)
dp[t][i]=dp[t-1][i+1]*(t-1-(cnt_plus-(i+1))-cnt_minus)*(i+1ll)*(i+1<=cnt_plus)+dp[t-1][i]*(t-1-(cnt_plus-i)-cnt_minus);
++cnt_minus;
}
else {
for(int i=0;i<=cnt_plus+1;++i){
if(i) dp[t][i]+=dp[t-1][i-1];
dp[t][i]+=dp[t-1][i]*i;
}
++cnt_plus;
}
}
printf("%lld\n",dp[n][0]);
}
return 0;
}
【题解】HDU4689 Derangement(有技巧的计数DP)的更多相关文章
- 题解报告:hdu 1028 Ignatius and the Princess III(母函数or计数DP)
Problem Description "Well, it seems the first problem is too easy. I will let you know how fool ...
- 计数dp+概率+大数——(抽屉问题解的个数)zoj3380
难的地方在于计数dp..给定范围[1,n]的数去填m个位置,要求不能出现超过I个相同的数, 那就用dp[i][j]表示在阶段i,已经填了j个位置的可能解法,那么只要枚举i填的位置数k∈[0,min(j ...
- [DP之计数DP]
其实说实在 我在写这篇博客的时候 才刚刚草了一道这样类型的题 之前几乎没有接触过 接触过也是平时比赛的 没有系统的做过 可以说0基础 我所理解的计数dp就是想办法去达到它要的目的 而且一定要非常劲非常 ...
- 【BZOJ】2111: [ZJOI2010]Perm 排列计数 计数DP+排列组合+lucas
[题目]BZOJ 2111 [题意]求有多少1~n的排列,满足\(A_i>A_{\frac{i}{2}}\),输出对p取模的结果.\(n \leq 10^6,p \leq 10^9\),p是素数 ...
- 【AtCoder】AGC022 F - Leftmost Ball 计数DP
[题目]F - Leftmost Ball [题意]给定n种颜色的球各k个,每次以任意顺序排列所有球并将每种颜色最左端的球染成颜色0,求有多少种不同的颜色排列.n,k<=2000. [算法]计数 ...
- 【BZOJ】4559: [JLoi2016]成绩比较 计数DP+排列组合+拉格朗日插值
[题意]n位同学(其中一位是B神),m门必修课,每门必修课的分数是[1,Ui].B神碾压了k位同学(所有课分数<=B神),且第x门课有rx-1位同学的分数高于B神,求满足条件的分数情况数.当有一 ...
- 【计数dp】Array Without Local Maximums
参考博客:[CF1068D]Array Without Local Maximums(计数DP) [题意] n<=1e5 dp[i][j][k]表示当前第i个数字为j,第i-1个数字与第i个之间 ...
- [Hdu-5155] Harry And Magic Box[思维题+容斥,计数Dp]
Online Judge:Hdu5155 Label:思维题+容斥,计数Dp 题面: 题目描述 给定一个大小为\(N*M\)的神奇盒子,里面每行每列都至少有一个钻石,问可行的排列方案数.由于答案较大, ...
- TZOJ 5962 Happy Matt Friends(计数DP)
描述 Matt hzs N friends. They are playing a game together. Each of Matt’s friends has a magic number. ...
随机推荐
- Java练习 SDUT-2761_编码
编码 Time Limit: 1000 ms Memory Limit: 65536 KiB Problem Description 给你一个由大写字母组成的组成的字符串,你可以用如下规则对其进行编码 ...
- Redis源码解析:10scan类命令的实现
像keys或者smembers命令,需要遍历数据集合中的所有元素.在一个大的数据库中使用,可能会阻塞服务器较长的一段时间,造成性能问题,因此不适用与生产环境. 在Redis2.8.0中引入了scan类 ...
- websocket实现五子棋联机对战
GoBang.html // 对弈的页面 <!DOCTYPE html> <html> <head> <meta charset="UTF-8&qu ...
- H3C 分页显示
- laravel多表登录出现路由调用错误
public function auth() { // Authentication Routes... $this->get('login', 'Auth\LoginController@sh ...
- ArrayList存储基本类型时的封装类
- H3C 配置高级ACL
- 京东基于Spark的风控系统架构实践和技术细节
京东基于Spark的风控系统架构实践和技术细节 时间 2016-06-02 09:36:32 炼数成金 原文 http://www.dataguru.cn/article-9419-1.html ...
- [转]Redis和Memcache区别,优缺点对比
1. Redis和Memcache都是将数据存放在内存中,都是内存数据库.不过memcache还可用于缓存其他东西,例如图片.视频等等. 2.Redis不仅仅支持简单的k/v类型的数据,同时还提供li ...
- python基础十一之迭代器和生成器
可迭代 内置方法中含有__iter__的数据类型都是可迭代的,只要是可迭代的就可以使用for循环,反之亦然. print(dir('')) # dir()函数可以获取当前数据类型的所有内置方法 返回值 ...