Codeforces Round 900 (Div. 3)

写在前面

比赛地址:https://codeforces.com/contest/1878

前天晚上他妈睡不着觉又不想看漫画打游戏于是到阳台上开把 div3 放松心情。

40min 过了 5 题把剩下两题都口了感觉没意思了于是睡觉。

太菜了还是,现在是 div3 随便 AK 但是 div2 过不了 E 的憋屈水平。

这场全是典中典,随便写写。

A

检查 \(k\) 是否出现即可。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
//=============================================================
//=============================================================
inline int read() {
  int f = 1, w = 0; char ch = getchar();
  for (; !isdigit(ch); ch = getchar()) if (ch == '-') f = -1;
  for (; isdigit(ch); ch = getchar()) w = (w << 3) + (w << 1) + (ch ^ '0'); 
  return f * w;
}
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  int T = read();
  while (T --) {
    int n = read(), k = read(), flag = 0;
    for (int i = 1; i <= n; ++ i) {
      int a = read();
      if (a == k) flag = 1;
    }
    printf("%s\n", flag ? "YES" : "NO");
  }
  return 0;
}

B

一个暴力的想法是令 \(a_1 = 2, a_2 = 3\),对于 \(a_i\) 依次检查令 \(a_i=a_{i-1}+1, 2,\dots\) 是否合法即可。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
const int kN = 2e5 + 10;
//=============================================================
LL a[kN];
//=============================================================
inline int read() {
  int f = 1, w = 0; char ch = getchar();
  for (; !isdigit(ch); ch = getchar()) if (ch == '-') f = -1;
  for (; isdigit(ch); ch = getchar()) w = (w << 3) + (w << 1) + (ch ^ '0'); 
  return f * w;
}
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  a[1] = 2, a[2] = 3;
  for (int i = 3; i < kN; ++ i) {
    a[i] = a[i - 1] + 1;
    while (3 * a[i] % (a[i - 1] + a[i - 2]) == 0) ++ a[i];
  }

  int T = read();
  while (T --) {
    int n = read();
    for (int i = 1; i <= n; ++ i) printf("%lld ", a[i]);
    printf("\n");
  }
  return 0;
}
/*
2 3 4 5 7 8 9 10
*/

C

\([1, n]\) 中选 \(k\) 个数之和的范围为 \([\sum_{1\le i\le k} i, \sum_{0\le i< k} n - i]\),判断 \(x\) 是否在区间内即可。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
const int kN = 2e5 + 10;
//=============================================================
LL sum[kN];
//=============================================================
inline LL read() {
  LL f = 1, w = 0; char ch = getchar();
  for (; !isdigit(ch); ch = getchar()) if (ch == '-') f = -1;
  for (; isdigit(ch); ch = getchar()) w = (w << 3ll) + (w << 1ll) + (ch ^ '0'); 
  return f * w;
}
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  for (int i = 1; i < kN; ++ i) sum[i] = sum[i - 1] + i;
  
  int T = read();
  while (T --) {
    int n = read(), k = read(); LL x = read();
    LL minx = sum[k], maxx = sum[n] - sum[n - k];
    printf("%s\n", minx <= x && x <= maxx ? "YES" : "NO");
  }
  return 0;
}
/*
1 2 3 4 5
*/

D

题面很长但是全是屁话。

发现区间 \([l_i, r_i]\) 互不重合且恰好覆盖整个字符串,操作 \(x\) 实际上是选择了 \(x\) 所在的区间 \([l_i, r_i] (l_i\le x\le r_i)\),并将其中一个关于区间中心 \(\frac{l_i + r_i}{2}\) 对称的字符串进行了翻转。手玩下发现操作顺序和最终的字符串形态无关,且最终形态中某个位置 \(k\) 的字符,要么是原字符 \(s_k\),要么是与 \(s_k\) 所在区间 \([l, r]\) 中心 \(\frac{l+r}{2}\) 对称的字符 \(s_{l+r - k}\)

