[AtCoder ARC061F]Card Game for Three 组合数好题
总结:组合数
这$F$题好难啊...只会部分分做法,下面两个方法都是部分分做法。满分做法我去看看...会的话就补一下
部分分做法
方法1:
首先$A$能赢的条件很明显,假设在所有的牌里面取出$A$张$A$牌,$B$张$B$牌,$C$张$C$牌,那么$A$能赢当且仅当$A=n,B<m,C<k$
所以假设我们在拿出了$n$张$A$牌的情况下,中间穿插着拿了$B$张$B$牌,$C$张$C$牌,则有
$$\sum_{i=0}^{i<=m+k}C(n-1,i+n-1)*3^{m+k-i}*\sum_{j=0}^{j<=m,j<=k}C(i,j)$$
首先在$i+n-1$张牌中取$n-1$张$A$牌的方案为$C(n-1,n+i-1)$
注意:最后一张牌一定需要是$A$,所以就只能有$n-1$
而且剩下的牌数的排列为$3^{m+k-i}$,要乘上去
以及在$i+n-1$中$B$和$C$的排列为$C(i,j)$,也要乘起来
所以就得到了上面的公式
但是,上面的做法我打挂了...这个式子是对的,我算后面那个$\sum_{j=0}^{j<=m,j<=k}C(i,j)$那里我没有枚举好...查不出来啊...
我在理解了满分做法之后终于发现我挂在哪里了...
在枚举后面的东西的时候我没有分类讨论。但是改完后貌似还分少了一类...
$subtask_1$10个点我错了$2$个...然后$subtask_2$被我强行水过去$2$个点..
用$C$牌数量来分类:分为$j<k$,$j<m$,$j<i$(第三类我没分...不想去改了...)
对于第一种情况:$x=\sum_{j=0}^{j<k}C(i,j)$
对于第二种情况:$x=\sum_{j=0}^{j<m}C(i,k)$
对于第三种情况:请读者独立思考(知道有第三种情况是因为我去看了一下满分的做法...)
#include <bits/stdc++.h> using namespace std ; #define N 5010
#define ll long long
const int mod = 1e9 + ; int n , m , k ;
ll fac[ N ] , ifac[ N ] ; ll power( ll a ,ll b ) {
ll base = a , ans = ;
while( b ) {
if( b& ) ans = ( ans * base ) % mod ;
base = ( base * base ) % mod ;
b >>= ;
}
return ans ;
}
ll mul( ll x ,ll y ) {
return ( 1ll * x * y ) % mod ;
} ll inv( ll x ) {
return power( x , mod - ) % mod ;
} int main() {
scanf( "%d%d%d" , &n , &m , &k ) ; fac[ ] = ;
for( int i = ; i < N ; i ++ ) fac[ i ] = mul( fac[ i - ] , i ) ;
for( int i = ; i < N ; i ++ ) ifac[ i ] = inv( fac[ i ] ) ; ll ans = , sum = n + k + m ; for( int i = n ; i <= sum; i ++ ) {// 取出n张A牌
for( int j = ; j <= m ;j ++ ) {//B牌数量
int C = i - n - j ;//C牌数量
if( C < || C > k ) continue ;
ll tmp = mul( fac[ i - ] , mul( ifac[ n - ] , mul( ifac[ j ] ,ifac[ C ] ) ) ) ;
tmp = mul( tmp , power( , sum - i ) ) ;
ans = ( ans + tmp ) % mod ;
}
} printf( "%lld\n" , ans ) ; }
部分分做法1(有一定错误...)
方法2:
换一种想法,在前$i$张卡片中拿出$n$张$A$,$j$张$B$,$C$张$C$
则可以推出一个公式
$$\sum_{i=0}^{i<=n+m+k}\frac{(i-1)!}{(n-1)!j!C!}3^{n+k+m-i}$$
如果看得懂那个方法一的话这个公式大概也是看得懂的吧...
就是$i-1$的全排列数除掉$n-1$的全排列数和$j$的全排列数和$C$的全排列数
这个是很基础的一个组合数的常识,这样子得到的就是我们要的当前情况的方案数,记住也要把当前剩下的那些的方案数也乘上去,即$3^{n+m+k-i}$
然后这里的除法是$mod$ $m$意义下的,所以要求一下逆元,代码中的$fac[i]$即为$i!$的值,$ifac[i]$即为$i!$的逆元
#include <bits/stdc++.h> using namespace std ; #define N 5010
#define ll long long
const int mod = 1e9 + ; int n , m , k ;
ll fac[ N ] , ifac[ N ] ; ll power( ll a ,ll b ) {
ll base = a , ans = ;
while( b ) {
if( b& ) ans = ( ans * base ) % mod ;
base = ( base * base ) % mod ;
b >>= ;
}
return ans ;
}
ll mul( ll x ,ll y ) {
return ( 1ll * x * y ) % mod ;
} ll inv( ll x ) {
return power( x , mod - ) % mod ;
} int main() {
scanf( "%d%d%d" , &n , &m , &k ) ; fac[ ] = ;
for( int i = ; i < N ; i ++ ) fac[ i ] = mul( fac[ i - ] , i ) ;
for( int i = ; i < N ; i ++ ) ifac[ i ] = inv( fac[ i ] ) ; ll ans = , sum = n + k + m ; for( int i = n ; i <= sum; i ++ ) {// 取出n张A牌
for( int j = ; j <= m ;j ++ ) {//B牌数量
int C = i - n - j ;//C牌数量
if( C < || C > k ) continue ;
ll tmp = mul( fac[ i - ] , mul( ifac[ n - ] , mul( ifac[ j ] ,ifac[ C ] ) ) ) ;
tmp = mul( tmp , power( , sum - i ) ) ;
ans = ( ans + tmp ) % mod ;
}
} printf( "%lld\n" , ans ) ; }
部分分做法2
钻研了很久的题解,终于差不多搞懂满分做法了...
满分做法
方法一确实是对的。这个满分做法就是用来改进那里的
观察耗时,耗时基本都是花费在枚举后面那一段,所以考虑优化那一段...(这个做法很清奇...正常写组合数不应该是优化那个式子吗...)
$$\sum_{i=0}^{i<=m+k}C(n-1,i+n-1)*3^{m+k-i}*\sum_{j=0}^{j<=m,j<=k}C(i,j)$$
还是这个式子,我们来优化掉后面那个$\sum_{j=0}^{j<=m,j<=k}C(i,j)$(就是我写错了还查不出来的那个玩意...不!我写到这里的时候忽然发现我在枚举的时候没有分类讨论!)
我们把后面那个$\sum$拆掉,分三部分讨论
假设我们现在手上有$x$张$C$牌
当$x<k$时,随便取...(C的数量都比你要取的多了所以肯定不会超限)即$\sum_{i=0}^{x<k}C(x,i)$
当$x<m$时,$C$的数量在$i$以内,同时不能取超过$k$个,即$\sum_{i=0}^{i<m}C(x,i)$
当$m<=x<m+k$时,$B,C$数量均不超过$i$,同时$B$不能取超过$m$个,$C$不能取超过$k$个,即$\sum_{i-m+1}^{i<k}C(x,i)$
然后怎么优化呢
如果你熟知杨辉三角这个东西的话,大概就会知道怎么优化了
假设我们已经知道了$i-1$时$x$的值,那么其实可以推出下面的几个式子:
情况$1$:$x=x*2$
情况$2$:$x=x*2-C(i,k)$
情况$3$:$x-x*2-C(i,k)-C(i,m)$
这个挺容易推的吧...如果部分分做法的第一种有推出来那么这个优化也就顺理成章了的样子(但是我推出来了一半...)
#include <bits/stdc++.h> using namespace std ; const int mod = 1e9 + ;
const int N = ;
#define ll long long int n , m , k ;
ll fac[ N ] , ifac[ N ] , p[ N ] ; ll mul( ll x , ll y ) {
return ( 1ll * x * y ) % mod ;
} ll add( ll x , ll y ) {
return ( x + y ) % mod ;
} ll power( ll a , ll b ) {
int ans = , base = a ;
while( b ) {
if( b& ) ans = mul( ans , base ) ;
base = mul( base , base ) ;
b >>= ;
}
return ans ;
} ll inv( ll x ) {
return power( x , mod - ) % mod ;
} ll C( ll x , ll y ) {
return ( fac[ x ] * ifac[ y ] % mod * ifac[ x - y ] % mod ) % mod ;
} int main() {
scanf( "%d%d%d" , &n , &m , &k ) ;
fac[ ] = 1ll ;
p[ ] = 1ll ;
for( int i = ; i < N ; i ++ ) {
fac[ i ] = fac[ i - ] * i % mod ;
p[ i ] = p[ i - ] * 3ll % mod ;
}
for( int i = ; i < N ; i ++ ) {
ifac[ i ] = inv( fac[ i ] ) ;
}
ll ans = , x = 1ll ;
n -- ;
if( m < k ) swap( m , k ) ;
for( int i = ; i <= m + k ; i ++ ) {
ans = ( ans + C( n + i , n ) * p[ m + k - i ] % mod * x ) % mod ;
/*
for( int j = 0 ; j <= min( i , m ) ; j ++ ) {
if( i - j > k || i - j < 0 ) break ;
x = add( x , C( i , j ) ) ;
}
*/
if( i < k ) x = ( x * 2ll ) % mod ;
else if( i < m ) x = ( x * 2ll - C( i , k ) ) % mod ;
else x = ( x * 2ll - C( i , k ) - C( i , m ) ) % mod ;
}
printf( "%lld\n" , add( ans , mod ) ) ;
return ;
}
数学题真的虐哭我...这题我研究了$3$天...
不过确实是一道组合数的好题...做完后觉得对组合数这玩意的理解更深了一些...
[AtCoder ARC061F]Card Game for Three 组合数好题的更多相关文章
- HDU 6114 Chess【逆元+组合数】(组合数模板题)
<题目链接> 题目大意: 車是中国象棋中的一种棋子,它能攻击同一行或同一列中没有其他棋子阻隔的棋子.一天,小度在棋盘上摆起了许多車……他想知道,在一共N×M个点的矩形棋盘中摆最多个数的車使 ...
- AtCoder Beginner Contest 022 A.Best Body 水题
Best Body Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://abc022.contest.atcoder.jp/tasks/abc02 ...
- 【atcoder】Two Sequences [arc092 D](思维题)
题目传送门:https://arc092.contest.atcoder.jp/tasks/arc092_b 这场arc好难啊...这场感觉不像正常的arc...其实这道题还可以更早写出来的,但是蒟蒻 ...
- Atcoder Tenka1 Programmer Contest D: IntegerotS 【思维题,位运算】
http://tenka1-2017.contest.atcoder.jp/tasks/tenka1_2017_d 给定N,K和A1...AN,B1...BN,选取若干个Ai使它们的或运算值小于等于K ...
- Atcoder CODE FESTIVAL 2016 qual C 的E题 Encyclopedia of Permutations
题意: 对于一个长度为n的排列P,如果P在所有长度为n的排列中,按照字典序排列后,在第s位,则P的value为s 现在给出一个长度为n的排列P,P有一些位置确定了,另外一些位置为0,表示不确定. 现在 ...
- AtCoder 杂题训练
前言: 因为要普及了,今年没一等就可以退役去学文化课了,所以暑假把历年noip普及组都刷了一遍,离noip还有50+天,想弄点强化训练什么的. 想了想,就这些天学文化课之余有空就把AtCoder之前那 ...
- SDUT1061Binomial Showdown(组合数)
http://acm.sdut.edu.cn/sdutoj/problem.php?action=showproblem&problemid=1061 题意 : 表示这个题的英文没看懂,就看懂 ...
- Iroha and a Grid AtCoder - 1974(思维水题)
就是一个组合数水题 偷个图 去掉阴影部分 把整个图看成上下两个矩形 对于上面的矩形求出起点到每个绿点的方案 对于下面的矩形 求出每个绿点到终点的方案 上下两个绿点的方案相乘后相加 就是了 想想为什么 ...
- poj1850(组合数)
题目链接:http://poj.org/problem;jsessionid=B0D9A01EC0F1043088A37454B6CED469?id=1850 题意:给字符串编号,该字符串必须满足由小 ...
随机推荐
- Android(八) HandlerThread
1.Looper Looper used to run a message loop for a thread. Threads by default do not have a message lo ...
- 【Python】详解Python多线程Selenium跨浏览器测试
前言 在web测试中,不可避免的一个测试就是浏览器兼容性测试,在没有自动化测试前,我们总是苦逼的在一台或多台机器上安装N种浏览器,然后手工在不同的浏览器上验证主业务流程和关键功能模块功能,以检测不同浏 ...
- 前端 HTML文档结构介绍
<!DOCTYPE HTML> <html> <head>...</head> <body>...</body> </ht ...
- 居然上了模板使用排行榜第一 happy一下
这段时间在学习css和div,顺便把博客给整了一下,然后不小心就上了FFandIE模板使用排行榜第一,happy一下下.不知道这个算不算排名,还是随机刷新.感觉应该是按流量统计的,这段时间有几篇文章一 ...
- 003-Nginx 设置Header 获取真实IP
1.X-Forwarded-For的定义: X-Forwarded-For:简称XFF头,它代表客户端,也就是HTTP的请求端真实的IP,只有在通过了HTTP 代理或者负载均衡服务器时才会添加该项.它 ...
- BGD-py实现学习【1】[转载]
转自:https://github.com/icrtiou/Coursera-ML-AndrewNg 1.源码-对数据读取 import numpy as np import pandas as pd ...
- 记两个国外CTF的弱pwn
两道题都来自CSAW CTF 18.PWN学得不够多,如果哪里错了,欢迎留言交流. 第一个题 get_it checksec检查之后,发现栈保护没开,很可能是栈溢出.IDA打开F5看伪源码. int ...
- np.repeat 与 np.tile
1.Numpy的 tile() 函数,就是将原矩阵横向.纵向地复制.tile 是瓷砖的意思,顾名思义,这个函数就是把数组像瓷砖一样铺展开来. 举个例子,原矩阵: import numpy as np ...
- mysql主从延迟(摘自http://www.linuxidc.com/Linux/2012-02/53995.htm)
http://www.linuxidc.com/Linux/2012-02/53995.htm
- 公司里面用的iTextSharp(教程)---简介
一.需求: 公司这次要做一个生成PDF的功能,需要设计,刚刚进入公司,组长把任务分配给了我,为了完成这个任务,苦学了许久的iTextSharp.现在记录下实现过程中了了解的一些东东,一起分享哈~~ 二 ...