题面传送门

题意:

给出两个长度为 \(n\) 的数组 \(a,b\) 和一个整数 \(s\)。

你可以任意重排数组 \(a,b\),总共 \((n!)^2\) 种方案。

现在又两个人 A,B 来玩游戏,两人轮流操作,A 先操作。

每次操作当前选手会取出各自数组的第一个元素 \(x\) 并将它删去,并令 \(s\) 减去 \(x\),如果发现 \(s\leq 0\),那么当前操作的人输,游戏结束。

求在所有 \((n!)^2\) 的方案中,有多少种能够使得 A 赢,答案模 \(10^9+7\)。

\(1 \leq n,a_i,b_i \leq 100\),\(1 \leq s \leq \sum(a_i+b_i)\),时限 \(=4s\)。

我竟然现场 40min 就 A 了这道题,incredible!

拿总方案数 \(-\) B 赢的方案数。

考虑 B 赢是什么情况。肯定我们删到了某个位置 \(i\),然后删除完第 \(i\) 个位置上的数之后 \(s \leq 0\),然后 A 就输了。

假设 A 在第 \(i\) 次删除的数为 \(x\)。那么在删除 \(x\) 之前,删除的所有数之和 \(<s\),而删完 \(x\) 之后删除的所有数之和 \(\geq s\)。

我们枚举这个 \(i\) 和 \(x\),假设删 \(a_i\) 之前删除的所有数之和为 \(sum\),那么我们可以得到不等式 \(s-x \leq sum \lt s\)。

那么题目转化为:你在 \(a_j\) 中选择 \(i-1\) 个数填在 \(a_1,a_2,\dots,a_{i-1}\) 的位置(已经放好的 \(x\) 就不能再被选择了),再从 \(b_j\) 中选择 \(i-1\) 个数放在 \(b_1,b_2,\dots,b_{i-1}\) 的位置,满足 \(a,b\) 数组中选出来的 \(2(i-1)\) 个数之和在 \([s-x,s)\) 之间,求方案数。

首先 \(n^3\) 的暴力 \(dp\) 是很显然的。你每枚举一个 \(x\),就相当于从 \(a\) 数组中踢掉了一个数,就重新做一遍背包,\(dpa_{i,j,k}\) 表示在 \(a_1,a_2,\dots,a_i\) 中选择 \(j\) 个数和为 \(k\) 的方案数。\(dpb\) 同理。然后枚举 \(sum\) 和选出来的 \(i-1\) 个 \(a_i\) 的和,用乘法原理计算贡献。\(a_1,a_2,\dots,a_{i-1}\) 可以随便排,方案数 \((i-1)!\),\(b_1,b_2,\dots,b_{i-1}\) 也可以随便排,方案数也是 \((i-1)!\)。\(a_{i+1},a_{i+2},\dots,a_{n}\),\(b_i,b_{i+1},\dots,b_n\) 也依次有一个全排列的系数,所以总的乘法系数是 \(((i-1)!)^2\times(n-i)!\times(n-i+1)!\)。总复杂度 \(n^5\)

不过借鉴 [CTSC2012] 假面 的经验,可以先预处理出整个序列的 \(dpa\),然后对每个 \(x\) 都跑一遍可删除背包。正推的方程式是 \(dp_{i,j,k}=dp_{i-1,j,k}+dp_{i-1,j-1,k-x}\)。现在是在已知 \(dp_{i}\) 的情况下求出 \(dp_{i-1}\)。注意到 \(dp_{i-1,0,k}=dp_{i,0,k}\),所以我们可以先还原出 \(dp_{i-1,0,k}\),再根据 \(dp_{i-1,1,k}=dp_{i,1,k}-dp_{i-1,0,k-x}\) 还原出 \(dp_{i-1,1,k}\),以此类推,还原的复杂度是 \(n^2\) 的。

接下来是计算答案,还是枚举 \(a\) 贡献的和 \(suma\),不过注意到 \(sum\) 的取值范围是一段区间,所有 \(b\) 贡献的和的范围也是个区间 \([s-x-suma,s-suma)\),前缀和转移,又可以优化掉一个 \(n\)。

时间复杂度 \(\mathcal O(n^4)\)