然后很容易发现某字符的取值,取决于这个位置被区间翻转操作覆盖次数的奇偶性,差分维护区间修改次数即可。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
const int kN = 2e5 + 10;
//=============================================================
int n, k, q, l[kN], r[kN];
char s[kN], t[kN], map[kN];
int d[kN];
//=============================================================
inline int read() {
  int f = 1, w = 0; char ch = getchar();
  for (; !isdigit(ch); ch = getchar()) if (ch == '-') f = -1;
  for (; isdigit(ch); ch = getchar()) w = (w << 3) + (w << 1) + (ch ^ '0'); 
  return f * w;
}
void Init() {
  n = read(), k = read();
  for (int i = 1; i <= n; ++ i) d[i] = 0;

  scanf("%s", s + 1);
  for (int i = 1; i <= k; ++ i) l[i] = read();
  for (int i = 1; i <= k; ++ i) r[i] = read();
  
  for (int i = 1; i <= k; ++ i) {
    for (int j = l[i]; j <= r[i]; ++ j) {
      map[j] = s[l[i] + r[i] - j];
    }
  }

  q = read();
  while (q --) {
    int x = read(), i = std::lower_bound(r + 1, r + k + 1, x) - r;
    int pl = std::min(x, l[i] + r[i] - x), pr = std::max(x, l[i] + r[i] - x);
    d[pl] ++, d[pr + 1] --;
  }
}
void Solve() {
  t[n + 1] = '\0';
  for (int i = 1; i <= n; ++ i) {
    d[i] += d[i - 1];
    t[i] = d[i] % 2 ? map[i] : s[i];
  }
}
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  int T = read();
  while (T --) {
    Init();
    Solve();
    printf("%s\n", t + 1);
  }
  return 0;
}
/*
abcdef
fedcba
fbcdea
*/

E

典中典之拆位。

维护数组 \(f_{i, j}\) 表示数列前缀 \(1\sim i\) 中满足第 \(j\) 位上为 1 的数的个数,区间的与和即可 \(O(\log w)\) 地求得。

回答询问时二分即可。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
const int kN = 2e5 + 10;
//=============================================================
int n, a, cnt[kN][31];
//=============================================================
inline int read() {
  int f = 1, w = 0; char ch = getchar();
  for (; !isdigit(ch); ch = getchar()) if (ch == '-') f = -1;
  for (; isdigit(ch); ch = getchar()) w = (w << 3) + (w << 1) + (ch ^ '0'); 
  return f * w;
}
void Init() {
  n = read();
  for (int i = 1; i <= n; ++ i) {
    int a = read();
    for (int j = 0, k = 1; j < 31; ++ j, k <<= 1) {
      cnt[i][j] = cnt[i - 1][j];
      if (a & k) ++ cnt[i][j];
    }
  }
}
int Calc(int l_, int r_) {
  int ret = 0, len = r_ - l_ + 1;
  for (int i = 0, j = 1; i < 31; ++ i, j <<= 1) {
    if (cnt[r_][i] - cnt[l_ - 1][i] == len) {
      ret |= j;
    }
  }
  return ret;
}
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  int T = read();
  while (T --) {
    Init();
    int q = read();
    while (q --) {
      int L = read(), k = read(), ans = -1;
      for (int l = L, r = n; l <= r; ) {
        int mid = (l + r) >> 1;
        if (Calc(L, mid) >= k) {
          ans = mid;
          l = mid + 1;
        } else {
          r = mid - 1;
        }
      }
      printf("%d ", ans);
    }
    printf("\n");
  }
  return 0;
}

F

典中典之维护质因子。

对于操作 1 的询问:

  • 若有 \(d(n) = n\)\(a=1\) 即可。
  • 否则 \(d(n)<n\)。条件 \(\gcd(a, n)\) 容易满足,随便找一个 \(n\) 中不存在的质数的幂 \(p^c\) 即可;在此基础上对于条件 \(d(n\cdot a) = n\),有:\(d(n\cdot a) = d(n)\times (c+1)(c\in \mathbf{N}^+)\)。则当且仅当 \(d(n) \mid n\) 时存在合法的 \(a\)

两种情况可以合并,则对于操作 1 仅需检查修改后的 \(n\) 是否满足 \(d(n)\mid n\) 即可。

发现要在操作 1 中动态维护 \(d(n)\)。又发现有典中典之条件 \(n\le 10^6\)\(x\le 10^6\),即 \(n\) 的质因子不会超过 \(10^6\),于是考虑预处理 \(1\sim 10^6\) 中所有数的质因子,维护当前的 \(n\) 的质因子及其次数即可轻松维护 \(d(n)\)

对于操作 1 的询问,先对 \(d(n)\) 做质因数分解,再与 \(n\) 的质因子进行比较即可。而对于操作 2,每次暴力地枚举把所有质因子出现次数重置显然不可取,套路地记时间戳懒惰删除即可。

