首页 > 其他分享 >Codeforces Round #829 (Div. 2) A-E

Codeforces Round #829 (Div. 2) A-E

时间:2022-10-25 20:11:05浏览次数:81  
标签:pos int 复杂度 Codeforces back 829 ans push Div

比赛链接

A

题解

知识点:枚举。

只要一个Q后面有一个A对应即可,从后往前遍历,记录A的数量,遇到Q则数量减一,如果某次Q计数为0则NO。

时间复杂度 \(O(n)\)

空间复杂度 \(O(1)\)

代码

#include <bits/stdc++.h>
#define ll long long

using namespace std;

bool solve() {
    int n;
    cin >> n;
    string s;
    cin >> s;
    s = "?" + s;
    int cnt = 0;
    for (int i = n;i >= 1;i--) {
        if (s[i] == 'Q') {
            if (cnt == 0) return false;
            cnt--;
        }
        else cnt++;
    }
    cout << "YES" << '\n';
    return true;
}

int main() {
    std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
    int t = 1;
    cin >> t;
    while (t--) {
        if (!solve()) cout << "NO" << '\n';
    }
    return 0;
}

B

题解

知识点:构造。

可以证明 \(\lfloor \frac{n}{2} \rfloor\) 是最优答案。交错构造, \(i+\lfloor \frac{n}{2} \rfloor\) 和 \(i\) ,注意 \(i\) 从 \(1\) 到 \(\lfloor \frac{n}{2} \rfloor\) ,在最后如果 \(n\) 是奇数则补一个 \(n\) 。

时间复杂度 \(O(n)\)

空间复杂度 \(O(1)\)

代码

#include <bits/stdc++.h>
#define ll long long

using namespace std;

bool solve() {
    int n;
    cin >> n;
    for (int i = 1;i <= n / 2;i++) {
        cout << i + n / 2 << ' ' << i << ' ';
    }
    if (n & 1) cout << n << ' ';
    cout << '\n';
    return true;
}

int main() {
    std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
    int t = 1;
    cin >> t;
    while (t--) {
        if (!solve()) cout << -1 << '\n';
    }
    return 0;
}

C

题解

知识点:构造。

可以两两构造。找到一对非 \(0\) 数 \(a[i],a[j]\) ,当 \(a[i] = a[j]\),如果 \(i,j\) 奇偶性相同则 \([i,i],[i+1,j]\) ,否则分段 \([i,j]\) ;当 \(a[i] \neq a[j]\) ,如果 \(i,j\) 奇偶性相同则 \([i,j]\) ,否则 \([i,i],[i+1,j]\) 。

注意两对之间以及首尾可能会存在空隙,最后要把上面答案遍历一遍,填补空隙。

时间复杂度 \(O(n)\)

空间复杂度 \(O(n)\)

代码

#include <bits/stdc++.h>
#define ll long long

using namespace std;

int a[200007];
bool solve() {
    int n;
    cin >> n;
    vector<int> pos;
    for (int i = 1;i <= n;i++) cin >> a[i];
    for (int i = 1;i <= n;i++) {
        if (a[i]) pos.push_back(i);
    }
    if (pos.size() & 1) return false;
    if (!pos.size()) {
        cout << 1 << '\n';
        cout << 1 << ' ' << n << '\n';
        return true;
    }
    vector<pair<int, int>> v;
    for (int i = 0;i < pos.size();i += 2) {
        if (a[pos[i]] == a[pos[i + 1]]) {
            if ((pos[i] & 1) == (pos[i + 1] & 1)) {
                v.push_back({ pos[i], pos[i] });
                v.push_back({ pos[i] + 1,pos[i + 1] });
            }
            else v.push_back({ pos[i],pos[i + 1] });
        }
        else {
            if ((pos[i] & 1) != (pos[i + 1] & 1)) {
                v.push_back({ pos[i], pos[i] });
                v.push_back({ pos[i] + 1,pos[i + 1] });
            }
            else v.push_back({ pos[i],pos[i + 1] });
        }
    }
    vector<pair<int, int>> ans;
    int prer = 0;
    for (auto [i, j] : v) {
        if (i != prer + 1) ans.push_back({ prer + 1, i - 1 });
        ans.push_back({ i,j });
        prer = j;
    }
    if (ans.back().second != n) ans.push_back({ ans.back().second + 1,n });
    cout << ans.size() << '\n';
    for (auto [i, j] : ans) {
        cout << i << ' ' << j << '\n';
    }
    return true;
}

int main() {
    std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
    int t = 1;
    cin >> t;
    while (t--) {
        if (!solve()) cout << -1 << '\n';
    }
    return 0;
}

D

题解

知识点:数论,贪心。

记录每个数字出现的次数,尝试从小到大合成出 \(x\) 。从 \(1\) 开始往后遍历,每次将 \(i\) 合成 \(i+1\) ,显然 \(i+1\) 个 \(i\) 将产生 \(1\) 个 \(i+1\) 。如果出现非 \(x\) 的数 \(i\) 不能全部使用 ,那么整个式子就无法被 \(x!\) 整除。

时间复杂度 \(O(n)\)

空间复杂度 \(O(1)\)

代码

#include <bits/stdc++.h>

using namespace std;

int cnt[500007];
int main() {
    std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
    int n, x;
    cin >> n >> x;
    for (int i = 1;i <= n;i++) {
        int tmp;
        cin >> tmp;
        cnt[tmp]++;
    }
    for (int i = 1;i < x;i++) {
        if (cnt[i] % (i + 1)) {
            cout << "NO" << '\n';
            return 0;
        }
        cnt[i + 1] += cnt[i] / (i + 1);
    }
    cout << "YES" << '\n';
    return 0;
}

E

