抱歉,您的浏览器无法访问本站

本页面需要浏览器支持(启用)JavaScript


了解详情 >

原题链接

n(n105)n(n\le 10^5)个球排成一排,第 ii 个球有颜色 ci(cin)c_i(c_i\le n) 和重量 wi(wi109)w_i(w_i\le 10^9)。 Snuke 每次可以选择两个颜色相同,且重量之和不超过xx 的球,交换他们的位置。 Snuke 每次可以选择两个颜色不同,且重量之和不超过 yy 的球,交换他们的位置。 问可以得到多少种不同的颜色序列。1x,y1091\le x,y\le 10^9,答案对 109+710^9+7 取模。[1]

题解

我们考虑什么时候连个球能进行交换。

显然如果有三个球 a,b,ca,b,c,如果aabb能交换,bbcc 能交换,那 aacc一定能通过 bb 的“媒介”交换(aa先于 bb 交换,bbcc 交换,aabb 交换)。

于是我们可以把 a,b,ca,b,c 缩在同一个连通块里。

由于不同颜色与相同颜色交换条件是不一样的,我们先考虑相同颜色。

显然只需要用最小值做“媒介”即可。

然后我们考虑跨颜色交换。我们发现只需要用全局最小值做“媒介”,也就是说对于某个点,先通过该颜色最小值换到全局最小值,再通过全局最小值换到其他点。如果与全局最小值颜色相同,那我们就只能用次小值做媒介。

这样我们只需要记录某种颜色的最小值、全局最小值、全局次小值即可。

然后我们不难发现含有不同颜色的连通块只有一个,那就是全局最小值所在的连通块。

而只有一种颜色的连通块对答案没有影响。

于是我们只考虑最小值所在的连通块。首先我们找到全局最小值、全局次小值与每种颜色的最小值。如果他们都不能交换,那显然是凉了(答案为11)。我们对每种颜色数出能与该种颜色最小值交换的点数,然后用每种连通块的最小值与全局最小值(全局次小值)比较,求出这个联通块。我们发现就是求这个联通块排列方案数,其实就是可重集合排列。

代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>

using namespace std;

typedef long long LL;

const int maxn = 1000005;
const int inf = 0x3f3f3f3f;
const int mod = 1000000007;

int n, A, B;
int fac[maxn], inv[maxn];
int c[maxn], w[maxn], minci[maxn];
int Gs[maxn];

inline int pow(int a, int b)
{
int ans = 1;
for(; b; b >>= 1, a = (LL) a * a % mod)
if(b & 1)
ans = (LL) ans * a % mod;
return ans;
}

inline int C(int n, int m)
{
if(!n)
return 1;
if(m < n)
return 0;
return (LL) fac[m] * (LL) inv[m - n] % mod * (LL) inv[n] % mod;
}

int main()
{
scanf("%d%d%d", &n, &A, &B);
memset(minci, 0x3f, sizeof(minci));
for(int i = 1; i <= n; ++i)
{
scanf("%d%d", &c[i], &w[i]);
minci[c[i]] = min(minci[c[i]], w[i]);
}
int minn = 0, minnn = 0;
for(int i = 1; i <= n; ++i)
{
if(minci[i] < minci[minn])
{
minnn = minn;
minn = i;
}
else if(minci[i] < minci[minnn])
minnn = i;
}
int gss = 0;
for(int i = 1; i <= n; ++i)
{
if(w[i] + minci[c[i]] <= A)
w[i] = minci[c[i]];
if(c[i] == minn && minci[minnn] + w[i] <= B)
{
Gs[c[i]]++;
gss++;
}
if(c[i] != minn && minci[minn] + w[i] <= B)
{
Gs[c[i]]++;
gss++;
}
}
int ans = 1;
fac[0] = 1;
for(int i = 1; i <= n; ++i)
fac[i] = (LL) i * fac[i - 1] % mod;
inv[n] = pow(fac[n], mod - 2);
for(int i = n - 1; ~i; --i)
inv[i] = (LL) inv[i + 1] * (i + 1) % mod;
for(int i = 1; i <= n; ++i)
{
if(Gs[i])
{
ans = (LL) ans * (LL) C(Gs[i], gss) % mod;
gss -= Gs[i];
}
}
printf("%d\n", ans);
return 0;
}

  1. 翻译来自luogu,有删改 ↩︎

评论