质数数量不是很大,总复杂度非常优秀。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
const int kN = 1e6 + 10;
//=============================================================
int n, q, nowt, num, tag[kN], cnt[kN];
std::vector <int> p, d[kN];
bool vis[kN];
//=============================================================
inline int read() {
  int f = 1, w = 0; char ch = getchar();
  for (; !isdigit(ch); ch = getchar()) if (ch == '-') f = -1;
  for (; isdigit(ch); ch = getchar()) w = (w << 3) + (w << 1) + (ch ^ '0'); 
  return f * w;
}
void Init() {
  for (int i = 2; i < kN; ++ i) {
    if (!vis[i]) p.push_back(i), d[i].push_back(i);
    for (int j = 2; i * j < kN; ++ j) {
      vis[i * j] = true;
      if (!vis[i]) d[i * j].push_back(i);
    }
  }
}
void Reset() {
  ++ nowt;
  num = 1;

  for (int i = 0, sz = d[n].size(), temp = n; i < sz; ++ i) {
    int x = d[n][i];
    if (tag[x] < nowt) cnt[x] = 0, tag[x] = nowt;
    while (temp % x == 0) ++ cnt[x], temp /= x;
    num *= cnt[x] + 1;
  }
}
bool Modify(int x_) {
  for (int i = 0, sz = d[x_].size(), temp = x_; i < sz; ++ i) {
    int x = d[x_][i];
    if (tag[x] < nowt) cnt[x] = 0, tag[x] = nowt;
   
    num /= cnt[x] + 1;
    while (temp % x == 0) ++ cnt[x], temp /= x;
    num *= cnt[x] + 1;
  }
  for (int i = 0, sz = p.size(), temp = num; i < sz; ++ i) {
    int x = p[i], c = 0;
    if (temp % x != 0) continue;
    if (temp < x) break;

    while (temp % x == 0) ++ c, temp /= x;
    if (tag[x] < nowt) cnt[x] = 0, tag[x] = nowt;
    if (c > cnt[x]) return 0;
  }
  return 1;
}
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  Init();

  int T = read();
  while (T --) {
    n = read(), q = read();
    Reset();
    while (q --) {
      int opt = read();
      if (opt == 1) printf("%s\n", Modify(read()) ? "YES" : "NO");
      else Reset();
    }
    printf("\n");
  }
  return 0;
}
/*
1
20 4
1 3
2
1 7
1 12

1 2 4 5 10 20
*/

G

不太典但是很恼弹的题。

\(h(u, v)\) 表示链 \((u,v)\) 上所有数的或和中 1 的数量。对于一次询问 \((x, y)\),考虑取答案的位置 \(z\) 在链上移动的影响。显然 \(z\) 越靠近 \(x\)\(h(x, z)\) 越小,\(h(y, z)\) 越大,反之亦然,存在单调性。于是有一个显然的想法是枚举 \(h(x, z)\) 的所有取值,对于每种取值找到令 \(z\) 最靠近 \(x\) 的位置最大化 \(h(y, z)\),对所有情况取最大值即可。

又发现 \(h(x, z)\) 至多只有 31 种取值,考虑在链 \((x, \operatorname{lca}(x, y))\) 上从 \(x\) 开始倍增地上跳 \(z\) 即可得到当 \(z\) 在链 \((x, \operatorname{lca}(x, y))\) 上时 \(h(x, z)\) 的所有取值,以及取到该值时最靠近 \(x\) 的位置。

\(z\) 在链 \((y, \operatorname{lca}(x, y))\) 上的时候怎么办?考虑换成枚举 \(h(y, z)\) 的取值,找到令 \(z\) 最靠近 \(y\) 的位置最大化 \(h(x, z)\) 即可。

倍增预处理一下链的或和即可。

