【51nod 1251】 Fox序列的数量(以及带限制插板法讲解)

为什么网上没有篇详细的题解【雾
可能各位聚聚觉得这道题太简单了吧 /kk

题意

首先题目是求满足条件的序列个数,条件为:出现次数最多的数仅有一个

分析

感谢 刚睡醒的 JZ姐姐在咱写题解忽然陷入自闭的时候为咱知道迷津 QwQ

那么套路来说第一想到的其实就是容斥辣 然后懒得想去网上黈力【雾
在然后,发现网上要么没推导公式(虽说是直接列不用化简的)要么就是推导公式是错的...

于是坐下来冷静分析:首先,我们考虑最高的出现次数为 x 的情况下,有多少个序列是满足条件的
容斥一发发现式子长这个样子(用到了隔板法,然后容斥顺便也能把有数字出现次数高于 x 的情况给容掉)

\[ans(x)=m\sum_{j=0}^{m-1} (-1)^j \binom{n-x*(j+1) + m-2 }{m-2} \]

大概是说,我们确定最高次数为 x ,那么我们先钦定一个数为恰好出现 x 次的数,然后剩下 m-1 个数插板容斥求方案 QwQ ,于是我们就可以递推求解辣:

\[ANS=\sum_{i=\lceil \frac{n}{m} \rceil }^n m\sum_{j=0}^{m-1} (-1)^j \binom{n-i*(j+1)+m-2} {m-2} \]

