多校冲刺 NOIP 20211106 模拟 (24)

T1

第一类斯特林数裸题

T2

一对相同的字符会产生一次重复的贡献,分计算26种字符的出现次数\(cnt_i\),然后总和减去\(\frac{cnt_i*(cnt_i-1)}{2}\)

T4

不难得出一个人只会向左一直走然后右转,或向右一直走然后左转,且不会经过旁边的人的初始位置

于是可以dp,\(f_{i,j}\)表示第 i 个人终止在第 j 个点,前 i 个人的最短时间

转移就考虑枚举第 i-1 个人的终止位置,和第 i 个人的终止位置,计算第 i 个人移动的时间

考虑到整个dp实际只会有\(O(n)\)的合法状态,可以考虑将第二维在unordered_map上进行

空间复杂度\(O(n)\),时间复杂度未知

代码

T1

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int mod = 998244353;
const int N = 3e3 + 11;
int n, k;
int f[N][N];
inline void md(int& x) {
    if (x >= mod)
        x -= mod;
    return;
}
inline int read() {
    int s = 0;
    char ch = getchar();
    while (ch > '9' || ch < '0') ch = getchar();
    while (ch >= '0' && ch <= '9') s = (s << 1) + (s << 3) + (ch ^ 48), ch = getchar();
    return s;
}
signed main() {
    FILE* x = freopen("broken.in", "r", stdin);
    x = freopen("broken.out", "w", stdout);
    n = read();
    k = read();
    f[0][0] = 1;
    int ans = 0;
    for (int i = 1; i <= n; ++i)
        for (int j = 1; j <= k; ++j) md(f[i][j] += f[i - 1][j - 1] + f[i - 1][j] * (i - 1) % mod);
    for (int i = 1; i <= k; ++i) md(ans += f[n][i]);
    cout << ans << endl;
    return 0;
}

T2

#include <bits/stdc++.h>
using namespace std;
#define ull unsigned long long
const ull p = 131;
const int N = 3e6 + 11;
char tx[N];
int n;
ull cnt[N];
ull mi[N];
int main() {
    FILE* h = freopen("turn.in", "r", stdin);
    h = freopen("turn.out", "w", stdout);
    cin >> (tx + 1);
    n = strlen(tx + 1);
    for (int i = 1; i <= n; ++i) ++cnt[tx[i] - 'a' + 1];
    ull sl = 0;
    for (int i = 1; i <= 26; ++i) sl += cnt[i] * (cnt[i] - 1) / 2;
    cout << 1ll * n * (n - 1) / 2 - sl + 1 << endl;
    return 0;
}

T4

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N = 3e6 + 11;
const int inf = 1e18;
int n, m;
int x[N], y[N];
int lsh[N], num;
unordered_map<int, int> f[N];
inline int max_(int a, int b) { return a > b ? a : b; }
inline int min_(int a, int b) { return a > b ? b : a; }
inline int calc(int l, int mid, int r) {
    return l = lsh[l], r = lsh[r], mid = lsh[mid], 2 * min_(mid - l, r - mid) + max_(mid - l, r - mid);
}
inline int read() {
    int s = 0;
    char ch = getchar();
    while (ch > '9' || ch < '0') ch = getchar();
    while (ch >= '0' && ch <= '9') s = (s << 1) + (s << 3) + (ch ^ 48), ch = getchar();
    return s;
}
void lsh_() {
    sort(lsh + 1, lsh + num + 1);
    int h = unique(lsh + 1, lsh + num + 1) - lsh;
    for (int i = 1; i <= n; ++i) x[i] = lower_bound(lsh + 1, lsh + h, x[i]) - lsh;
    for (int i = 1; i <= m; ++i) y[i] = lower_bound(lsh + 1, lsh + h, y[i]) - lsh;
    return;
}
signed main() {
    FILE* p = freopen("duplication.in", "r", stdin);
    p = freopen("duplication.out", "w", stdout);
    n = read(), m = read();
    for (int i = 1; i <= n; ++i) lsh[++num] = x[i] = read();
    for (int i = 1; i <= m; ++i) lsh[++num] = y[i] = read();
    sort(x + 1, x + n + 1);
    sort(y + 1, y + m + 1);
    lsh_();
    if (n == 1) {
        if (lsh[y[1]] > lsh[x[1]])
            cout << lsh[y[m]] - lsh[x[1]] << endl;
        else if (lsh[y[m]] < lsh[x[1]])
            cout << lsh[x[1]] - lsh[y[1]] << endl;
        else
            cout << calc(y[1], x[1], y[m]) << endl;
        return 0;
    }
    for (int i = x[1]; i < x[2]; ++i) {
        if (lsh[y[1]] < lsh[x[1]])
            f[1][i] = calc(y[1], x[1], i);
        else
            f[1][i] = lsh[i] - lsh[x[1]];
    }
    for (int i = 2; i < n; ++i) {
        for (int k = x[i]; k <= x[i + 1] - 1; ++k) f[i][k] = inf;
        for (int j = x[i - 1] + 1; j <= x[i]; ++j)
            for (int k = x[i]; k <= x[i + 1] - 1; ++k)
                f[i][k] = min_(f[i][k], max_(f[i - 1][j - 1], calc(j, x[i], k)));
    }
    if (lsh[y[m]] > lsh[x[n]])
        f[n][y[m]] = inf;
    else
        f[n][x[n]] = inf;
    for (int i = x[n - 1] + 1; i <= x[n]; ++i) {
        if (lsh[y[m]] > lsh[x[n]])
            f[n][y[m]] = min_(f[n][y[m]], max_(f[n - 1][i - 1], calc(i, x[n], y[m])));
        else
            f[n][x[n]] = min_(f[n][x[n]], max_(f[n - 1][i - 1], calc(i, x[n], x[n])));
    }
    if (lsh[y[m]] > lsh[x[n]])
        cout << f[n][y[m]] << endl;
    else
        cout << f[n][x[n]] << endl;
    return 0;
}
posted @ 2021-11-15 06:21  sitiy  阅读(34)  评论(0编辑  收藏  举报