edu 183 div2
edu 183 div2
div2
D
假若存在一个满足条件的构造,则最终的排列一定是由若干极长递增子段拼成的,一个区间如果只属于某一个极长递增子段,则这个区间就不包含逆序对,也就不会对 \(k\) 产生贡献;如果一个区间跨越了多个极长递增子段,则这个区间就包含逆序对,也就会对 \(k\) 产生贡献。相对于考虑包含了逆序对的区间,不包含逆序对的区间更容易考虑,所以我们从不包含逆序对的区间入手,尝试构造一个有 \({n(n - 1) \over 2} - k\) 个不含逆序对的区间的排列。
在考虑不包含逆序对区间的数量时,我们只关心有多少个极长递增子段和各自的长度,假设第 \(i\) 个极长递增子段的长度为 \(l_i\),则总共的不含逆序对的区间的数量就是 \(\sum _i {l_i(l_i - 1) \over 2}\),我们这道了这个式子的答案(即\({n(n - 1) \over 2} - k\)),现在要做的就是构造一组满足式子的 \(l_i\)。由于数据范围小,且也满足 dp 的条件,接下来的做法多种多样,dp 也好爆搜也好都能通过,下面给出爆搜的代码
std::vector<int> ans;
int n;
int num;
bool vis[N + 5][N * N];
bool dfs(int cur, int k) {
if (cur == 0) {
return k == 0;
}
int mx = cur * (cur - 1) / 2;
if (vis[cur][k] || k > mx || k < 0) {
return false;
}
vis[cur][k] = true;
for (int i = 1; i <= cur; ++i) {
if (dfs(cur - i, k - i * (i - 1) / 2)) {
for (int j = i - 1; j >= 0; --j) {
ans.push_back(num - j);
}
num -= i;
return true;
}
}
return false;
}
void solve() {
int k = 0;
std::cin >> n >> k;
for (int i = 1; i <= n; ++i) {
for (int j = 0; j * 2 < n * n; ++j) {
vis[i][j] = false;
}
}
num = n;
ans.clear();
if (dfs(n, n * (n - 1) / 2 - k)) {
for (auto &i : ans) {
std::cout << i << ' ';
}
std::cout << '\n';
}
else {
std::cout << "0\n";
}
return;
}
E
小清新数据结构题
假设第 \(i\) 个人对观影人数的阈值是 \(p_i\),则对于 \(i\) 来说,需要至少 \(p_i\) 个 \(j\) 满足 \(p_j < p_i\)。于是我们可以处理出 \(cnt_p\) 表示观影人数阈值严格小于 \(p\) 的有多少人,当 \(cnt_p - p < 0\) 时,观影人数阈值为 \(p\) 的人一定都不会观影。但是 \(0 \leq cnt_p - p\) 却不一定说明观影人数阈值为 \(p\) 的会去观影,因为可能存在 \(q < p\) 且 \(cnt_q - q < 0\)。所以我们要找的就是最小的满足 \(cnt_p - p < 0\) 的 \(p\),\(cnt_p\) 就是总共回去观影的人。于是维护一下 \(cnt_p\) 和 \(cnt_p - p\),实现区间修改、区间询问 \(cnt_p - p\) 的最小值和单点查询就好了,线段树易维护。下面给出线段树节点的定义和单点查询的代码:
struct Info {
int mn;
int r; // 最右边的值
int tag;
Info (int pos = 0, int val = 0) : mn(val - pos), r(val), tag(0) {}
Info (const Info &u, const Info &v) {
mn = std::min(u.mn, v.mn);
r = v.r;
tag = 0;
}
Info operator + (const Info &u) {
return Info(*this, u);
}
void add(int val) {
mn += val;
r += val;
tag += val;
return;
}
} tr[M << 3];
// 找到第一个 mn 小于 0 的位置
Info find(int cur, int l, int r) {
if (l == r) {
return tr[cur];
}
push_down(cur);
int m = l + r >> 1;
if (tr[cur << 1].mn < 0) {
return find(cur << 1, l, m);
}
else {
return find(cur << 1 | 1, m + 1, r);
}
}
需要注意的是线段树最后要维护 \(2e6\) 个数,空间别开小了。
浙公网安备 33010602011771号