然后我们发现复杂度爆炸了呢 Σ(⊙▽⊙"a 管他呐,交上去就好辣

然后咱发现后面那个式子如果 \(n-i*(j+1)+m-2\) 大于 \(m-2\) 的话贡献为 0 ,一顿计算发现 j 要小于 n/i 才有贡献,那么第二个枚举的上界的 m-1 改一改就好咯...

\[ANS=\sum_{i=\lceil \frac{n}{m} \rceil }^n m\sum_{j=0}^{min(m-1,n/i-1)} (-1)^j \binom{n-i*(j+1)+m-2} {m-2} \]

然后调和级数分析一波,发现复杂度是 \(n·H(n)\) 的,能过...

顺便稍微讲讲带限制的板怎么插好了 QAQ
注意这里说的限制是对于任意两个隔板之间的元素数量而言,如果隔板间元素数量的限制是不同,可能需要什么黑科技(反正咱只会指数级别容斥【逃)

那么假设现在有 n 个元素,要分成 m 堆元素,也就是有 m-1 个插板

0.不带限制的插板

显然是 \(\binom{n+m-1}{m-1}\)

意思就是我们把 n 个元素和 m-1 个插板排成一列,计算方案数
那么我们只要计算 m-1 个插板有多少种摆法就好了,隔板确定完了,元素的放法自然就确定了
说白了就是求了个 m-1 个插板在所有 n+m-1 个物品中的位置方案数

1.带下界的插板

假设下界为 x ,即隔出来的每堆元素数量不小于 x
答案是 \(\binom{n+m-m·x-1}{m-1}\)

我们发现这玩意儿比较简单,只要把原本要拿来插板的元素先减去 \(m*x\) (也就是预先给每堆元素分掉这个下界 ) 然后再去插板就好辣

2.(JZ 姐姐教的QwQ)带上界的插板

假设上界为 x ,即隔出来的每堆元素数量不大于 x
答案是 \(\sum_{i=0}^{m} (-1)^i \binom{m}{j} \binom{n+m-i*(x+1)-1}{m-1}\)

有容斥呢 QwQ 解释一下,就是说我们枚举超出上限的元素堆个数,然后强制他们选了 (x+1) 个,这样不管它们接下来怎么选都已经超限了,然后我们加个 \((-1)\) 的幂次容斥一下就好辣
如果康不懂的话请重修容斥 QvQ

这样咱发现上面的那个式子一下子就很明了了呢 QvQ

Code

注意阶乘及逆元的 1e5 数组是假的,要两倍的!花了咱 5 点头盾 get 到的细节 【枯

//by Judge
#define HGS_AK_IOI true
#include<bits/stdc++.h>
#define mod 1000000007
#define Rg register
#define fp(i,a,b) for(Rg int i=(a),I=(b)+1;i<I;++i)
#define fd(i,a,b) for(Rg int i=(a),I=(b)-1;i>I;--i)
#define go(u) for(Rg int i=head[u],v=e[i].to;i;v=e[i=e[i].nxt].to)
#define open(S) freopen(S".in","r",stdin),freopen(S".out","w",stdout)
#define ll long long
using namespace std;
const int M=2e5+3;
typedef int arr[M];
#ifndef Judge
#define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
#endif
char buf[1<<21],*p1=buf,*p2=buf;
template<class T>inline T Min(T x,T y){return x<y?x:y;}
template<class T>inline T Max(T x,T y){return x>y?x:y;}
template<class T>inline bool cmax(T& a,T b){return a<b?a=b,1:0;}
template<class T>inline bool cmin(T& a,T b){return a>b?a=b,1:0;}
inline int read(){ int x=0,f=1; char c=getchar();
    for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
    for(;isdigit(c);c=getchar()) x=x*10+c-'0'; return x*f;
} int n,m,ans; arr fac,inv;
inline int mul(int x,int y){return 1ll*x*y%mod;}
inline void Pls(int& x,int y){if((x+=y)>=mod)x-=mod;}
inline int qpow(int x,int p=mod-2){ Rg int s=1;
    for(;p;p>>=1,x=mul(x,x)) if(p&1) s=mul(s,x); return s;
}
inline void prep(int n=2e5){
    fac[0]=1; fp(i,1,n) fac[i]=mul(fac[i-1],i);
    inv[n]=qpow(fac[n]); fd(i,n,1) inv[i-1]=mul(inv[i],i);
}
inline int Ceil(int x,int y){ return x%y>0?x/y+1:x/y; }
inline int C(int n,int m){ if(n<m) return 0;
    return mul(fac[n],mul(inv[m],inv[n-m]));
}
inline void Solv(){ n=read(),m=read(),ans=0;
    if(m==1) return printf("1\n"),void();
    if(n==1) return printf("%d\n",m),void();
    fp(i,Ceil(n,m),n) fp(j,0,Min(n/i-1,m-1))
        Pls(ans,mul(j&1?(mod-1):1, mul(C(m-1,j),C(n+m-i*(j+1)-2,m-2)) ));
    printf("%d\n",mul(ans,m));
}
int main(){
#ifdef Judge
freopen("1.in","r",stdin);
#endif
    int T=read(); prep();
    while(T--) Solv(); return 0;
}

原文地址:https://www.cnblogs.com/Judge/p/11827173.html

时间: 2024-11-29 06:28:51

【51nod 1251】 Fox序列的数量(以及带限制插板法讲解)的相关文章

【51nod】1251 Fox序列的数量

题解 容斥题 我们枚举出现次数最多的数出现了K次 然后我们需要计算的序列是所有数字出现个数都不超过K - 1次 我们枚举不合法的数字的数目j,说明这个排列里除了我们固定出现K次的数至少有j个数是不合法的,先让这j个数每个数出现k次,然后再随意排列 j最大是N / K 那么复杂度就是调和级数了 代码 #include <bits/stdc++.h> //#define ivorysi #define enter putchar('\n') #define space putchar(' ') #

Comparing Your Heros拓扑序列的数量

给出N行英雄的比较,每一行包含两个英雄的名字,代表第一个英雄比第二个英雄更受欢迎. 英雄的数目不超过16个.问有多少种可能的受欢迎程度的序列满足N行英雄的比较. 由于只有英雄数目不超过16个,可以用二进制来解决. x的的位表示还有哪些点没有处理.在这些点中选择一个入度没零的点,求剩下的点可能排列的数目. 1 #include <iostream> 2 #include <cstdio> 3 #include <cstring> 4 #include <cmath&

【BZOJ】【1251】序列终结者

Splay 还是splay序列维护,这题我WA了的原因是:在Push_up的时候,当前子树的max我是直接取的L.R和v[x]的最大值,但是如果没有左/右儿子,默认是会访问0号结点的mx值,而这个值没有初始化成-INF,所以就会导致所有负max值全部变为0…… 1 /************************************************************** 2 Problem: 1251 3 User: Tunix 4 Language: C++ 5 Resul

51nod 1476 括号序列的最小代价 (括号题套路+反悔贪心)

题意:给一串只有'(' , ')' , '?' 的括号序列,每个?可以变成)或者(,代价分别为bi和ai,求变成合法序列的最小代价 思路:学习自最近的网络赛&&51nod贪心专题视频的思想,“反悔”,一般在获取收益有限制的情况下使用 先按某种“优”的策略贪心,如果不满足限制条件了,取一个修改后代价尽可能小的状态修改成满足条件的状态,得到新的满足限制下的最优解 这种贪心常常可以借助优先队列实现 然后是括号题的套路:把(当做1,把)当做-1,做前缀和 这题中,先当做所有的?都换成右括号,这是显

CodeVS 1359 数字计数 51nod 1042 数字0-9的数量 Pascal

题目大意: 我的代码又臭又长,但是毕竟是我这个jr想了几天才推出的公式,看别的大神都写数位DP,所以我决定分享一下我的思路.我认为我的思路一向是最好理解的! 要分两种情况讨论: 1.0的情况.我们首先推出0~9中只有1个0,在0~99中有(1+(9*100*1))个0{第一位可以为1~9,第二位可以为0~9,0只可以放在后者,所以乘1},在0~999中有((1+(9*100*1))+(9*101*2))个0~6666为例,先算出0~999中0的个数,在算出1000~5999中0的个数,则为(10

51nod 1276 1276 岛屿的数量 (很好玩的题目

题意: 有N个岛连在一起形成了一个大的岛屿,如果海平面上升超过某些岛的高度时,则这个岛会被淹没.原本的大岛屿则会分为多个小岛屿,如果海平面一直上升,则所有岛都会被淹没在水下. 给出N个岛的高度.然后有Q个查询,每个查询给出一个海平面的高度H,问当海平面高度达到H时,海上共有多少个岛屿.例如: 岛屿的高度为:{2, 1, 3, 2, 3}, 查询为:{0, 1, 3, 2}. 当海面高度为0时,所有的岛形成了1个岛屿. 当海面高度为1时,岛1会被淹没,总共有2个岛屿{2} {3, 2, 3}. 当

51nod 1132 覆盖数字的数量 V2

http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1132 题意是给定a,b,l,r求[l,r]内有几个整数可以表示成ax+by(x,y为非负整数). 直接算l<=ax+by<=r会重复计算一个数的多种表示方法,而两种表示方法(x,y)总是相差k*(b,-a),因此限制y取最小值进行去重 即x>=0,y>=0,l<=ax+by<=r,y<a/gcd(a,b)五个半平面的交的整点个数,可以分类

51nod 1009 数字1的数量 数位dp

1009 数字1的数量 基准时间限制:1 秒 空间限制:131072 KB 给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数. 例如:n = 12,包含了5个1.1,10,12共包含3个1,11包含2个1,总共5个1. Input 输入N(1 <= N <= 10^9) Output 输出包含1的个数 Input示例 12 Output示例 5 #include<bits/stdc++.h> using namespace std; #define

51nod 1042 数字0-9的数量 (数位dp、dfs、前导0)

1042 数字0-9的数量 基准时间限制:1 秒 空间限制:131072 KB 分值: 10 难度:2级算法题 收藏 关注 取消关注 给出一段区间a-b,统计这个区间内0-9出现的次数. 比如 10-19,1出现11次(10,11,12,13,14,15,16,17,18,19,其中11包括2个1),其余数字各出现1次. Input 两个数a,b(1 <= a <= b <= 10^18) Output 输出共10行,分别是0-9出现的次数 Input示例 10 19 Output示例