D. New Year and Arbitrary Arrangement

分析

\(dp[i][j]\) 表示已有 \(i\) 个 \(a\) 和 \(j\) 个 \(ab\) 的情况下继续构造能得到的 \(ab\) 个数的期望。

考虑 DFS 记忆化搜索。

有两个要注意的地方:

令 \(p_a\) 为添加 \(a\) 的概率,\(p_b\) 为添加 \(b\) 的概率。

  1. 当 \(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)\) 计算。
  2. 起始状态应该是 \(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

分析

官方题解 很详细了,自己补充几点。

  1. 考虑 \(m\) 个二进制数,假设 \(m=4\) ,那么有 0001,0010,0100,1000 。这些数一定是某一二进制位有 \(1\) 的最小的数,用 \(f\) 表示这个关系。\(f[0]=0001\),\(f[1]=0010\) ... 。这些数中任意两个数按位与后都为 \(0\) ,那么可以考虑求这个集合划分方法的数目,即贝尔数 ,将各个子集中的数分别按位或起来,举个例子 1000,0100,0011 。这种情况下,第一位和第二位为 \(1\) 的二进制数都是 0011 ,可以发现,各种划分方案中我们得到的 \(f\) 并不会完全相同,所以用这个划分方案数就对应着 \(m\) 位二进制数的解(具体可见官方题解的证明)。
  2. 题目要求 \(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 为间隔分组,每个组内单独讨论,要满足题目条件有两种连法:

  1. 两个 G 不直接相连,通过连接 B 间接连上,也要通过连接 R 间接连上,花费为两倍两个 G 直接的距离。
  2. 两个 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 部分题解的更多相关文章

  1. Good Bye 2017 A B C

    Good Bye 2017 A New Year and Counting Cards 题目链接: http://codeforces.com/contest/908/problem/A 思路: 如果 ...

  2. Codeforces:Good Bye 2018(题解)

    Good Bye 2018! 题目链接:https://codeforces.com/contest/1091 A. New Year and the Christmas Ornament 题意: 给 ...

  3. Hello 2018, Bye 2017

    2017年过去了,过去一年经历了太多,改变了好多好多,可以说人生进入了另一个阶段,有可能是成熟吧. 回顾2017 去年换了新工作,离开了将近工作了8年的公司,不带走一丝云彩,为其任劳任怨,最后没有任何 ...

  4. 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 时加一 ...

  5. JXOI 2017 简要题解

    「JXOI2017」数列 题意 九条可怜手上有一个长度为 \(n\) 的整数数列 \(r_i\) ,她现在想要构造一个长度为 \(n\) 的,满足如下条件的整数数列 \(A\) : \(1\leq A ...

  6. Good Bye 2017(送命场)

    9815人数场,9500+围观神仙打架...断断续续打Codeforces也快有一年啦,第一次打Good Bye场,满怀前排膜tourist的心愿参加了这场送命场,虽然没看到tourist.不过还是得 ...

  7. 【Good Bye 2017 C】 New Year and Curling

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举前i-1个圆. 哪些圆和它相交. 取圆心纵坐标最大的那个圆就可以了. [代码] #include <bits/stdc++ ...

  8. 【Good Bye 2017 B】 New Year and Buggy Bot

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举一下全排列.看看有多少种可以到达终点即可. [代码] #include <bits/stdc++.h> using ...

  9. 【Good Bye 2017 A】New Year and Counting Cards

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 是元音字母或者是奇数就递增. [代码] #include <bits/stdc++.h> using namespace ...

随机推荐

  1. HTML5 表单自学记录

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  2. Spring Cacheable 注解不缓存null值

    用Cacheable注解时,发现空值,也会被缓存下来.如果我们期望空值不被缓存,可以做如下设置: @Cacheable(key = "#id", unless="#res ...

  3. ItemCF_基于物品的协同过滤

    ItemCF_基于物品的协同过滤 1.    概念 2.    原理 如何给用户推荐? 给用户推荐他没有买过的物品--103 3.    java代码实现思路 数据集: 第一步:构建物品的同现矩阵 第 ...

  4. JVM调优总结(6):新一代的垃圾回收算法

    垃圾回收的瓶颈 传统分代垃圾回收方式,已经在一定程度上把垃圾回收给应用带来的负担降到了最小,把应用的吞吐量推到了一个极限.但是他无法解决的一个问题,就是Full GC所带来的应用暂停.在一些对实时性要 ...

  5. PowerShell入门

    最近需要写个Windows的脚本,以前一直使用cmd.exe来写批处理脚本,这次接触到了PowerShell,准备把学习过程中学到的知识点整理在这里: 相关文章: 1.https://www.cnbl ...

  6. PHP做文件限速下载

    <?php include("DBDA.class.php"); $db = new DBDA(); $bs = $_SERVER["QUERY_STRING&qu ...

  7. javac -cp java -cp

    ///////////////////////////////////////////////////////////////////////////////////// 编译java文件的命令都知道 ...

  8. python-num18 - django进阶一

    一.深入django的路由系统 下面为django的请求生命周期 下面来看下整个生命周期中的路由系统: 在Django的urls中我们可以根据一个URL对应一个函数名来定义路由规则如下: " ...

  9. USB 3.0传输规格

    通用序列总线(USB) 从1996问世以来,一统个人电脑外部连接界面,且延伸至各式消费性产品,早已成为现代人生活的一部分.2000年发表的USB 2.0 High-speed规格,提供了480Mbps ...

  10. 实现在点击asp:button按钮后,不刷新当前页面

    方法1:return false <asp:Button ID="Button1" runat="server" Text="Button&qu ...