Codeforces Global Round 7 题解

作者: xht37 分类: 题解 发布时间: 2020-03-22 03:40

Visits: 230

Codeforces Global Round 7

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

题目太神仙了,先咕咕咕着。

发表评论

电子邮件地址不会被公开。 必填项已用*标注