2017百度之星资格赛:1005. 寻找母串(卡特兰数+分块打表)
2017-08-06 02:54
302 查看
寻找母串
Accepts: 105
Submissions: 887
Time Limit: 2000/1000 MS (Java/Others)
Memory Limit: 32768/32768 K (Java/Others)
Problem Description
对于一个串S,当它同时满足如下条件时,它就是一个01偏串: 1、只由0和1两种符组成; 2、在S的每一个前缀中,0的个数不超过1的个数; 3、S中0的个数和1的个数相等。
现在给定01偏串S,请计算一下S在所有长度为n的01偏串中作为子串出现的次数的总和。 由于结果比较大,结果对1e9+7取余后输出。
样例解释: 在第二个样例中,长度为4的偏串共两个1010,1100。10在1010中出现了两次,在1100中出现了1次。所以答案是3。
Input
第一行给出一个整数T(1<=T<=40),表示测试数据的数目。 每一组测试包含一个整数n和字符串S,中间用空格分开。(1<=|S|<=100000,1<=n<=1000000000)
输入保证S是一个01偏串。
Output
对于每一组数据,输出一个整数占一行,表示答案。
Sample Input
Copy
2 2 10 4 10
Sample Output
1 3
可能我姿势不对。。这表怎么就打了1年
这题看起来束手无策,但其实你可以先写个暴力观察小数据,这个时候你就会发现其实这题比你想象的要简单的多
(比如你会发现其实答案之和字符串的长度|S|有关,和字符串的内容是无关的)
没错!这题有规律,答案就是
但是别高兴的太早,这个时候你又会发现其实这题比你想象的要难得多
因为这题的n范围巨大(约10亿)而求10亿的组合数是做不到的
所有先考虑化简公式看看,令x = (n-|S|)/2+1有
其中F[x]是第x项卡特兰数
而卡特兰数有O(n)的递推公式
这样的话理论上可以O(n)求出所有的答案,可还是不行。。。
复杂度已经不能再优化了。。所以只能考虑打表
可是你又开不了10亿的数组,但是这题也没有说要O(1)询问呀
没错!分块打表
你只需要后台O(n)暴力出第100000个卡特兰数,第200000个卡特兰数……第500000000个卡特兰数就好了
也就是开个5000+的数组c[],其中c[i]是第100000*i个卡特兰数
然后对于每组询问(n, |S|),先计算x=(n-|S|)/2+1,然后再找到x所在的那一块(比x小离x最近的100000的倍数),
之后暴力转移就好,最多转移100000次
因为有除法所以要乘法逆元,总体复杂度O(100000log(n))
还有注意n为奇数的时候答案一定为0,因为母串要保证0和1的数量相等,所以不存在长度为奇数的母串
除此之外,n<|S|答案也为0
下面附上暴力程序+打表程序
#include<stdio.h> #include<string.h> char str[100005], jud[100005]; int Jud(int n) { int i, s1, s0; s1 = s0 = 0; for(i=1;i<=n;i++) { if(jud[i]=='1') s1++; else s0++; if(s0>s1) return 0; } if(s1!=s0) return 0; return 1; } int main(void) { int T, n, m, i, j, k, sum, ans; scanf("%d", &T); while(T--) { sum = ans = 0; scanf("%d%d", &n, &m); for(i=1;i<=m/2;i++) str[i] = '1'; for(i;i<=m;i++) str[i] = '0'; str[m+1] = 0; m = strlen(str+1); for(i=0;i<=(1<<n)-1;i++) { for(j=0;j<=n-1;j++) { jud[j+1] = '0'; if(i&(1<<j)) jud[j+1] = '1'; } jud[j+1] = 0; if(Jud(n)) { sum++; for(j=1;j<=n;j++) { for(k=1;k<=m;k++) { if(jud[j-1+k]!=str[k] || j-1+k>n) break; } if(k==m+1) ans++; } } } printf("%d\n", ans); } return 0; } /* #include<stdio.h> #define mod 1000000007 #define LL long long LL Pow(LL a, LL b) { LL sum = 1; while(b) { if(b%2) sum = sum*a%mod; a = a*a%mod; b /= 2; } return sum; } int main(void) { LL i, now = 1; freopen("ct.txt", "w", stdout); printf("{1"); for(i=1;i<=500000000;i++) { now = 2*(2*i-1)*now%mod*Pow(i+1, mod-2)%mod; if(i%100000==0) printf(",%lld", now); } printf("};"); return 0; } */
相关文章推荐
- HDU 6084 | 2017百度之星资格赛 1005 卡特兰数 + 分块打表
- 2017百度之星资格赛(A)1005 今夕何夕(日期计算)
- Astar2017资格赛1005 寻找母串
- 百度之星 1005寻找母串
- 2017百度之星 资格赛 1003
- 2017百度之星资格赛1003 度度熊与邪恶大魔王(完全背包)
- 百度之星资格赛1003 度度熊与邪恶大魔王(dp)--2017
- hdu 6082 2017百度之星资格赛
- HDU 6084 寻找母串(卡特兰数)
- HDU 6084 寻找母串(卡特兰数)
- 2017百度之星初赛(A) 1005 今夕何夕(阅读题)
- 2015百度之星资格赛--1005 下棋【双向BFS/找规律】
- 2017百度之星资格赛:1002. 度度熊的王国战略
- 卡特兰数+分段打表 hdu6084寻找母串
- 2017百度之星初赛:B-1005. 度度熊的交易计划(最小费用流)
- HDU 6148 && 2017 百度之星复赛 1005 Valley Numer(数位dp)
- 2017百度之星资格赛 1002-图 1003-DP 1004-01背包(解选择方案)
- (最小费用流)hdu 6118(2017百度之星初赛B 1005) 度度熊的交易计划
- 2017百度之星资格赛1002度度熊的王国战略
- 2017百度之星资格赛 1003:度度熊与邪恶大魔王(DP)