[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 题意:给字符串编号,该字符串必须满足由小 ...
随机推荐
- zkSNARK 零知识验证
参考文献 ZCash7篇,有社区翻译版,但还是推荐看原汁原味的 https://z.cash/blog/snark-explain.html Vitalik3篇,小天才作者我就不介绍了,这三篇 ...
- idea 使用方法
1:设置自定义自动补全,使用$END$代表最后光标所在的位置. http://blog.csdn.net/u012569796/article/details/54694418 2:使用 shift+ ...
- 第22章 CLR寄宿和AppDomain
22.1 CLR寄宿 CLR Hosting(CLR 宿主)的概念:初始启动.Net Application时,Windows进程的执行和初始化跟传统的Win32程序是一样的,执行的还是非托管代码,只 ...
- Chrome Input框老是有输入记录的终极解决方案
尤其是日期框,输入记录都挡住日期弹框了. 浏览器地址栏输入: chrome://settings/autofill,按钮关掉就可以了.
- Django的基本开发环境配置和MTV模型
一.MTV模型 Django的MTV分别代表: Model(模型):负责业务对象与数据库的对象(ORM) Template(模版):负责如何把页面展示给用户 View(视图):负责业务逻辑,并在适当的 ...
- css 播放器按钮实现
效果图 html代码 //播放按钮 <div id="playBtn" class="circle" style="margin: 20px 0 ...
- SQL查询遍历数据方法一 [ 临时表 + While循环]
以下以SQL Server 2000中的NorthWind数据库中的Customers表为例, 用 临时表 + While循环 的方法, 对Customers表中的CompanyName列进行遍历 c ...
- XDU1024简单逆序对(贪心||分治)
题目描述 逆序对问题对于大家来说已经是非常熟悉的问题了,就是求i<j时,a[i] > a[j]的组数.现在请你求出一串数字中的逆序对的个数,需要注意的是,这些数字均在[0,9]之内. 输入 ...
- animation-fill-mode
animation-fill-mode: none:默认值.不设置对象动画之外的状态 forwards:结束后保持动画结束时的状态,但当animation-direction为0,则动画不执行,持续保 ...
- ACM数论之旅10---大组合数-卢卡斯定理(在下卢卡斯,你是我的Master吗?(。-`ω´-) )
记得前几章的组合数吧 我们学了O(n^2)的做法,加上逆元,我们又会了O(n)的做法 现在来了新问题,如果n和m很大呢, 比如求C(n, m) % p , n<=1e18,m<=1e18 ...