Atcoder

一个900分的题耗了我这么久……而且官方题解还那么短……必须纪念一下……

思路

发现每种元素必须出现两次以上的限制极为恶心,所以容斥,枚举出现0/1次的元素个数分别有几个。设出现1次的元素有\(i\)个,无限制的的有\(k\)个,\(i\)个被分到了\(l\)个集合里。

此时只有集合不能相同的限制。

发现那\(l\)个集合无论如何都不会和其他集合相同,所以方案数是\({(2^{k})}^l\)。

剩下的相当于从\(2^k\)中选几种集合,是\(2^{2^k}\)。

所以答案就是

\[\sum_{i=0}^n \sum_{k=0}^n \sum_{l=0}^n {n\choose k}{n-k\choose i}(-1)^{n-k} S(i,l) 2^{2^k} 2^{kl}
\]

其中\(S(i,l)\)为第二类斯特林数。

此时获得了一个\(O(n^3)\)做法,尝试优化掉一个求和符。

考虑优化\(i\)。令\(m=l,p=n-k\),我们要求的就是

\[F(m,p)=\sum_{i=0}^n S(i,m){p\choose i}
\]

注意到\(S(i,m)=S(i-1,m-1)+m\times S(i-1,m)\),而\({p\choose i}=\sum_{j=0}^{p-1}{j\choose i-1}\),所以有

\[\begin{align*}
F(m,p)&=\sum_{i=0}^n [S(i-1,m-1)+m\times S(i-1,m)]\sum_{j=0}^{p-1}{j\choose i-1}\\
&=\sum_{j=0}^{p-1}\sum_{i=0}^{n-1} [S(i,m-1)+m\times S(i,m)]{j\choose i}\\
&=\sum_{j=0}^{p-1} [m\times F(m,j)+F(m-1,j)]
\end{align*}
\]

这里没有边角项是因为\(n>j\),所以\({j\choose n}=0\)。

于是维护前缀和就可以算出\(F(m,p)\)了。

最终答案

\[\sum_{k=0}^n \sum_{l=0}^n {n\choose k}(-1)^{n-k} 2^{2^k+kl} F(l,n-k)
\]

那个2的幂次还需要光速幂。

已经不想看官方题解是什么了……不如抓个ntf来帮我做题。

update:发现我使劲优化的那个东西就是\(S(p+1,m+1)\)……被教做人了……

代码