#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define fz(i,a,b) for(int i=a;i<=b;i++)
#define fd(i,a,b) for(int i=a;i>=b;i--)
#define ffe(it,v) for(__typeof(v.begin()) it=v.begin();it!=v.end();it++)
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a))
#define fillbig(a) memset(a,63,sizeof(a))
#define pb push_back
#define ppb pop_back
#define mp make_pair
typedef long long ll;
typedef pair<int,int> pii;
const int MOD=1e9+7;
const int MAXN=100+5;
const int MAXS=2e4+5;
int n,s,a[MAXN],b[MAXN];
int dpa[MAXN][MAXS],dpb[MAXN][MAXS],sdp[MAXN][MAXS];
int ndp[MAXN][MAXS];
int fac[MAXN];
int solve(){
int sa=0,sb=0,ss=0;
fac[0]=1;for(int i=1;i<=n;i++) fac[i]=1ll*fac[i-1]*i%MOD;
for(int i=1;i<=n;i++) sa+=a[i];
for(int i=1;i<=n;i++) sb+=b[i];
ss=sa+sb;dpa[0][0]=1;
for(int i=1;i<=n;i++) for(int j=i;~j;j--) for(int k=sa;~k;k--)
if(j>=1&&k>=a[i]) dpa[j][k]=(dpa[j][k]+dpa[j-1][k-a[i]])%MOD;
dpb[0][0]=1;
for(int i=1;i<=n;i++) for(int j=i;~j;j--) for(int k=sb;~k;k--)
if(j>=1&&k>=b[i]) dpb[j][k]=(dpb[j][k]+dpb[j-1][k-b[i]])%MOD;
for(int i=0;i<=n;i++){
sdp[i][0]=dpb[i][0];
for(int j=1;j<=ss;j++) sdp[i][j]=(sdp[i][j-1]+dpb[i][j])%MOD;
}
int ans=0;
for(int i=1;i<=n;i++){
for(int j=0;j<=n;j++) for(int k=0;k<=sa;k++) ndp[j][k]=dpa[j][k];
for(int j=0;j<n;j++) for(int k=0;k<=sa-a[i];k++){
ndp[j+1][k+a[i]]=(ndp[j+1][k+a[i]]-ndp[j][k]+MOD)%MOD;
}
int l=max(s-a[i],0),r=s-1;
for(int j=0;j<n;j++){
int sum=0;
for(int k=0;k<=sa;k++){
if(k>r) break;
int nl=l-k,nr=r-k;
sum=(sum+1ll*ndp[j][k]*sdp[j][nr]%MOD)%MOD;
if(nl>0) sum=(sum-1ll*ndp[j][k]*sdp[j][nl-1]%MOD+MOD)%MOD;
}
ans=(ans+1ll*sum*fac[j]%MOD*fac[j]%MOD*fac[n-j-1]%MOD*fac[n-j]%MOD)%MOD;
}
}
int tot=1ll*fac[n]*fac[n]%MOD;
return (tot-ans+MOD)%MOD;
}
class PrettyLiar{
public:
int count(int S,vector<int> kaede,vector<int> kanade){
s=S;n=kaede.size();
for(int i=0;i<n;i++) a[i+1]=kaede[i],b[i+1]=kanade[i];
return solve();
}
};
//PrettyLiar pro;

