Codeforces Pinely Round 3 (Div. 1 + Div. 2) A~F

A.Distinct Buttons(思维)

题意:

你在开始时站在点(0,0)(0,0)(0,0),同时,手上有一个遥控器,上面有四个按钮:

  • U:移动到(x,y+1)(x, y + 1)(x,y+1)的位置

  • R:移动到(x+1,y)(x + 1, y)(x+1,y)的位置

  • D:移动到(x,y−1)(x, y - 1)(x,y1)的位置

  • L:移动到(x−1,y)(x - 1, y)(x1,y)的位置

如果四个按钮都被按下过,那么遥控器将会被损坏,问能否到达给出的所有nnn个点。

分析:

如果只能使用三个按键,那么只有在需要到达的所有点均在以下四个面中的一个时才能完成:

  • 所有点都在xxx轴上方

  • 所有点都在xxx轴下方

  • 所有点都在yyy轴左侧

  • 所有点都在yyy轴右侧

输入时使用数组记录出现的位置,最后判断输出即可。

代码:

#include<bits/stdc++.h>

using namespace std;
typedef long long LL;
const int MAXN = 3e5 + 5e2;

void solve() {
    int n;
    cin >> n;
    int a[5] = {0, 0, 0, 0};
    for (int i = 0; i < n; i++) {
        int x, y;
        cin >> x >> y;
        if (x > 0) {
            a[0] = 1;
        } else if (x < 0) {
            a[1] = 1;
        }
        if (y > 0) {
            a[2] = 1;
        } else if (y < 0) {
            a[3] = 1;
        }
    }
    if (a[0] + a[1] + a[2] + a[3] > 3) cout << "No" << endl;
    else cout << "Yes" << endl;
}

int main() {
    int Case;
    cin >> Case;
    while (Case--) {
        solve();
    }
    return 0;
}

B.Make Almost Equal With Mod(思维)

题意:

给出一个数组a1,a2,...,ana_1, a_2, ..., a_na1,a2,...,an,你可以选择一个数字kkk,使数组中所有数字对kkk取模,问kkk等于多少时,可以使得数组中的数字再操作后恰好包含两种不同的数字。

分析:

依次枚举222的次方数即可。

说明:

  • 选择222作为kkk时,剩下的结果仅包含0,10, 10,1

  • 如果剩下的数字全部为000111,继续选择22=42^2 = 422=4作为kkk,若选择222时剩下的数字为000,那么选择k=4k = 4k=4时剩下的就是0,20, 20,2,同理,剩下数字均为111,则选择k=4k = 4k=4时剩下的数字就是1,31, 31,3

  • 依此类推,由于每次取模的结果只会有两种可能性,那么当结果中两种情况均出现就找到了合法的kkk

代码:

#include<bits/stdc++.h>

using namespace std;
typedef long long LL;
const int MAXN = 3e5 + 5e2;

LL a[MAXN];

void solve() {
    int n;
    cin >> n;
    for (int i = 1; i <= n; i++) cin >> a[i];
    for (LL i = 2; ; i <<= 1) {
        set<LL> S;
        for (int j = 1; j <= n; j++) {
            S.insert(a[j] % i);
            if (S.size() > 2) break;
        }
        if (S.size() == 2) {
            cout << i << endl;
            return;
        }
    }
}

int main() {
    int Case;
    cin >> Case;
    while (Case--) {
        solve();
    }
    return 0;
}

C.Heavy Intervals(思维)

题意:

给出nnn个区间[li,ri][l_i, r_i][li,ri],每个区间包含一个权值cic_ici,且区间的价值为:ci×(ri−li)c_i \times (r_i - l_i)ci×(rili),你可以在保证区间合法(li<ri)(l_i < r_i)(li<ri)的情况下,对所有区间的li,ri,cil_i, r_i, c_ili,ri,ci进行任意重排,问所有区间的价值之和最小是多少?

分析:

既然要让价值之和最小,那么大的权值cic_ici就要与长度更小的区间匹配,那要怎么在保证区间合法的情况下,让构造的区间尽可能长呢?

可以使用类似括号匹配的思想,先对lil_ilirir_iri进行排序,把lil_ilirir_iri视为左右括号进行括号匹配,并将构造成的区间长度记录下来。

完成匹配后,对权值cic_ici和构造的区间长度lenlenlen进行排序,并按最大的权值和最小的区间长度进行匹配,次大的权值和次小的区间长度进行匹配,依次类推,就能得到最小的价值之和。

代码:

#include<bits/stdc++.h>

using namespace std;
typedef long long LL;
const int MAXN = 3e5 + 5e2;

