Good Bye 2017 部分题解
D. New Year and Arbitrary Arrangement
分析
\(dp[i][j]\) 表示已有 \(i\) 个 \(a\) 和 \(j\) 个 \(ab\) 的情况下继续构造能得到的 \(ab\) 个数的期望。
考虑 DFS 记忆化搜索。
有两个要注意的地方:
令 \(p_a\) 为添加 \(a\) 的概率,\(p_b\) 为添加 \(b\) 的概率。
- 当 \(i + j \geq k\) 时,这个情况下添加一个 \(b\) 构造就停止了,但是在这个 \(b\) 之前显然可以无限添加 \(a\) ,这后面的期望为 \((i + j) * p_b + p_a*(i+j+1)*p_b+p_a^2*(i+j+2)*p_b+...\) 这个式子可以化简,\(O(1)\) 计算。
- 起始状态应该是 \(dp[0][0] = dp[1][0] * p_a + dp[0][0] * p_b\) ,\(dp[0][0] * p_b\) 说明我们可以不断添加前导 \(b\) 。可以发现 \(dp[0][0] = (p_b^0+p_b^1+p_b^2...) * dp[1][0] * p_a=\frac{1}{1-p_b}*dp[1][0]*p_a=dp[1][0]\),所以我们可以直接计算一个以 \(a\) 开头的序列 ,即计算 \(dp[1][0]\) 。
code
#include<bits/stdc++.h>
using namespace std;
const int MOD = 1e9 + 7;
long long POW(long long x, int k) {
long long ret = 1;
while(k) {
if(k & 1) ret = (ret * x) % MOD;
x = x * x % MOD;
k >>= 1;
}
return ret;
}
long long k, pa, pb, dp[1001][1001];
long long dfs(int a, int ab) {
if(a + ab >= k) return a + ab + pa * POW(pb, MOD - 2) % MOD;
if(dp[a][ab] != -1) return dp[a][ab];
return dp[a][ab] = (dfs(a + 1, ab) * pa % MOD + dfs(a, ab + a) * pb % MOD) * POW(pa + pb, MOD - 2) % MOD;
}
int main() {
memset(dp, -1, sizeof dp);
cin >> k >> pa >> pb;
long long p = __gcd(pa, pb);
pa /= p;
pb /= p;
cout << dfs(1, 0) << endl;
return 0;
}
E. New Year and Entity Enumeration
分析
官方题解 很详细了,自己补充几点。
- 考虑 \(m\) 个二进制数,假设 \(m=4\) ,那么有 0001,0010,0100,1000 。这些数一定是某一二进制位有 \(1\) 的最小的数,用 \(f\) 表示这个关系。\(f[0]=0001\),\(f[1]=0010\) ... 。这些数中任意两个数按位与后都为 \(0\) ,那么可以考虑求这个集合划分方法的数目,即贝尔数 ,将各个子集中的数分别按位或起来,举个例子 1000,0100,0011 。这种情况下,第一位和第二位为 \(1\) 的二进制数都是 0011 ,可以发现,各种划分方案中我们得到的 \(f\) 并不会完全相同,所以用这个划分方案数就对应着 \(m\) 位二进制数的解(具体可见官方题解的证明)。
- 题目要求 \(T\) 是 \(S\) 的子集。以样例为例,
11010
00101
11000
竖着去看,有两个 101 两个 010 一个 100 。考虑两个 101 ,我们再横着去看,即 00 11 00 ,我们知道一个 \(k\) 进制数的任意一种方案一定包括 \(k\) 个 \(0\) 和 \(k\) 个 \(1\) 这两个二进制数,我们分开去求一定可以得到满足题目的方案,若 \(b[i]\) 为贝尔数,那么这个答案就是 \(b[2] * b[2] * b[1]\)。
code
#include<bits/stdc++.h>
using namespace std;
const int N = 1e3 + 10;
const int MOD = 1e9 + 7;
long long C[N][N];
map<long long, int> mp;
long long a[N], b[N], c[N][N];
int main() {
for(int i = 0; i < N; i++) {
C[i][0] = 1;
for (int j = 1; j <= i; j++) {
C[i][j] = (C[i - 1][j - 1] + C[i - 1][j]) % MOD;
}
}
int n, m;
cin >> m >> n;
while(n--) {
long long y = 0;
for (int i = 0; i < m; i++) {
long long x;
scanf("%1lld", &x);
a[i] |= x << n;
}
}
for (int i = 0; i < m; i++) {
mp[a[i]]++;
}
b[0] = b[1] = 1;
for (int i = 2; i <= m; i++) {
for (int j = 0; j < i; j++) {
(b[i] += C[i - 1][j] * b[i - j - 1]) %= MOD;
}
}
long long ans = 1;
for(auto it : mp) {
(ans *= b[it.second]) %= MOD;
}
cout << ans << endl;
return 0;
}
F. New Year and Rainbow Roads
分析
以 G 为间隔分组,每个组内单独讨论,要满足题目条件有两种连法:
- 两个 G 不直接相连,通过连接 B 间接连上,也要通过连接 R 间接连上,花费为两倍两个 G 直接的距离。
- 两个 G 直接相连,那么 B 全部连上的话就成了一个环了,考虑去掉花费最大的一条边,R 类似。
code
#include<bits/stdc++.h>
using namespace std;
int preg, prer, preb, fr, lr, fb, lb, mxr, mxb;
int main() {
int n;
cin >> n;
long long ans = 0;
for (int i = 0; i < n; i++) {
int p;
char c[2];
scanf("%d%s", &p, c);
if(c[0] == 'G') {
if(prer) mxr = max(mxr, p - prer);
if(preb) mxb = max(mxb, p - preb);
if(!preg) { // 第一个 G 前面
if(fr) {
ans += p - fr;
}
if(fb) {
ans += p - fb;
}
} else { // 两个 G 之间
ans += min(2LL * (p - preg), 3LL * (p - preg) - mxb - mxr);
}
prer = p;
preb = p;
preg = p;
lb = lr = p;
mxb = mxr = 0;
} else if(c[0] == 'B') {
if(!fb) fb = p;
if(preb) mxb = max(mxb, p - preb);
preb = p;
lb = p;
} else if(c[0] == 'R') {
if(!fr) fr = p;
if(prer) mxr = max(mxr, p - prer);
prer = p;
lr = p;
}
}
if(!preg) {
if(lr != fr) ans += lr - fr;
if(lb != fb) ans += lb - fb;
} else {
ans += lr - preg;
ans += lb - preg;
}
cout << ans << endl;
return 0;
}
G. New Year and Original Order
分析
\(dp[i][j][k][o]\) 表示前 \(i\) 个数中有 \(j\) 个大于等于 \(k\) 的数时的构数方案数,\(o\) 为了保证边界,构造的数不大于 \(X\)。
举个例子,比方说一个数 \(3312\) 我们要累积这个数对答案的贡献,实际上是 \(1233\) ,大于等于 \(1\) 的数有 \(4\) 个,我们可以认为贡献了 \(1111\) ,大于等于 \(2\) 的数有 \(3\) 个,贡献了 \(111\) ,大于等于 \(3\) 的数有 \(2\) 个,贡献了 \(11\) 。
最后枚举 \(k\),累计计算一下答案即可。
code
#include<bits/stdc++.h>
using namespace std;
const int MOD = 1e9 + 7;
const int N = 707;
char s[N];
int dp[N][N][10][2];
int main() {
scanf("%s", s + 1);
int n = strlen(s + 1);
for (int i = 1; i < 10; i++) {
dp[0][0][i][0] = 1;
}
for (int i = 0; i < n; i++) {
for (int j = 0; j <= i; j++) {
for (int k = 1; k < 10; k++) {
for (int o = 0; o < 2; o++) {
for (int p = 0; p <= (!o ? s[i + 1] - '0' : 9); p++) {
(dp[i + 1][j + (p >= k)][k][o | p < s[i + 1] - '0'] += dp[i][j][k][o]) %= MOD;
}
}
}
}
}
long long ans = 0;
for(int k = 1; k < 10; k++) {
for (long long i = 1, p = 1; i <= n; i++, p = (p * 10 + 1) % MOD) {
(ans += p * (dp[n][i][k][0] + dp[n][i][k][1]) % MOD) %= MOD;
}
}
printf("%I64d\n", ans);
return 0;
}
Good Bye 2017 部分题解的更多相关文章
- Good Bye 2017 A B C
Good Bye 2017 A New Year and Counting Cards 题目链接: http://codeforces.com/contest/908/problem/A 思路: 如果 ...
- Codeforces:Good Bye 2018(题解)
Good Bye 2018! 题目链接:https://codeforces.com/contest/1091 A. New Year and the Christmas Ornament 题意: 给 ...
- Hello 2018, Bye 2017
2017年过去了,过去一年经历了太多,改变了好多好多,可以说人生进入了另一个阶段,有可能是成熟吧. 回顾2017 去年换了新工作,离开了将近工作了8年的公司,不带走一丝云彩,为其任劳任怨,最后没有任何 ...
- Good Bye 2017 D. New Year and Arbitrary Arrangement
看了别人的题解 首先这题是一个dp dp[i][j] i是当前有多少个a j是当前有多少个ab子序列 dp[i][j] = dp[i+1][j]*Pa + dp[i][i+j]*Pb; i,j 时加一 ...
- JXOI 2017 简要题解
「JXOI2017」数列 题意 九条可怜手上有一个长度为 \(n\) 的整数数列 \(r_i\) ,她现在想要构造一个长度为 \(n\) 的,满足如下条件的整数数列 \(A\) : \(1\leq A ...
- Good Bye 2017(送命场)
9815人数场,9500+围观神仙打架...断断续续打Codeforces也快有一年啦,第一次打Good Bye场,满怀前排膜tourist的心愿参加了这场送命场,虽然没看到tourist.不过还是得 ...
- 【Good Bye 2017 C】 New Year and Curling
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举前i-1个圆. 哪些圆和它相交. 取圆心纵坐标最大的那个圆就可以了. [代码] #include <bits/stdc++ ...
- 【Good Bye 2017 B】 New Year and Buggy Bot
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举一下全排列.看看有多少种可以到达终点即可. [代码] #include <bits/stdc++.h> using ...
- 【Good Bye 2017 A】New Year and Counting Cards
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 是元音字母或者是奇数就递增. [代码] #include <bits/stdc++.h> using namespace ...
随机推荐
- Dubbo+Zookeeper+SpringMVC+Maven整合实现微服务项目
互联网的发展,网站应用的规模不断扩大,常规的垂直应用架构已无法应对,分布式服务架构以及流动计算架构势在必行,Dubbo是一个分布式服务框架,在这种情况下诞生的.现在核心业务抽取出来,作为独立的服务,使 ...
- CSS浏览器兼容问题集-第二部分
11.高度不适应 高度不适应是当内层对象的高度发生变化时外层高度不能自动进行调节,特别是当内层对象使用margin 或paddign 时. 例: #box {background-color:# ...
- Jenkins 通过ssh 拷贝文件到远程机器上。
想实现的目的是: 在构建之前,从jenkins master上拷贝脚本到需要运行的机器上(linux ssh). 本来是通过publish over ssh 的transfer set可以直接设置,但 ...
- DOM基础操作
本文地址:http://www.cnblogs.com/veinyin/p/7606972.html 1 访问 HTML 元素 常用方法 document.getElementById(" ...
- marquee滚动效果
转载两篇文章: http://blog.sina.com.cn/s/blog_49ce67fc0100atb4.html https://baike.1688.com/doc/view-d359560 ...
- 【洛谷 P4072】 [SDOI2016]征途(斜率优化)
好久没写斜率优化板子都忘了, 硬是交了十几遍.. 推一下柿子就能得到答案为 \[m*\sum x^2-(\sum x)^2\] 后面是个定值,前面简单dp,斜率优化一下就行了. \(f[i][j]=f ...
- styled-components真的好吗?
最近在学习react,然后遇到react中css该怎么写这个问题,上知乎上看了好多大牛都说styled-components好用是大势所趋. 但我自己用了感觉体验却很差,我在这里说说我为啥觉得styl ...
- 2017ACM暑期多校联合训练 - Team 8 1006 HDU 6138 Fleet of the Eternal Throne (字符串处理 AC自动机)
题目链接 Problem Description The Eternal Fleet was built many centuries ago before the time of Valkorion ...
- NYOJ 117 求逆序数 (树状数组)
题目链接 描述 在一个排列中,如果一对数的前后位置与大小顺序相反,即前面的数大于后面的数,那么它们就称为一个逆序.一个排列中逆序的总数就称为这个排列的逆序数. 现在,给你一个N个元素的序列,请你判断出 ...
- 爬虫--PySpider框架
PySpider框架 PySpider框架的作用