#include<bits/stdc++.h>
clock_t t=clock();
namespace my_std{
using namespace std;
#define pii pair<int,int>
#define fir first
#define sec second
#define MP make_pair
#define rep(i,x,y) for (int i=(x);i<=(y);i++)
#define drep(i,x,y) for (int i=(x);i>=(y);i--)
#define go(x) for (int i=head[x];i;i=edge[i].nxt)
#define templ template<typename T>
#define sz 3555
typedef long long ll;
typedef double db;
mt19937 rng(chrono::steady_clock::now().time_since_epoch().count());
templ inline T rnd(T l,T r) {return uniform_int_distribution<T>(l,r)(rng);}
templ inline bool chkmax(T &x,T y){return x<y?x=y,1:0;}
templ inline bool chkmin(T &x,T y){return x>y?x=y,1:0;}
templ inline void read(T& t)
{
t=0;char f=0,ch=getchar();double d=0.1;
while(ch>'9'||ch<'0') f|=(ch=='-'),ch=getchar();
while(ch<='9'&&ch>='0') t=t*10+ch-48,ch=getchar();
if(ch=='.'){ch=getchar();while(ch<='9'&&ch>='0') t+=d*(ch^48),d*=0.1,ch=getchar();}
t=(f?-t:t);
}
template<typename T,typename... Args>inline void read(T& t,Args&... args){read(t); read(args...);}
char __sr[1<<21],__z[20];int __C=-1,__zz=0;
inline void Ot(){fwrite(__sr,1,__C+1,stdout),__C=-1;}
inline void print(register int x)
{
if(__C>1<<20)Ot();if(x<0)__sr[++__C]='-',x=-x;
while(__z[++__zz]=x%10+48,x/=10);
while(__sr[++__C]=__z[__zz],--__zz);__sr[++__C]='\n';
}
void file()
{
#ifdef NTFOrz
freopen("a.in","r",stdin);
#endif
}
inline void chktime()
{
#ifndef NTFOrz
cout<<(clock()-t)/1000.0<<'\n';
#endif
}
ll mod;
ll ksm(ll x,int y){ll ret=1;for (;y;y>>=1,x=x*x%mod) if (y&1) ret=ret*x%mod;return ret;}
ll ksm(ll x,int y,ll mod){ll ret=1;for (;y;y>>=1,x=x*x%mod) if (y&1) ret=ret*x%mod;return ret;}
ll inv(ll x){return ksm(x,mod-2);}
// inline ll mul(ll a,ll b){ll d=(ll)(a*(double)b/mod+0.5);ll ret=a*b-d*mod;if (ret<0) ret+=mod;return ret;}
}
using namespace my_std; int n; ll fac[sz],_fac[sz];
void init(){fac[0]=_fac[0]=1;rep(i,1,sz-1) _fac[i]=inv(fac[i]=fac[i-1]*i%mod);}
ll C(int n,int m){return n>=m&&m>=0?fac[n]*_fac[m]%mod*_fac[n-m]%mod:0;} ll S[sz][sz],Ssum[sz][sz];
ll F[sz][sz],Fsum[sz][sz]; ll pw2[sz];
namespace SHIT
{
#define S 32768
ll pw1[S],pw2[S];
void init()
{
pw1[0]=1;rep(i,1,S-1) pw1[i]=pw1[i-1]*2%mod;
pw2[0]=1;pw2[1]=pw1[S-1]*2%mod;rep(i,2,S-1) pw2[i]=pw2[i-1]*pw2[1]%mod;
}
ll ksm(int x){return pw2[x>>15]*pw1[x&32767]%mod;}
#undef S
} int main()
{
file();
read(n,mod);
init();SHIT::init();
S[0][0]=1;rep(i,1,n) rep(j,1,i) (S[i][j]+=S[i-1][j-1]+S[i-1][j]*j%mod)%=mod;
rep(i,0,n) rep(j,1,n) (Ssum[i][j]=Ssum[i][j-1]+S[i][j])%=mod;
rep(p,0,n) rep(i,0,n) (F[0][p]+=S[i][0]*C(p,i)%mod)%=mod;
Fsum[0][0]=F[0][0];rep(p,1,n) Fsum[0][p]=(Fsum[0][p-1]+F[0][p])%mod;
rep(m,1,n) rep(i,0,n) (F[m][0]+=S[i][m]*C(0,i)%mod)%=mod,Fsum[m][0]=F[m][0];
rep(m,1,n) rep(p,1,n) (F[m][p]=Fsum[m][p-1]*m%mod+Fsum[m-1][p-1])%=mod,Fsum[m][p]=(Fsum[m][p-1]+F[m][p])%mod;
pw2[0]=1;rep(i,1,n) pw2[i]=pw2[i-1]*2%(mod-1);
ll ans=0;
rep(k,0,n)
rep(l,0,n)
(ans+=F[l][n-k]*(((n+k)&1)?mod-1:1ll)%mod*SHIT::ksm((k*l+pw2[k])%(mod-1))%mod*C(n,k)%mod)%=mod;
cout<<ans;
return 0;
}