LL l[MAXN], r[MAXN], c[MAXN], len[MAXN];

stack<int> st;

void solve() {
    int n;
    cin >> n;
    for (int i = 0; i < n; i++) {
        cin >> l[i];
    }
    for (int i = 0; i < n; i++) {
        cin >> r[i];
    }
    for (int i = 0; i < n; i++) {
        cin >> c[i];
    }
    sort(l, l + n);
    sort(r, r + n);
    sort(c, c + n);
    int pos_l = 0, pos_r = 0, cnt = 0;
    for (int i = 0; i < n * 2; i++) {
        if (pos_l < n && pos_r < n) {
            if (l[pos_l] < r[pos_r]) {
                st.push(l[pos_l++]);
            } else {
                len[cnt++] = r[pos_r++] - st.top();
                st.pop();
            }
        } else {
            len[cnt++] = r[pos_r++] - st.top();
            st.pop();
        }
    }
    LL ans = 0;
    sort(len, len + n);
    for (int i = 0, j = n - 1; i < n; i++, j--) {
        ans += len[i] * c[j];
    }
    cout << ans << endl;
}

int main() {
    int Case;
    cin >> Case;
    while (Case--) {
        solve();
    }
    return 0;
}

D.Split Plus K(思维)

题意:

给出nnn个正整数a1,a2,...,ana_1, a_2, ..., a_na1,a2,...,an,你可以进行若干次以下操作:

  • 选择一个数字xxx,并将这个数字删除。

  • 选择两个正整数y,zy, zy,z满足y+z=x+ky + z = x + ky+z=x+k,并将这两个数字放回。

问:能否在经过若干次操作后,使数组中所有数字相同,如果可以,输出最少操作次数,否则,输出-1.

分析:

由于每次产生的两个数字y,zy, zy,z的总和会比原本的数字xxxkkk,因此,如果一个数字被分解了mmm次,那么得到的m+1m + 1m+1个数字的总和就是x+m×kx + m \times kx+m×k

b0,b1,...,bmb_0, b_1, ..., b_mb0,b1,...,bm为最后生成的m+1m + 1m+1个数字,由于分解时会增加kkk,且会增加mmm次,那么可以将mmmkkk分配给b1∼bmb_1 \sim b_mb1bm,即最后生成的数字为b0,b1+k,b2+k,...,bm+kb_0, b_1 + k, b_2 + k, ..., b_m + kb0,b1+k,b2+k,...,bm+k

由题目可得以下两个式子:

  • b0=b1+k=...=bm+kb_0 = b_1 + k = ... = b_m + kb0=b1+k=...=bm+k

  • b0+(b1+k)+...+(bm+k)=ai+m×kb_0 + (b_1 + k) + ... + (b_m + k) = a_i + m \times kb0+(b1+k)+...+(bm+k)=ai+m×k

由于无法知道最后分解出的数字数量mmm到底是多少,因此需要对式子进行化简,让第二个式子两边同时减去m+1m + 1m+1kkk,可得:

  • b0−k=b1=...=bmb_0 - k = b_1 = ... = b_mb0k=b1=...=bm

  • (b0−k)+b1+...+bm=ai−k(b_0 - k) + b_1 + ... + b_m = a_i - k(b0k)+b1+...+bm=aik

而此时所有的b1∼bmb_1 \sim b_mb1bm以及b0−kb_0 - kb0k均为ai−ka_i - kaik的因子,因此,可以在输入后,将所有aia_iai均减去kkk

然后,需要考虑,如果减去kkk后的aaa数组中出现了同时包含正负数或同时出现000和其他数字,此时是无法完成构造的,直接输出−1-11

最后考虑最后生成的数字,既然要让操作次数尽可能少,那么拆出的数字就要尽可能大,怎么选择最大的结果呢?只有选择减去kkk之后的所有ai−ka_i - kaik的最大公约数GGG

由于每次操作会增加一个数字,因此,将ai−ka_i - kaik分解为若干个GGG所需的操作次数为ai−kG−1\frac{a_i - k}{G} - 1Gaik1

代码:

#include<bits/stdc++.h>

using namespace std;
typedef long long LL;
const int MAXN = 3e5 + 5e2;

LL n, k, a[MAXN];

void solve() {
    cin >> n >> k;
    LL maxn = -1e18, minn = 1e18;
    for (int i = 1; i <= n; i++) cin >> a[i], a[i] -= k, maxn = max(maxn, a[i]), minn = min(minn, a[i]);
    if (maxn == minn) {
        cout << 0 << endl;
        return;
    }
    if (minn < 0 && maxn >= 0 || minn == 0 && maxn > 0) {
        cout << "-1" << endl;
        return;
    }
    LL g = a[1];
    for (int i = 2; i <= n; i++) g = __gcd(g, a[i]);
    LL ans = 0;
    for (int i = 1; i <= n; i++) {
        ans += a[i] / g - 1;
    }
    cout << ans << endl;
}

