ZJL 的妹子序列

暴力就是 \(\Theta(n\times m)\)

如果 \(n,m \le 10^5\) ?

考虑问题的转换,设 \(a_i\) 表示 \(i\) 小的在它后面的数的个数

\(0\le a_i \le i-1\),显然任何一个满足要求的 \(a\) 数列都可以从大到小放数字构成一个满足要求的排列

那么就是要求 \(0\le a_i \le i-1,\sum_{i=1}^{n}a_i=m\) 的方案数

考虑生成函数

\[\prod_{i=1}^{n}\sum_{j=0}^{i-1}x^j
\]

\[\prod_{i=1}^{n}\frac{1-x^i}{1-x}
\]

方法一

求 \(Ln\) 后,即

\[\sum_{i=1}^{n}Ln(1-x^i)-nLn(1-x)
\]

\[Ln(1-x^i)=-\sum_{j=1}\frac{x^{ij}}{j}
\]

证明:

\[Ln(F(x))=G(x)
\]

\[\frac{F'(x)}{F(x)}=G'(x)
\]

\[\frac{-ix^{i-1}}{1-x^i}=G'(x)
\]

\[\sum_{j=0}-ix^{i-1}x^{ij}=G'(x)
\]

\[\sum_{j=0}\frac{-ix^{ij+i}}{ij+i}=G(x)
\]

\[\sum_{j=1}\frac{-x^{ij}}{j}
\]

证毕

以调和级数的复杂度 \(\Theta(nlnn)\)求出来后 \(exp\) 就好了

方法二

\[\prod_{i=1}^{n}\frac{1-x^i}{1-x}
\]

\[\prod_{i=1}^{n}(1-x^i)(\frac{1}{1-x})^n
\]

后面的

\[(\frac{1}{1-x})^n=(\sum_{i=0}x^i)^n
\]

即就是从 \(n\) 个不同的集合中间可以重复的取出 \(i\) 个的方案数的生成函数

那么就是

\[\sum_{i=0}(^{n+i-1}_{n-1})x^i
\]

考虑前面的

\[\prod_{i=1}^{n}(1-x^i)
\]

相当于一个带符号的背包计数问题

而第 \(i\) 个的体积是 \(i\)

经典 \(DP\),就是选出一个上升的序列的方案数

设 \(f[i][j]\) 表示选了 \(i\) 个,和为 \(j\) 的方案

\(f[i][j]=f[i][j-i]-f[i-1][j-i]\) (带符号)

考虑到可能有某个的大小超过 \(n\)

那么 \(f[i][j]+=f[i-1][j-(n+1)]\)

由于每个体积不同,所以最多选出 \(\sqrt{n}\) 个

复杂度 \(\Theta(n\sqrt{n})\)

#include <bits/stdc++.h>
using namespace std; namespace IO {
const int maxn((1 << 21) + 1); char ibuf[maxn], *iS, *iT, c;
int f; char Getc() {
return (iS == iT ? (iT = (iS = ibuf) + fread(ibuf, 1, maxn, stdin), (iS == iT ? EOF : *iS++)) : *iS++);
} template <class Int> void In(Int &x) {
for (f = 1, c = Getc(); c < '0' || c > '9'; c = Getc()) f = c == '-' ? -1 : 1;
for (x = 0; c <= '9' && c >= '0'; c = Getc()) x = (x << 3) + (x << 1) + (c ^ 48);
x *= f;
}
} using IO :: In; const int mod(998244353);
const int maxn(2e5 + 5); int fac[maxn], ifac[maxn], inv[maxn], n, m, f[500][maxn]; inline void Inc(int &x, int y) {
x += y;
if (x >= mod) x -= mod;
} inline int C(int x, int y) {
if (y > x) return 0;
return 1LL * fac[x] * ifac[y] % mod * ifac[x - y] % mod;
} int main() {
fac[0] = fac[1] = ifac[0] = ifac[1] = inv[1] = 1;
for (int i = 2; i <= 200000; ++i) {
inv[i] = 1LL * (mod - mod / i) * inv[mod % i] % mod;
fac[i] = 1LL * fac[i - 1] * i % mod;
ifac[i] = 1LL * ifac[i - 1] * inv[i] % mod;
}
In(n), In(m);
int sz = 500, ans = 0;
f[0][0] = 1;
for (int i = 1; i < sz; ++i)
for (int j = 0; j <= m; ++j) {
if (j >= i) f[i][j] = (f[i][j - i] - f[i - 1][j - i] + mod) % mod;
if (j >= n + 1) Inc(f[i][j], f[i - 1][j - n - 1]);
}
for (int i = 0; i <= m; ++i) {
int ret = 0;
for (int j = 0; j < sz; ++j) Inc(ret, f[j][i]);
Inc(ans, 1LL * ret * C(n + m - i - 1, n - 1) % mod);
}
printf("%d\n", ans);
return 0;
}

[HAOI2009]逆序对数列(加强)的更多相关文章

  1. bzoj2431:[HAOI2009]逆序对数列

    单组数据比51nod的那道题还弱...而且连优化都不用了.. #include<cstdio> #include<cstring> #include<cctype> ...

  2. BZOJ 2431: [HAOI2009]逆序对数列( dp )

    dp(i,j)表示1~i的全部排列中逆序对数为j的个数. 从1~i-1的全部排列中加入i, 那么可以产生的逆序对数为0~i-1, 所以 dp(i,j) = Σ dp(i-1,k) (j-i+1 ≤ k ...

  3. 2431: [HAOI2009]逆序对数列

    2431: [HAOI2009]逆序对数列 Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 954  Solved: 548[Submit][Status ...

  4. P2513 [HAOI2009]逆序对数列

    P2513 [HAOI2009]逆序对数列 题目描述 对于一个数列{ai},如果有iaj,那么我们称ai与aj为一对逆序对数.若对于任意一个由1~n自然数组成的数列,可以很容易求出有多少个逆序对数.那 ...

  5. bzoj千题计划153:bzoj2431: [HAOI2009]逆序对数列

    http://www.lydsy.com/JudgeOnline/problem.php?id=2431 dp[i][j] 表示i的排列,有j个逆序对的方案数 加入i+1,此时i+1是排列中最大的数, ...

  6. BZOJ2431 HAOI2009 逆序对数列 【DP】*

    BZOJ2431 HAOI2009 逆序对数列 Description 对于一个数列ai{a_i}ai​,如果有i<j且ai>aja_i>a_jai​>aj​,那么我们称aia ...

  7. 洛谷P2513 [HAOI2009]逆序对数列

    P2513 [HAOI2009]逆序对数列 题目描述 对于一个数列{ai},如果有i<j且ai>aj,那么我们称ai与aj为一对逆序对数.若对于任意一个由1~n自然数组成的数列,可以很容易 ...

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

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

  9. Bzoj 2431 HAOI2009 逆序对数列

    Description 对于一个数列{ai},如果有i**<**j且ai>aj,那么我们称ai与aj为一对逆序对数.若对于任意一个由1~n自然数组成的数列,可以很容易求出有多少个逆序对数. ...

  10. [HAOI2009]逆序对数列

    题目描述 对于一个数列{ai},如果有i<j且ai>aj,那么我们称ai与aj为一对逆序对数.若对于任意一个由1~n自然数组成的数列,可以很容易求出有多少个逆序对数.那么逆序对数为k的这样 ...

随机推荐

  1. datetime 模块

    datetimo 模块和time模块类似,只不过直接帮你定好了格式 import datetime time =datetime.datetime.now() print(time,type(time ...

  2. 机器学习笔记(四)--sklearn数据集

    sklearn数据集 (一)机器学习的一般数据集会划分为两个部分 训练数据:用于训练,构建模型. 测试数据:在模型检验时使用,用于评估模型是否有效. 划分数据的API:sklearn.model_se ...

  3. leetcode-409-Longest Palindrome(统计字母出现次数)

    题目描述: Given a string which consists of lowercase or uppercase letters, find the length of the longes ...

  4. Gradle学习系列(三)

    上一篇我们已经学习了 Gradle 打包,那么这一节 就开始讲讲本章的第二版块:打包及使用Gradle过程中的常见错误问题. 有时候我们需要编辑一个开源项目之类的,希望把它导入了我们自己的IDE工具中 ...

  5. Android开发不可或缺的十大网站及工具

    1. Google 做开发前完全是小白,真心不知道有Google这东西,只晓得百度,遇到问题直接百度,不是黑百度,百度在娱乐八卦方面确实靠谱,但是技术方面查出来的东西基本千篇一律,有些答案甚至还会起到 ...

  6. JavaScript设计模式(一)

    什么是设计模式呢? 就是指对于类似的问题,我们可以用大致相同的思想.方法去解决之,而这种通用的思想.方法就是设计模式.学习设计模式可以帮助我们在遇到问题时迅速地搜索出一种清晰的思路来实现之. 第一部分 ...

  7. Javac的命令(-Xlint)

    在OptionName类中的枚举定义如下: XLINT("-Xlint"), XLINT_CUSTOM("-Xlint:"), -Xlint     Enabl ...

  8. jdbc oracle clob

    import java.io.BufferedReader; import java.io.Reader; import java.io.Writer; import java.sql.Callabl ...

  9. Python基础(10) - 异常

    Python 异常:程序出现了错误而在正常控制流以外采取的行为 Python中常见的异常: 1. NameError:尝试访问一个未声明的变量 >>> something Trace ...

  10. jmap查看内存使用情况与生成heapdump--转

    http://www.webspherechina.net/home/space.php?uid=191&do=blog&id=7572 如果想分析自己的JAVA Applicatio ...