P5644-[PKUWC2018]猎人杀【NTT,分治】

正题

题目链接:https://www.luogu.com.cn/problem/P5644


题目大意

nnn个人,每个人被选中的权重是aia_iai。每次按照权重选择一个没有死掉的人杀死,求第111个人最后死的概率。输出答案对998244353998244353998244353取模。

wi>0,∑i=1nwi≤105w_i>0,\sum_{i=1}^nw_i\leq 10^5wi>0,i=1nwi105


解题思路

这个死掉之后概率的分母会变所以挺麻烦的,考虑一下变成每次随便选择一个人,如果没有死就杀掉,这样每个人被选择的概率就不变了。

然后考虑到计算恰好最后一个死很麻烦,可以假设第111个人死之后至少还剩下集合TTT的人,然后容斥这样就不需要考虑到剩下的人必须在前面都选过一次了。

P(T)P(T)P(T)表示111死之后剩下集合TTT的人的概率,怎么求这个东西,我们可以枚举一下杀到111之前的轮数(记SSS为全集,W(S)=∑x∈SwxW(S)=\sum_{x\in S}w_xW(S)=xSwx
P(T)=∑i=0∞(W(S)−W(T)−w1W(S))iw1W(S)P(T)=\sum_{i=0}^{\infty}(\frac{W(S)-W(T)-w_1}{W(S)})^i\frac{w_1}{W(S)}P(T)=i=0(W(S)W(S)W(T)w1)iW(S)w1
等比数列求和展开一下就是
P(T)=(W(S)−W(T)−w1W(S))∞−1W(S)−W(T)−w1W(S)−1w1W(S)P(T)=\frac{(\frac{W(S)-W(T)-w_1}{W(S)})^\infty-1}{\frac{W(S)-W(T)-w_1}{W(S)}-1}\frac{w_1}{W(S)}P(T)=W(S)W(S)W(T)w11(W(S)W(S)W(T)w1)1W(S)w1
然后因为那个∞\infty的东西是收敛(也就是等于000)的所以
P(T)=W(S)W(T)+w1w1W(S)=w1w1+W(T)P(T)=\frac{W(S)}{W(T)+w_1}\frac{w_1}{W(S)}=\frac{w_1}{w_1+W(T)}P(T)=W(T)+w1W(S)W(S)w1=w1+W(T)w1
就好了

然后答案就是
∑T∈S(−1)∣T∣P(T)=∑T∈S(−1)∣T∣w1w1+W(T)\sum_{T\in S}(-1)^{|T|}P(T)=\sum_{T\in S}(-1)^{|T|}\frac{w_1}{w_1+W(T)}TS(1)TP(T)=TS(1)Tw1+W(T)w1

但是这样的复杂度是O(2n)O(2^n)O(2n)的显然不可能过。

但是我们不难发现的是因为W(S)≤105W(S)\leq 10^5W(S)105,所以我们可以设f(i)f(i)f(i)表示对于所有集合TTT使得W(T)=iW(T)=iW(T)=i的容斥系数和那么答案就变成了
∑i=0W(S)f(i)w1w1+i\sum_{i=0}^{W(S)}f(i)\frac{w_1}{w_1+i}i=0W(S)f(i)w1+iw1

但是这个fff怎么求,其实看上去就很生成函数,f(i)f(i)f(i)相等于∏i=2n(1−xwi)\prod_{i=2}^n(1-x^{w_i})i=2n(1xwi)的第iii次项系数。

这个东西我们分治+NTTNTTNTT求就好了(因为这个做法实际上和分治NTTNTTNTT有区别)

时间复杂度O(mlog⁡2m)O(m\log^2 m)O(mlog2m)m=W(S)m=W(S)m=W(S)


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
const ll N=4e5+10,P=998244353;
ll n,w[N],r[N],x[N],y[N];
struct Poly{ll n,a[N];
}F[20];bool v[20];
ll power(ll x,ll b){ll ans=1;while(b){if(b&1)ans=ans*x%P;x=x*x%P;b>>=1;}return ans;
}
void NTT(ll *f,ll n,ll op){for(ll i=0;i<n;i++)if(i<r[i])swap(f[i],f[r[i]]);for(ll p=2;p<=n;p<<=1){ll len=p>>1,tmp=power(3,(P-1)/p);if(op==-1)tmp=power(tmp,P-2);for(ll k=0;k<n;k+=p){ll buf=1;for(ll i=k;i<k+len;i++){ll tt=f[i+len]*buf%P;f[i+len]=(f[i]-tt+P)%P;f[i]=(f[i]+tt)%P;buf=buf*tmp%P;}}}if(op==-1){ll inv=power(n,P-2);for(ll i=0;i<n;i++)f[i]=f[i]*inv%P;}return;
}
void Mul(Poly &a,Poly &b){for(ll i=0;i<a.n;i++)x[i]=a.a[i];for(ll i=0;i<b.n;i++)y[i]=b.a[i];ll l=1;while(l<a.n+b.n)l<<=1;for(ll i=0;i<l;i++)r[i]=(r[i>>1]>>1)|((i&1)?(l>>1):0);for(ll i=a.n;i<l;i++)x[i]=0;for(ll i=b.n;i<l;i++)y[i]=0;NTT(x,l,1);NTT(y,l,1);for(ll i=0;i<l;i++)x[i]=x[i]*y[i]%P;NTT(x,l,-1);for(ll i=0;i<l;i++)a.a[i]=x[i];a.n=a.n+b.n-1;return;
}
ll findq(){for(ll i=0;i<20;i++)if(!v[i]){v[i]=1;return i;}
}
ll Solve(ll l,ll r){if(l==r){ll p=findq();for(ll i=0;i<=w[l];i++)F[p].a[i]=0;F[p].a[0]=1;F[p].a[w[l]]=P-1;F[p].n=w[l]+1;return p;}ll mid=(l+r)>>1;ll ls=Solve(l,mid),rs=Solve(mid+1,r);Mul(F[ls],F[rs]);v[rs]=0;return ls;
}
signed main()
{scanf("%lld",&n);for(ll i=1;i<=n;i++)scanf("%lld",&w[i]);ll p=Solve(2,n),ans=0;for(ll i=0;i<F[p].n;i++)(ans+=F[p].a[i]*w[1]%P*power(w[1]+i,P-2)%P)%=P;printf("%lld\n",(ans+P)%P);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/318847.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

对传统应用进行容器化改造

本文由 陈计节 翻译自 FP Complete 网站上的文章 CONTAINERIZING A LEGACY APPLICATION: AN OVERVIEW&#xff0c;原作者 Emanuel Borsboom。以下为译文全文&#xff0c;如需阅读英文原文&#xff0c;请转到文末获取链接&#xff1a;本文接下来简要介绍什么是容器化&#xff0c…

2021牛客暑期多校训练营2 L-WeChat Walk(分块)

L-WeChat Walk 每个大点记录一下邻接点的最大步数 每次修改的时候&#xff0c;枚举修改点的邻接的大点来更新 修改大点的时候直接判是不是比邻接点都大 代码抄的std好不容易才看懂~ Code1 #include<bits/stdc.h> using namespace std; template <class Tint> T…

牛客题霸 [求平方根] C++题解/答案

牛客题霸 [求平方根] C题解/答案 题目描述 实现函数 int sqrt(int x). 计算并返回x的平方根 题解&#xff1a; 要求返回平方根&#xff0c;我们就找一个i&#xff0c;使得ii<x&&(i1)(i1)>x 这样的i就是我们要找的答案 注意&#xff0c;x有可能为负数&#xf…

二分算法:平均值(洛谷 UVA1451)

解析 这道题寻找平均值的max&#xff0c;答案明显具有单调性&#xff0c;所以采用二分算法 从0到1不断取中点mid作为平均值的可能点&#xff0c;看是否存在不短于l的数列均值&#xff1e;mid不难得到以下代码&#xff1a; double st0,ed1;for(int i1;i<10;i){double mid(s…

P5287-[HNOI2019]JOJO【KMP】

正题 题目链接:https://www.luogu.com.cn/problem/P5287 题目大意 开始一个空串&#xff0c;nnn个操作 在末尾加入xxx个ccc字符&#xff08;保证和ccc和前面的字符不同&#xff09;返回到第xxx次操作之后 每次操作完成后求所有前缀的最长的borderborderborder长度和 1≤n≤…

牛客题霸 [数组中只出现一次的数字] C++题解/答案

牛客题霸 [数组中只出现一次的数字] C题解/答案 题目描述 一个整型数组里除了两个数字之外&#xff0c;其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。 题解&#xff1a; 用map来记录每个数字出现几次&#xff0c;然后再循环一遍看哪个数字出现一次&#x…

Asp.NetCore依赖注入和管道方式的异常处理及日志记录

前言在业务系统&#xff0c;异常处理是所有开发人员必须面对的问题&#xff0c;在一定程度上&#xff0c;异常处理的能力反映出开发者对业务的驾驭水平&#xff1b;本章将着重介绍如何在 WebApi 程序中对异常进行捕获&#xff0c;然后利用 Nlog 组件进行记录&#xff1b;同时&a…

2021“MINIEYE杯”中国大学生算法设计超级联赛(2)I love counting(Trie树)

I love counting O{Mlog⁡aMAX(BN/B)}O\{M\log{a_{\text{MAX}}}(\text{BN/B})\}O{MlogaMAX​(BN/B)} md考场写的莫队Trie一直T #include<bits/stdc.h> using namespace std; using lllong long; using piipair<int,int>; using plipair<ll,int>; constexpr …

字符串:凯撒密码(洛谷P1914)

解析 只需将每一位ascll码加n即可 但要注意的是c的ascll码是有上限的 我一开始是这么写的&#xff1a; for(int i1;i<l;i){s[i] n;while(s[i]>z) s[i] - 26;printf("%c",s[i]);}结果&#xff1a; (真的是随便打的&#xff0c;不巧有些攻击性。。&#xff0…

P6113-[模板]一般图最大匹配【带花树】

正题 题目链接:https://www.luogu.com.cn/problem/P6113 题目大意 给出一张无向图&#xff0c;求最大匹配。 1≤n≤103,1≤m≤51041\leq n\leq 10^3,1\leq m\leq 5\times 10^41≤n≤103,1≤m≤5104 解题思路 带花树的模板&#xff0c;我也不会讲/kel 所以看下面两篇大佬的博…

牛客题霸 [跳台阶] C++题解/答案

牛客题霸 [跳台阶] C题解/答案 题目描述 一只青蛙一次可以跳上1级台阶&#xff0c;也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法&#xff08;先后次序不同算不同的结果&#xff09;。 题解&#xff1a; 递归的入门题 如果只剩一个台阶&#xff0c;只有一种跳…

.NET Core实战项目之CMS 第四章 入门篇-Git的快速入门及实战演练

写在前面上篇文章.NET Core实战项目之CMS 第三章 入门篇-源码解析配置文件及依赖注入我带着大家通过分析了一遍ASP.NET Core的源码了解了它的启动过程&#xff0c;然后又带着大家熟悉了一遍配置文件的加载方式&#xff0c;最后引出了依赖注入以及控制反转的概念&#xff01;如果…

高精度:麦森数*(洛谷P1045)

P1045 [NOIP2003 普及组] 麦森数 解析 看似只是正常的一个高精 然而 暗藏杀机 一开始随手那么一写 。。。 (即使用了快速幂)时间复杂度过于感人 后来我们发现&#xff1a; 第一问位数的计算不必真的算出来&#xff0c;只需把2的p次幂转化为10的k次幂即可&#xff08;具体请…

2021“MINIEYE杯”中国大学生算法设计超级联赛(2)I love exam(背包)

I love exam 不知道为啥刚开始不写&#xff0c;那么简单的背包预处理dp&#xff0c;太菜了吧 fi,jf_{i,j}fi,j​对于第i门课来说花费j天得到的最大分数 gi,j,pg_{i,j,p}gi,j,p​考虑前i门课&#xff0c;花费j天复习得到的最大分数 #include<bits/stdc.h> using namespa…

P2012-拯救世界2【EGF】

正题 题目链接:https://www.luogu.com.cn/problem/P2012 题目大意 121212种东西排列成长度为nnn的序列&#xff0c;要求前四种出现奇数次&#xff0c;后四种出现偶数次&#xff0c;求方案。TTT组数据&#xff0c;对10910^9109取模。 1≤n<263,1≤T≤21051\leq n< 2^{63}…

彼之蜜糖,吾之砒霜——聊聊软件开发中的最佳实践

“描述一个事物&#xff0c;唯有一个名词定义它的概念&#xff0c;唯有一个动词揭露它的行为&#xff0c;唯有一个形容词表现它的特征。要做的&#xff0c;就是用心去寻找那个名词、那个动词、那个形容词……”—— 福楼拜 (Gustave Flaubert)我想讲个故事。很久很久以前&#…

贪心: Array Splitting(数列分段)(洛谷CF1175D)

解析 这题可以转化一下&#xff1a; &#xff08;《神笔马良》。。。。&#xff09; 计算这些长方形对应下标的总加和 我们可以一层一层往上垒,假设第i层起始点为xi&#xff0c;总和为sumi&#xff0c;再设从1到i的前缀和为si 显然第一层x11&#xff0c;sum1sn 对于第二层x2&…

牛客题霸 [合并有序链表] C++题解/答案

牛客题霸 [合并有序链表] C题解/答案 题目描述 将两个有序的链表合并为一个新链表&#xff0c;要求新的链表是通过拼接两个链表的节点来生成的。 题解&#xff1a; 首先判断l1和l2是否为空 然后依次比较l1和l2的值&#xff0c;然后存到新的链表里&#xff0c;当有一方全部结…

2021牛客暑期多校训练营2 G.League of Legends(转化+单调队列)

G.League of Legends Zechariah_2001题解 对于可以包含其他区间的大区间&#xff0c;要使得答案最优无非就是两种分组方式&#xff1a;单独一组或者与被包含的区间一组。单独一组那么贡献就是区间长度&#xff1b;如果说与被包含的区间一组&#xff0c;对答案贡献为0&#xff…

P5056-[模板]插头dp

正题 题目链接:https://www.luogu.com.cn/problem/P5056 题目大意 n∗mn*mn∗m的网格&#xff0c;求有多少条回路可以铺满整个棋盘。 解题思路 插头dpdpdp的&#xff0c;写法是按照题解上的写法。 状态用的是括号匹配&#xff0c;然后用了哈希邻接表&#xff08;挂表&#x…