int main() {
    int Case;
    cin >> Case;
    while (Case--) {
        solve();
    }
    return 0;
}

E.Multiple Lamps(二进制)

题意:

现有nnn盏灯,编号为111nnn。一开始,所有的灯都是关闭的。

nnn个按钮。第iii个按钮控制所有编号为iii的倍数的灯。当一个灯其对应的按钮被使用时,如果它是关闭的,它将打开;如果它是打开的,它将关闭。

必须按照以下规则按一些按钮:

  • 至少要按一个按钮。
  • 不能多次按同一按钮。
  • mmm个数对(ui,vi)(u_i,v_i)(ui,vi),如果按下按钮uiu_iui,必须同时按下按钮viv_ivi(不一定是在按下按钮uiu_iui之后)。如果按下按钮viv_ivi,则不需要按按钮uiu_iui

为了环保,请按下一些按钮,使其最后最多点亮⌊n5⌋⌊\frac{n}{5}⌋5n盏灯,否则输出−1−11

分析:

如果按下所有的按钮,灯iii就会被iii的所有因数各操作一次,所以如果iii有奇数个因数,灯iii最终就会是亮的。对于这种情况,当⌊n⌋≤⌊n/5⌋⌊\sqrt{n}⌋ \le ⌊n/5⌋nn/5,即当20≤n20 \le n20n时,按下所有按钮即可。

n≤19n \le 19n19时,最多打开⌊195⌋=3⌊\frac{19}{5}⌋=3519=3盏灯。可以从111nnn遍历按钮,当且仅当灯iii处于错误状态时按下按钮iii。最多会遍历333个灯的所有子集,并检查相应的按钮选择是否有效(即mmm约束成立)。可以通过打表预处理出来,然后进行checkcheckcheck

代码:

#include<bits/stdc++.h>

using namespace std;
typedef long long LL;
const int MAXN = 2e5 + 5;
vector<int> arr[20];
int st[20];
int n, m;
int a[MAXN], b[MAXN];

int lowbit(int x) {
    return x & (-x);
}

void init() {
    for (int i = 5; i <= 19; i++) {
        for (int j = 1; j < (1 << i); j++) {
            for (int k = j; k; k ^= lowbit(k)) {
                int x = __builtin_ctz(k) + 1;
                for (int y = x; y <= i; y += x)
                    st[y] ^= 1;
            }
            int sum = 0;
            for (int k = 1; k <= i; k++) {
                sum += st[k], st[k] = 0;
            }
            if (sum * 5 <= i)
                arr[i].push_back(j);
        }
    }
}

bool solve() {
    for (int i: arr[n]) {
        i <<= 1;
        bool f = 0;
        for (int j = 1; j <= m; j++) {
            f |= (i & (1 << a[j])) && !(i & (1 << b[j]));
        }
        if (f)
            continue;
        cout << __builtin_popcount(i) << endl;
        while (i) {
            cout << __builtin_ctz(i) << ' ';
            i ^= lowbit(i);
        }
        cout << endl;
        return 1;
    }
    return 0;
}

int main() {
    init();
    int T;
    cin >> T;
    while (T--) {
        cin >> n >> m;
        for (int i = 1; i <= m; i++) {
            cin >> a[i] >> b[i];
        }
        if (n >= 20) {
            cout << n << endl;
            for (int i = 1; i <= n; i++)
                cout << i << ' ';
            cout << endl;
            continue;
        }
        if (!solve())
            cout << "-1" << endl;
    }
}

F1.Small Permutation Problem(Easy Version)(DP)

题意:

给定一个整数nnn和一个数组a1,a2,…,ana_1,a_2,…,a_na1,a2,,an,数据范围为[0,n][0,n][0,n]

对于每个iii,若满足以下条件,则p1,p2,…,pn([1,2,…,n])p_1,p_2,…,p_n([1,2,…,n])p1,p2,,pn([1,2,,n])是一个好的排列:

ai≠−1a_i≠−1ai=1时,在[p1,p2,…,pi][p_1,p2,…,p_i][p1,p2,,pi]中小于等于iii的数的个数恰好为aia_iai

计算[1,2,…,n][1,2,…,n][1,2,,n]中好的排列的个数,结果对998244353998244353998244353取模。

分析:

本题为动态规划的思想,设dpidp_idpi表示前iii个位置填充了aia_iai个小于等于iii的元素的方案数量。因为aia_iaiai−1a_{i-1}ai1的关系只有可能为0,1,20,1,20,1,2。故本题总共可分为三种情况讨论:

  • ai=ai+1a_i=a_{i+1}ai=ai+1,则dpi=dpi+1dp_i=dp_{i+1}dpi=dpi+1
  • ai=ai+1+1a_i=a_{i+1}+1ai=ai+1+1,则需要填入一个小于等于iii的数字,有两种方案:方案1:在前iii个位置填入iii数字,可以填入的方法有(i−1)−ai−1+1(i-1)-a_{i-1}+1(i1)ai1+1;方案2:在第iii个位置填写一个小于iii的数字,可以将iii或者之前未填写的小于等于iii的数字填入,可以填入的数字有i−1−ai−1i-1-a_{i-1}i1ai1
  • ai=ai+1+2a_i=a_{i+1}+2ai=ai+1+2,则需要在前i−1i-1i1个位置放数字iii,在第iii个位置放一个小于iii的数字

将上述三种情况分别考虑,遍历即可获得答案,注意需要特判ana_nan,若不为nnn,则直接输出000

代码:

#include <bits/stdc++.h>

using namespace std;
typedef long long LL;
const LL MOD = 998244353;

int n;
LL a[200010];

void solve() {
    cin >> n;
    for (int i = 1; i < n + 1; i++) {
        cin >> a[i];
    }
    if (a[1] > 1) {
        printf("0\n");
        return;
    }
    vector<LL> dp(n + 1, 0);
    dp[1] = 1;

    for (int i = 2; i < n + 1; i++) {
        if (a[i] == a[i - 1]) {
            dp[i] = dp[i - 1];
        } else if (a[i] == a[i - 1] + 1 and a[i] <= i) {
            (dp[i] += dp[i - 1] * ((i - 1) - a[i - 1] + 1)) %= MOD;
            (dp[i] += dp[i - 1] * ((i - 1) - a[i - 1])) %= MOD;
        } else if (a[i] == a[i - 1] + 2 and a[i] <= i) {
            (dp[i] += dp[i - 1] * ((i - 1) - a[i - 1]) % MOD * ((i - 1) - a[i - 1]) % MOD) %= MOD;
        }
    }
    if (a[n] == n) {
        cout << dp[n] << endl;
    } else {
        cout << 0 << endl;
    }
}

int main() {
    int t;
    cin >> t;
    while (t--)
        solve();
    return 0;
}

F2.Small Permutation Problem(Hard Version)(数学)

分析:

对于F2F2F2:将题目的点想象成一个n×nn \times nn×n的图,排列中第iii个数为pip_ipi时,即可视为选中了(i,pi)(i,p_i)(i,pi)这个点。

题目条件对aia_iai的限制,就可以看成是在左上角i×ii \times ii×i的矩阵中,选了aia_iai个点

因为有一些位置是−1-11,所以直接找到前一个限制的位置laslaslas。初始时,可以令las=0las=0las=0alas=0a_{las}=0alas=0,上一个约束处是laslaslas,限制为alasa_{las}alas。当前约束处是iii,限制为aia_iai

这样处理后可以发现出现一个LLL形区域可以选点,这部分需要选v=ai−alasv=a_i-a_{las}v=aialas个点,

如图,w1=i−lasw_1=i-lasw1=ilash1=las−alash_1=las-a_{las}h1=lasalash2=i−lash_2=i-lash2=ilasw2=i−alasw_2=i-a_{las}w2=ialas

枚举w1×h1w_1 \times h_1w1×h1这个矩阵中,选了kkk个点,这kkk个点选了之后,同行同列不能再选点。

那么对于下面的部分就还剩h2×(w2−k)h_2 \times (w_2-k)h2×(w2k)这个矩阵,需要选剩下的v−kv-kvk个点,可以从000枚举到ai−alasa_i-a_{las}aialas枚举这个kkk,因为邻相差之和最终等于nnn,所以复杂度是O(n)O(n)O(n)

对于一个h×wh \times wh×w的矩阵,在其中选择kkk个点的方案是:C(h,k)×C(w,k)×k!C(h,k) \times C(w,k) \times k!C(h,k)×C(w,k)×k!,即先选择kkk个横坐标在哪,再选择kkk个纵坐标在哪,再将纵坐标顺序打乱拼到横坐标上。

代码:

#include <bits/stdc++.h>

using namespace std;
typedef long long LL;

const int MAXN = 2e5 + 5;
const LL MOD = 998244353;