Topcoder 15405 - PrettyLiar(可删除背包+前缀和优化 dp)的更多相关文章

  1. LOJ 6089 小Y的背包计数问题 —— 前缀和优化DP

    题目:https://loj.ac/problem/6089 对于 i <= √n ,设 f[i][j] 表示前 i 种,体积为 j 的方案数,那么 f[i][j] = ∑(1 <= k ...

  2. P5241 序列(滚动数组+前缀和优化dp)

    P5241 序列 挺神仙的一题 看看除了dp好像没什么其他办法了 想着怎么构个具体的图出来,然鹅不太现实. 于是我们想办法用几个参数来表示dp数组 加了几条边肯定要的吧,于是加个参数$i$表示已加了$ ...

  3. CDOJ 1307 ABCDE 前缀和优化dp

    ABCDE 题目连接: http://acm.uestc.edu.cn/#/problem/show/1307 Description Binary-coded decimal (BCD) is a ...

  4. bzoj 1044 [HAOI2008]木棍分割——前缀和优化dp

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1044 前缀和优化. 但开成long long会T.(仔细一看不用开long long) #i ...

  5. bzoj 3398 [Usaco2009 Feb]Bullcow 牡牛和牝牛——前缀和优化dp / 排列组合

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3398 好简单呀.而且是自己想出来的. dp[ i ]表示最后一个牡牛在 i 的方案数. 当前 ...

  6. bzoj2431: [HAOI2009]逆序对数列(前缀和优化dp)

    2431: [HAOI2009]逆序对数列 Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 2312  Solved: 1330[Submit][Stat ...

  7. CF601C Kleofáš and the n-thlon(期望+前缀和优化dp)

    传送门 解题思路 要求这个人的排名,我们可以先求出某个人比他排名靠前的概率,然后再乘上\(m-1\)即为答案.求某个人比他排名靠前可以用\(dp\),设\(f[i][j]\)表示前\(i\)场比赛某人 ...

  8. 2018.09.25 51nod1597 有限背包计数问题(背包+前缀和优化)

    传送门 dp好题. 我认为原题的描述已经很清楚了: 你有一个大小为n的背包,你有n种物品,第i种物品的大小为i,且有i个,求装满这个背包的方案数有多少. 两种方案不同当且仅当存在至少一个数i满足第i种 ...

  9. Codeforces Round #274 (Div. 1) C. Riding in a Lift 前缀和优化dp

    C. Riding in a Lift Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/480/pr ...

随机推荐

  1. 初学python-day6 for循环和流程控制(已更新循环做三角形图形!!)

    for循环 1.格式 for    变量    in   集合: 循环体 2.概述 当程序执行for循环,按顺序从集合中获取元素变量保存当前循环得到的值,再去执行循环体.当集合中数据都被取完,则此刻跳 ...

  2. 【UE4】GAMES101 图形学作业0:矩阵初识

    作业描述 给定一个点P=(2,1), 将该点绕原点先逆时针旋转45◦,再平移(1,2), 计算出变换后点的坐标(要求用齐次坐标进行计算). UE4 知识点 主要矩阵 FMatrix FBasisVec ...

  3. 如何访问位于内网的Ubuntu主机

    如何访问位于内网的Ubuntu主机 内网主机为Ubuntu桌面版 内网主机Ubuntu字符串界面版 SSH远程主机管理工具推荐 SSH远程文件访问工具推荐 如何访问位于内网的Ubuntu主机 内网主机 ...

  4. the Agiles Scrum Meeting 2

    会议时间:2020.4.10 21:00 1.每个人的工作 今天已完成的工作 yjy:debug:班级创建了个人项目不能访问班级:教师窗口的前端bug. issues:Bug:教师创建博客时显示项目为 ...

  5. 大闸蟹的 O O 战记

    一. 第四单元架构设计分析 第一次作业,UML类图 第一次作业的主要任务是完成对UML类图的解析并实现查询等操作,需要在课程组给定的框架中添加函数.对于UML类图,其存储是按照元素来存储的,其将所有的 ...

  6. 2021.10.26考试总结[冲刺NOIP模拟16]

    T1 树上的数 \(DFS\)一遍.结构体存边好像更快? \(code:\) T1 #include<bits/stdc++.h> using namespace std; namespa ...

  7. QEvent

    QEvent类是所有事件类的基类,每一个对象都包含事件参数.Qt的主事件循环(QCoreApplication::exec())从事件队列中接收本地窗口系统的事件,并将它们翻译成QEvent,将这些事 ...

  8. Sharding-JDBC基本使用,整合Springboot实现分库分表,读写分离

    结合上一篇docker部署的mysql主从, 本篇主要讲解SpringBoot项目结合Sharding-JDBC如何实现分库分表.读写分离. 一.Sharding-JDBC介绍 1.这里引用官网上的介 ...

  9. IDEA免费激活至2099年教程,亲测可用

    申明,本教程 Intellij IDEA 最新版激活教程,激活码均收集与网络,请勿商用,仅供个人学习使用,如有侵权,请联系作者删除.如条件允许,建议大家购买正版. 以下是本人免费激活到 2099 年的 ...

  10. 【Go语言细节】反射

    什么是反射 维基百科上反射的定义: 在计算机科学中,反射是指计算机程序在运行时(Run time)可以访问.检测和修改它本身状态或行为的一种能力.用比喻来说,反射就是程序在运行的时候能够"观 ...