CF 1091E New Year and the Factorisation Collaboration

昨晚Good Bye 2018D题没做出来,车翻大了……

官方题解      传送门

初赛知识:一个无向图所有顶点度数之和为偶数。然而这东西还有一个高端的名字:Handshaking lemma

但是这并不是本题的重点,另外一个看上去很高端的东西才是本题的重点:Erdős–Gallai theorem

对于一个无向图的度数序列$d$,先从大到小排序,即满足$d_1 \geq d_2 \geq d_3 \geq \dots \geq d_n$,

那么对于$\forall k \in [1, n]$,均满足

$$\sum_{i = 1}^{k}d_i \leq k(k - 1) + \sum_{i = k + 1}^{n}min(k, d_i)$$

意思就是先选出度数前$k$大的点然后让它们生成一张完全图,然后剩下的点无论怎么连一定是一张合法的无向图。

我们注意到在这题中,如果$x,y$是两个合法的答案(不妨设$x<y$),那么如果$z$满足$z \in (x, y)$并且$x \mod 2 == z \mod 2$,$z$也是一个合法的答案。也就是说,我们只要做出这个答案的区间$[L, R]$,然后检验每一个$i \in [L, R]$是否满足那个初赛知识就好了。

考虑如何找这个区间。

首先把度数序列从大到小排个序然后弄个前缀和,我们去扫描每一个位置,把当前扫到的位置$i$作为Erdős–Gallai theorem中的$k$,因为后面都是有序序列,所以那个$min$只要二分找到一个分界点$j$使左边都大于$i$,右边都小于等于$i$,结合前缀和就可以算出式子两边的值。

假设左边的和为$a$,右边的和为$b$,考虑$n + 1$个点可以放在哪个位置(假设第$n + 1$个点的度数为$x$),有以下几种情况:

1、$a > b$,如果$a > b + i$,那么直接无解。

2、观察到当$n + 1$个点放在$3$的时候,有$b + x \geq a$,那么$x \geq a - b$。

3、当$n + 1$个点放在$1$的时候,第$i$个位置实际上变成了第$i + 1$个位置,但是这并不影响前缀和的计算,这时候满足$a - d_i + x \leq b + i$,那么$x \leq b + i - a + d_i$。

时间复杂度$O(nlogn)$。

#include <cstdio>
#include <cstring>
#include <vector>
#include <algorithm>
using namespace std;
typedef long long ll;

const int N = 5e5 + 5;

int n;
ll a[N], sum[N];
vector <int> ans;

bool cmp(ll x, ll y) {
    return x > y;
}

template <typename T>
inline void read(T &X) {
    X = 0; char ch = 0; T op = 1;
    for (; ch > '9'|| ch < '0'; ch = getchar())
        if (ch == '-') op = -1;
    for (; ch >= '0' && ch <= '9'; ch = getchar())
        X = (X << 3) + (X << 1) + ch - 48;
    X *= op;
}

template <typename T>
inline void chkMin(T &x, T y) {
    if (y < x) x = y;
}

template <typename T>
inline void chkMax(T &x, T y) {
    if (y > x) x = y;
}

int main() {
    read(n);
    for (int i = 1; i <= n; i++) read(a[i]);
    sort(a + 1, a + 1 + n, cmp);
    for (int i = 1; i <= n; i++) sum[i] = sum[i - 1] + a[i];
    
    ll ln = 0, rn = n;
    for (int i = 1; i <= n; i++) {
        int j = lower_bound(a + 1 + i, a + 1 + n, i, cmp) - a;
        ll lsum = sum[i], rsum = 1LL * (j - i - 1) * i + sum[n] - sum[j - 1] + 1LL * i * (i - 1);
        if (lsum > rsum) {
            if (lsum - rsum > i) return puts("-1"), 0;
            chkMax(ln, lsum - rsum);
        }
        chkMin(rn, a[i] + rsum + i - lsum);
    }
    
    for (int i = ln; i <= rn; i++)
        if (!((i - sum[n]) & 1)) ans.push_back(i);
    
    if (ans.empty()) puts("-1");
    else {
        int siz = ans.size();
        for (int i = 0; i < siz; i++)
            printf("%d%c", ans[i], i == siz - 1 ? '\n' : ' ');
    }
    
    return 0;
}
View Code

 

 

猜你喜欢

转载自www.cnblogs.com/CzxingcHen/p/10201761.html
今日推荐