inline LL qpow(LL b, LL p) {
    LL res = 1;
    while (p) {
        if (p & 1) {
            res = res * b % MOD;
        }
        b = b * b % MOD;
        p >>= 1;
    }
    return res;
}

LL n, fac[MAXN], a[MAXN], ifac[MAXN];

LL A(LL n, LL m) {
    if (n < m || n < 0 || m < 0) {
        return 0;
    } else {
        return fac[n] * ifac[n - m] % MOD;
    }
}

LL C(LL n, LL m) {
    if (n < m || n < 0 || m < 0) {
        return 0;
    } else {
        return fac[n] * ifac[m] % MOD * ifac[n - m] % MOD;
    }
}

LL sum1(LL n, LL m) {
    return C(n, m) * A(n, m) % MOD;
}

void solve() {
    cin >> n;
    for (int i = 1; i <= n; ++i) {
        cin >> a[i];
    }
    fac[0] = 1;
    for (int i = 1; i <= n; ++i) {
        fac[i] = fac[i - 1] * i % MOD;
    }
    ifac[n] = qpow(fac[n], MOD - 2);
    for (int i = n - 1; ~i; --i) {
        ifac[i] = ifac[i + 1] * (i + 1) % MOD;
    }
    a[0] = 0;
    if (a[n] != -1 && a[n] != n) {
        cout << "0" << endl;
        return;
    }
    a[n] = n;
    int j = 0;
    LL ans = 1;
    for (int i = 1; i <= n; ++i) {
        if (a[i] != -1) {
            int t = a[i] - a[j], x = j - a[j], y = i - a[j];
            if (t < 0) {
                cout << "0" << endl;
                return;
            }
            LL res = 0;
            for (int i = 0; i <= x && i <= t; ++i) {
                res = (res + ((i & 1) ? MOD - 1 : 1) * sum1(x, i) % MOD * sum1(y - i, t - i)) % MOD;
            }
            ans = ans * res % MOD;
            j = i;
        }
    }
    cout << ans << endl;
}

int main() {
    int T;
    cin >> T;
    while (T--) {
        solve();
    }
    return 0;
}

学习交流


以下为学习交流QQ群,群号: 546235402,每周题解完成后都会转发到群中,大家可以加群一起交流做题思路,分享做题技巧,欢迎大家的加入。

Codeforces Round 1036 是一场同时面向 Div.1Div.2 参赛者的比赛,通常这类比赛会包含多个具有挑战性的编程题目,涵盖算法、数据结构、数学等多个领域。比赛的题解和题目信息可以帮助参赛者回顾解题思路,提升编程能力。 ### 比赛基本信息 - **比赛名称**:Codeforces Round #1036 (Div. 1 and Div. 2) - **比赛时间**:具体时间为 UTC+X(根据实际举办日期和时间表) - **比赛链接**:[Codeforces 官方页面](https://codeforces.com/contest/1343) - **题解发布位置**:通常在比赛结束后不久,官方或社区成员会在 Codeforces 博客、GitHub 或其他技术平台上发布题解。 ### 题目类型与难度分布 该轮比赛通常包括 5 到 7 道题目,难度从简单实现到复杂算法不等。例如: - **A题**:通常是简单的模拟或数学问题。 - **B题**:可能涉及字符串处理或基础贪心策略。 - **C题**:中等难度,可能需要掌握基本的数据结构如数组、排序等。 - **D题及以后**:较高难度,可能涉及图论、动态规划、数论等高级算法。 ### 参赛情况与亮点 - **参与人数**:通常超过 10,000 名选手参加。 - **热门话题**:比赛中某些题目可能会引发广泛讨论,尤其是那些需要用到巧妙构造或优化技巧的问题。 - **知名选手表现**:顶尖选手如 tourist、Um_nik 等通常会以极快的速度完成所有题目,并占据排行榜前列。 ### 示例代码片段 以下是一个典型的 Codeforces 题目解法示例,适用于某道中等难度题目: ```cpp #include <bits/stdc++.h> using namespace std; int main() { int t; cin >> t; while(t--) { long long l, r; cin >> l >> r; // 假设 e 是一个预处理好的符合条件的数组 // 使用二分查找来统计区间 [l, r] 内的有效数字个数 long long ans = upper_bound(e.begin(), e.end(), r) - lower_bound(e.begin(), e.end(), l); cout << ans << endl; } return 0; } ``` ### 题解资源推荐 - **Codeforces 官方博客**:通常会有详细的题解和作者说明。 - **GitHub 仓库**:许多参赛者会将自己的解法上传至 GitHub,便于他人学习。 - **知乎专栏 / 优快云 / 博客园**:中文社区中也常有高质量的赛后总结与分析文章。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符  | 博主筛选后可见
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值