题解

知识点:概率dp。

设 \(f[i]\) 代表将 \(i\) 个还没排好的 \(1\) (如 1100101 有 \(2\) 个 \(1\) 没排好)排好的期望步数。

对于 \(f[i]\) ,下一步排好一个 \(1\) (即到达 \(i-1\) 状态)的概率是 \(\dfrac{i^2}{C_n^2}\) ,下一步啥都没变的概率就是 \(1-\dfrac{i^2}{C_n^2}\),于是有:

\[\begin{aligned} f[i] &= (f[i-1]+1) \cdot \dfrac{i^2}{C_n^2} + (f[i]+1) \cdot (1-\dfrac{i^2}{C_n^2})\\ \dfrac{i^2}{C_n^2} \cdot f[i] &= \dfrac{i^2}{C_n^2} \cdot f[i-1] + 1\\ f[i] &= f[i-1] + \dfrac{C_n^2}{i^2} \end{aligned} \]

一步到达 \(i-1\) 后再排完的期望这步的概率一步啥也没干的期望这步的概率就是 \(f[i]\) 。

于是可以递推,\(f[0] = 0\) ,求的是 \(f[cnt1]\) ,\(cnt1\) 是初始没排好 \(1\) 的个数。

这里其实有个概率论的定理:如果一个事件的结果A发生的概率是 \(P\) ,则一直做这件事直到第一次发生结果A的期望 \(X\) 是 \(\dfrac{1}{P}\) 。

证明:

\[\begin{aligned} X &= 1\cdot P+(X+1)\cdot (1-P)\\ P\cdot X &= 1\\ X &= \frac{1}{P} \end{aligned} \]

时间复杂度 \(O(n\log n)\)

空间复杂度 \(O(n)\)

代码

#include <bits/stdc++.h>
#define ll long long

using namespace std;

const int mod = 998244353;

int a[200007];
ll qpow(ll a, ll k) {
    ll ans = 1;
    while (k) {
        if (k & 1) ans = (ans * a) % mod;
        k >>= 1;
        a = (a * a) % mod;
    }
    return ans;
}

bool solve() {
    int n;
    cin >> n;
    for (int i = 1;i <= n;i++) cin >> a[i];
    int cnt0 = count(a + 1, a + n + 1, 0);
    int cnt1 = count(a + 1, a + cnt0 + 1, 1);
    int c2 = 1LL * n * (n - 1) / 2 % mod;
    int ans = 0;
    for (int i = 1;i <= cnt1;i++) {
        ans = (ans + 1LL * c2 * qpow(1LL * i * i % mod, mod - 2) % mod) % mod;
    }
    cout << ans << '\n';
    return true;
}

int main() {
    std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
    int t = 1;
    cin >> t;
    while (t--) {
        if (!solve()) cout << -1 << '\n';
    }
    return 0;
}

标签:pos,int,复杂度,Codeforces,back,829,ans,push,Div
From: https://www.cnblogs.com/BlankYang/p/16826154.html

相关文章

  • Codeforces Round #830 (Div. 2) C1
    C1.Sheikh(Easyversion)显然对于添加一个数进入区间是只增不减的这就提醒了我们此题具有单调性我们最后的答案肯定就是这一整个区间我们考虑怎么找到最小的答案相......
  • Codeforces Round #756 (Div. 3) F
    F.ATMandStudents金典对于一个区间我们不能让他+的过程中出现负数我们ST表处理前缀和区间最小数然后再二分长度暴力枚举左右端点时间复杂度是O(nlogn)哦还要注意的......
  • D. Factorial Divisibility
    D.FactorialDivisibilitytimelimitpertest2secondsmemorylimitpertest256megabytesinputstandardinputoutputstandardoutputYouaregivenan......
  • jquery实现div可移动窗口、拖动改变窗口大小、最大化\还原、窗口置顶
    图例1、拖动标题来移动窗口(红框方案):其他样式,如:     2、最大化:  点击后最大化按钮变成还原按钮,可以还原窗口。  3、关闭  4、拖动改变窗......
  • CF838B Diverging Directions
    题目链接:​​传送门​​分析一下他给的这个图就知道一个祖先到它的儿子只有生成树的边可以走而一个儿子走到祖先有两种方法一个是先直接到1,再从1走到那个祖先还有一个很......
  • Educational Codeforces Round 138 (Rated for Div. 2)练习笔记
    \(\text{A.CowardlyRooks}\)有一张\(n\timesn(n\leq8)\)的国际象棋棋盘,上面放了\(m(m\leq8)\)个城堡(能攻击在同一直线的棋子),第\(i\)个城堡位于\((x_i,y_i)\)......
  • div背景设置
       border:先给div设置一个边框容易观察出大小background:给div插入背景图片用url(),当图片小,会默认把div铺满 (如图),设置no-repeat后则只有一张图片,不会自动铺满。然后......
  • Codeforces Round #829-1754A+B与1753A+B+C 题解
    1754A-TechnicalSupport题意给定一个只包含大写字母\(\texttt{Q}\)和\(\texttt{A}\)的字符串,如果字符串里的每一个\(\texttt{Q}\)都能与在其之后的\(\texttt{A......
  • div
      div定义:就是一个容器,可以整块移动1.创建一个div标签 2.对div标签进行格式设置 3.margin(边缘):先上下(xxpx)差不多300px居中;再左右 (xxpx)(auto=文本中的cent......
  • Codeforces 1674 E. Breaking the Wall
    题意给n个数的数列a[n],可以进行任意次操作,每次选取一个位置i,a[i]-=2,a[i-1]-=1,a[i+1]-=1,问最少几次操作可以让任意两个值<=0提示需要进行分类讨论,分成三种情况讨论1.......