Codeforces Global Round 7 题解
Visits: 658
Bad Ugly Numbers
特判掉 $n=1$ 的情况,然后 $8999\cdots 9$ 就符合要求。
inline void solve() {
    int n;
    rd(n);
    if (n == 1) return puts("-1"), void();
    putchar('8');
    while (--n) putchar('9');
    puts("");
}
int main() {
    int T;
    rd(T);
    while (T--) solve();
    return 0;
}
Maximums
依照题意,我们可以得到一个递推式 $a_n = \max_{i=1}^{n-1} a_i + b_n$,时间复杂度 $\mathcal O(n)$。
const int N = 2e5 + 7;
int n;
ll a[N], x;
int main() {
    rd(n), rda(a, n);
    for (int i = 1; i <= n; i++)
        print(x + a[i], " \n"[i==n]), x = max(x, x + a[i]); 
    return 0;
}
Permutation Partitions
最大值肯定可以选择 $n,n-1,\cdots,n-k+1$ 这 $k$ 个数相加。
求方案数时,相邻两个区间的间隔一定在两个选择的数中间,因此将位置之差相乘即可。
时间复杂度 $\mathcal O(n)$。
const int N = 2e5 + 7;
int n, k;
ll a[N], ans;
modint Ans = 1;
int main() {
    rd(n), rd(k), rda(a, n);
    for (int i = 1; i <= k; i++) ans += n + 1 - i;
    print(ans, ' ');
    vi p;
    for (int i = 1; i <= n; i++)
        if (a[i] > n - k) p.pb(i);
    for (ui i = 1; i < p.size(); i++)
        Ans *= p[i] - p[i-1];
    print(Ans);
    return 0;
}
Prefix-Suffix Palindrome
显然可以先贪心地取互为反串的最长前后缀。
取完扔掉之后,相当于要在剩下的字符串的前缀或后缀找到一个最长的回文串。
manacher 即可,时间复杂度 $\mathcal O(n)$。
const int N = 1e6 + 7;
int n, m;
char s[N], t[N];
inline int work(char *s, int n) {
    string ss = "$#";
    vi p;
    for (int i = 1; i <= n; i++) ss += s[i], ss += "#";
    p.pb(1);
    int mx = 0, id = 0, ans = 1;
    for (int i = 1; i < (int)ss.length(); i++) {
        p.pb(mx > i ? min(p[2*id-i], mx - i) : 1);
        while (i + p[i] < (int)ss.length() && ss[i+p[i]] == ss[i-p[i]]) ++p[i];
        if (i == p[i]) ans = max(ans, p[i]);
        if (i + p[i] > mx) mx = i + p[i], id = i;
    }
    return ans - 1;
}
inline void solve() {
    rds(s, n);
    int p = 1;
    while (p <= n && s[p] == s[n+1-p]) ++p;
    if (p == n + 1) {
        for (int i = 1; i <= n; i++) putchar(s[i]);
        return puts(""), void();
    }
    m = 0;
    for (int i = p; i <= n + 1 - p; i++) t[++m] = s[i];
    int l = work(t, m);
    reverse(t + 1, t + m + 1);
    int r = work(t, m);
    reverse(t + 1, t + m + 1);
    if (l < r) reverse(t + 1, t + m + 1), swap(l, r);
    for (int i = 1; i < p; i++) putchar(s[i]);
    for (int i = 1; i <= l; i++) putchar(t[i]);
    for (int i = p - 1; i; i--) putchar(s[i]);
    puts("");
}
int main() {
    int T;
    rd(T);
    while (T--) solve();
    return 0;
}
Bombs
首先有答案是递减的。
对于当前答案 $x$,我们如何判断它时候能被保留下来呢?
可以发现,$x$ 能留下当且仅当存在一个位置满足它右边 $\ge x$ 的数大于它右边的炸弹数。
于是线段树维护全局 $\max$ 和区间修改即可,时间复杂度 $\mathcal O(n \log n)$。
const int N = 3e5 + 7;
int n, a[N], b[N], p[N], ans;
struct T {
    int l, r, x, z;
} t[N<<2];
void build(int p, int l, int r) {
    t[p].l = l, t[p].r = r;
    if (l == r) return;
    build(ls, l, md), build(rs, md + 1, r);
    t[p].x = max(t[ls].x, t[rs].x);
}
void spd(int p) {
    if (t[p].z) {
        t[ls].x += t[p].z;
        t[ls].z += t[p].z;
        t[rs].x += t[p].z;
        t[rs].z += t[p].z;
        t[p].z = 0;
    }
}
void add(int p, int l, int r, int k) {
    if (t[p].l >= l && t[p].r <= r) return t[p].x += k, t[p].z += k, void();
    spd(p);
    if (l <= md) add(ls, l, r, k);
    if (r > md) add(rs, l, r, k);
    t[p].x = max(t[ls].x, t[rs].x);
}
int main() {
    rd(n);
    for (int i = 1; i <= n; i++) rd(a[i]), p[a[i]] = i; 
    for (int i = 1; i <= n; i++) rd(b[i]);
    build(1, 1, n), ans = n, print(n, ' ');
    add(1, 1, p[n], 1);
    for (int i = 1; i < n; i++) {
        add(1, 1, b[i], -1);
        while (t[1].x <= 0) add(1, 1, p[--ans], 1);
        print(ans, ' ');
    }
    return 0;
}
Wise Men
题目太神仙了,先咕咕咕着。
Spiderweb Trees
题目太神仙了,先咕咕咕着。


