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. 2017 济南精英班 Day1

    不管怎么掰都是n*m-1 #include<cstdio> using namespace std; int main() { freopen("bpmp.in",&q ...

  2. CF767 C.Garland DFS

    LINK 题意:给定一棵树,每个节点拥有权值,问能否找到两个点,断开它们与父节点的边能使树分成权值和相等的三部分.权值可以为负 思路:进行两遍DFS,第一遍找最深的子树和为sum/3的节点,标记掉找到 ...

  3. 如何发布一个自定义Node.js模块到NPM(详细步骤,附Git使用方法)

    咱们闲话不多说,直接开始! 由于我从没有使用过MAC,所以我不保证本文中介绍的操作与MAC一致. 文章开始我先假定各位已经在window全局安装了Node.js,下面开始进行详细步骤介绍: 本文本着, ...

  4. 线程池-Threadlocal

    ThreadLoclc初衷是线程并发时,解决变量共享问题,但是由于过度设计,比如弱引用的和哈希碰撞,导致理解难度大.使用成本高,反而成为故障高发点,容易出现内存泄露,脏数据.贡献对象更新等问题.单从T ...

  5. 洛谷 Transformations 方块转换

    Description 一块N x N(1<=N<=10)正方形的黑白瓦片的图案要被转换成新的正方形图案.写一个程序来找出将原始图案按照以下列转换方法转换成新图案的最小方式: 1:转90度 ...

  6. Django之kindeditor

    1.什么是kindeditor? KindEditor是一套开源的HTML可视化编辑器,主要用于让用户在网站上获得所见即所得编辑效果,兼容IE.Firefox.Chrome.Safari.Opera等 ...

  7. [\u4e00-\u9fa5] //匹配中文字符

     [\u4e00-\u9fa5] //匹配中文字符 ^[1-9]\d*$    //匹配正整数^[A-Za-z]+$   //匹配由26个英文字母组成的字符串^[A-Z]+$      //匹配由26 ...

  8. 【IDEA】IDEA中配置tomcat虚拟路径的两种方法

    首先要确保使用的是本地的tomcat服务器,而不是maven插件. -------------------------第一种:使用IDEA工具自动配置(推荐这种)------------------- ...

  9. 消息队列ActiveMQ的使用详解

    通过上一篇文章 <消息队列深入解析>,我们已经消息队列是什么.使用消息队列的好处以及常见消息队列的简单介绍. 这一篇文章,主要带大家详细了解一下消息队列ActiveMQ的使用. 学习消息队 ...

  10. 使用linux下的C操作SQLLITE

    from: http://baike.so.com/doc/1529694.html 由于Linux下侧重使用命令,没有win的操作容易上手,所以在测试C操作SQLITE时会比较容易出现错误,给大家做 ...