对于每次询问要进行枚举取值、倍增、统计 1 的个数,总时间复杂度上界理论上是 \(O(n\log n + q\log n\log^2 w)\) 级别的,但是跑不满,给了 5S 跑了 1372 ms 。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
#define lowbit(x) ((x)&(-x))
const int kN = 2e5 + 10;
//=============================================================
int n, q, a[kN], c[kN];
int edgenum, head[kN], v[kN << 1], ne[kN << 1];
int fa[kN][21], g[kN][21], cnt[kN][21], dep[kN];
//=============================================================
inline int read() {
  int f = 1, w = 0; char ch = getchar();
  for (; !isdigit(ch); ch = getchar()) if (ch == '-') f = -1;
  for (; isdigit(ch); ch = getchar()) w = (w << 3) + (w << 1) + (ch ^ '0'); 
  return f * w;
}
void Add(int u_, int v_) {
  v[++ edgenum] = v_;
  ne[edgenum] = head[u_];
  head[u_] = edgenum;
}
int Calc(int val_) {
  int ret = 0;
  for (int i = val_; i; i -= lowbit(i)) ++ ret;
  return ret;
}
void Dfs(int u_, int fa_) {
  dep[u_] = dep[fa_] + 1;
  fa[u_][0] = fa_;
  g[u_][0] = a[u_] | a[fa_];

  for (int i = 1; i <= 20; ++ i) {
    fa[u_][i] = fa[fa[u_][i - 1]][i - 1];
    g[u_][i] = g[u_][i - 1] | g[fa[u_][i - 1]][i - 1];
  }
  for (int i = head[u_]; i; i = ne[i]) {
    int v_ = v[i];
    if (v_ == fa_) continue;
    Dfs(v_, u_);
  }
}
int Lca(int u_, int v_) {
  if (dep[u_] < dep[v_]) std::swap(u_, v_);
  for (int i = 20; i >= 0; -- i) {
    if (dep[fa[u_][i]] >= dep[v_]) {
      u_ = fa[u_][i];
    }
  }
  if (u_ == v_) return u_;

  for (int i = 20; i >= 0; -- i) {
    if (fa[u_][i] != fa[v_][i]) {
      u_ = fa[u_][i];
      v_ = fa[v_][i];
    }
  }
  return fa[u_][0];
}
void Init() {
  n = read();
  edgenum = 0;
  for (int i = 1; i <= n; ++ i) head[i] = 0;

  for (int i = 1; i <= n; ++ i) a[i] = read(), c[i] = Calc(a[i]);
  for (int i = 1; i < n; ++ i) {
    int u_ = read(), v_ = read();
    Add(u_, v_), Add(v_, u_);
  }
  Dfs(1, 0);
}
int Sum(int u_, int d_) {
  if (!d_) return a[u_];

  int ret = 0;
  for (int i = 20; i >= 0; -- i) {
    int step = (1 << i);
    if (step <= d_) {
      d_ -= step;
      ret |= g[u_][i];
      u_ = fa[u_][i];
    }
  }
  return ret;
}
int Query(int x_, int y_) {
  int lca = Lca(x_, y_), ans = 0;
  int vx = Sum(x_, dep[x_] - dep[lca]), vy = Sum(y_, dep[y_] - dep[lca]);

  int s1 = a[x_], s2 = vx | vy, c1 = Calc(s1), c2 = Calc(s2);
  for (int now_ = x_; ; ) {
    ans = std::max(ans, c1 + c2);
    for (int i = 20; i >= 0; -- i) {
      if (dep[fa[now_][i]] >= dep[lca] && Calc(s1 | g[now_][i]) == c1) {
        s1 |= g[now_][i];
        now_ = fa[now_][i];
      }
    }
    if (now_ == lca) break;
    s1 |= g[now_][0], c1 = Calc(s1);
    now_ = fa[now_][0];
    s2 = Sum(now_, dep[now_] - dep[lca]) | vy, c2 = Calc(s2);
  }

  s1 = vx | vy, s2 = a[y_], c1 = Calc(s1), c2 = Calc(s2);
  for (int now_ = y_; ; ) {
    ans = std::max(ans, c1 + c2);
    for (int i = 20; i >= 0; -- i) {
      if (dep[fa[now_][i]] >= dep[lca] && Calc(s2 | g[now_][i]) == c2) {
        s2 |= g[now_][i];
        now_ = fa[now_][i];
      }
    }
    if (now_ == lca) break;
    s2 |= g[now_][0], c2 = Calc(s2);
    now_ = fa[now_][0];
    s1 = Sum(now_, dep[now_] - dep[lca]) | vx, c1 = Calc(s1);
  }
  return ans;
}
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  int T = read();
  while (T --) {
    Init();
    q = read();
    while (q --) {
      int x_ = read(), y_ = read();
      printf("%d ", Query(x_, y_));
    }
    printf("\n");
  }
  return 0;
}
/*
1
7
4 7 7 4 10 8 10
6 1
3 1
2 1
7 4
1 5
4 2
4
7 5
2 3
4 5
2 5



6
9 5 6 2 4 6
5 1
2 1
1 6
4 3
1 3
4
6 1
1 4
4 3
3 5
7
5 1 3 7 5 1 6
2 1
5 4
2 3
3 4
7 6
6 3
2
4 2
7 7
*/

写在最后

学到了什么:

  • 典中典中典中典。
  • 呃呃好像什么都没学到。
  • 放松了心情。
  • 学到了打 div3 就是玩刷子游戏,以后摸鱼也打 div2 了。
posted @ 2023-09-30 16:12  Luckyblock  阅读(72)  评论(0编辑  收藏  举报