ARC096 E Everything on It [容斥,斯特林数]的更多相关文章

  1. bzoj 4671 异或图——容斥+斯特林反演+线性基

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4671 考虑计算不是连通图的方案,乘上容斥系数来进行容斥. 可以枚举子集划分(复杂度是O(Be ...

  2. bzoj 4671 异或图 —— 容斥+斯特林反演+线性基

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4671 首先,考虑容斥,就是设 \( t[i] \) 表示至少有 \( i \) 个连通块的方 ...

  3. 【bzoj4671】异或图(容斥+斯特林反演+线性基)

    传送门 题意: 给出\(s,s\leq 60\)张图,每张图都有\(n,n\leq 10\)个点. 现在问有多少个图的子集,满足这些图的边"异或"起来后,这张图为连通图. 思路: ...

  4. 【CF715E】Complete the Permutations(容斥,第一类斯特林数)

    [CF715E]Complete the Permutations(容斥,第一类斯特林数) 题面 CF 洛谷 给定两个排列\(p,q\),但是其中有些位置未知,用\(0\)表示. 现在让你补全两个排列 ...

  5. AT4119-[ARC096C]Everything on It【斯特林数,容斥】

    正题 题目链接:https://www.luogu.com.cn/problem/AT4119 题目大意 一个集合\(S=\{k\in[1,n]\cup N\}\),它的所有子集作为元素组成的集合中要 ...

  6. BZOJ4671 异或图(容斥+线性基)

    题意 定义两个结点数相同的图 \(G_1\) 与图 \(G_2\) 的异或为一个新的图 \(G\) ,其中如果 \((u, v)\) 在 \(G_1\) 与 \(G_2\) 中的出现次数之和为 \(1 ...

  7. POJ1091跳蚤(容斥 + 唯一分解 + 快速幂)

      题意:规定每次跳的单位 a1, a2, a3 …… , an, M,次数可以为b1, b2, b3 …… bn, bn + 1, 正好表示往左,负号表示往右, 求能否调到左边一位,即 a1* b1 ...

  8. HDU 4059 容斥初步练习

    #include <iostream> #include <cstring> #include <cstdio> #include <algorithm> ...

  9. HDU 4336 Card Collector (期望DP+状态压缩 或者 状态压缩+容斥)

    题意:有N(1<=N<=20)张卡片,每包中含有这些卡片的概率,每包至多一张卡片,可能没有卡片.求需要买多少包才能拿到所以的N张卡片,求次数的期望. 析:期望DP,是很容易看出来的,然后由 ...

随机推荐

  1. 经实验验证,修正对using namespace std的认识

    备注①:name:符号.指的实体包括:变量.函数.类 备注②:认为全局命名空间也是一个包,在此称作 ROOT:: 或 global:: (这样就有了两个特别的包:一个是全局包,一个是std包.但对于编 ...

  2. Texture(ASDK)、ComponentKit、LayoutKit、YogaKit

    YogaKit 最轻量,改动量最小,目的最纯粹,同时也最类似于使用 frame ,需要自己造一波在 UITableView 中使用的轮子(各类 frame 结果缓存方案).同类的备选方案是 FlexB ...

  3. Java 之 线程池

    一.线程池思想概述 如果使用线程的时候就去创建一个线程,这样实现起来非常简便,但是会出现一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了,这样频繁创建线程就会大大降低 ...

  4. java InputStream的使用

    package cn.kongxh.io3;import java.io.File ;import java.io.InputStream ;import java.io.FileInputStrea ...

  5. Dijkstra算法求最短路径 Java实现

    基本原理: 迪杰斯特拉算法是一种贪心算法. 首先建立一个集合,初始化只有一个顶点.每次将当前集合的所有顶点(初始只有一个顶点)看成一个整体,找到集合外与集合距离最近的顶点,将其加入集合并检查是否修改路 ...

  6. Laravel实现用户的注册、登录

    一.安装 Laravel(使用 Laravel5.5) 通过 Composer 创建项目 composer create-project --prefer-dist laravel/laravel s ...

  7. 环境搭建:添加 xgboost 到 Anaconda

    原文参考:https://blog.csdn.net/lvsehaiyang1993/article/details/80619495 原文博主:Big_quant

  8. 详解CentOS6.7部署Tomcat及主配置文件

    Java程序实现部署及应用 POSIX :可移植操作系统,编程操作系统接口规范,实现跨平台编译运行. API:应用程序编程接口 ABI:应用程序二进制接口 描述了应用程序和操作系统之间,一个应用和它的 ...

  9. C语言几个术语: 数据对象,左值,右值

    1. 数据对象 赋值表达式语句的目的是把值存储到内存位置上. 用于存储值的数据存储区域统称为数据对象. 2. 左值 左值是C语言的术语, 用于标识特定数据对象的名称或表达式. 对象指的是实际的数据存储 ...

  10. 【Java】Java环境变量配置

    一.windows系统 右键你的电脑(计算机/此电脑)打开属性->高级系统设置->环境变量,在系统变量里配置三个环境变量. 假设jdk的安装路径为C:\Program Files\Java ...