Codeforces Round 921 (Div. 2)

写在前面

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

在床上躺了两天终于复活了妈的。

A

发现字符串 \(s\) 合法当且仅当 \(s\) 可以被分为至少 \(n\),其中每段都包含 \(k\) 种字符至少 1 次

正确性可以归纳证明,这里懒得写了感性理解下。

于是只需构造:abc...abc...abc......

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
//=============================================================
//=============================================================
//=============================================================
int main() {
  //freopen("1.txt", "r", stdin);
  std::ios::sync_with_stdio(0), std::cin.tie(0);
  int T; std::cin >> T;
  while (T --) {
    int n, k; std::cin >> n >> k;
    for (int i = 1; i <= n; ++ i) {
      for (int j = 0; j < k; ++ j) {
        printf("%c", (char) 'a' + j);
      }
    }
    printf("\n");
  }
  return 0;
}
/*
abc cba abc
*/

B

典中典之枚举 \(\operatorname{gcd}\)

考虑枚举 \(x\) 的因数 \(d\),检查答案是否可以为 \(d\),即检查是否存在 \(\frac{x}{d}\ge n\)(可以被分为大于 \(n\) 份)即可。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
//=============================================================
//=============================================================
//=============================================================
int main() {
  //freopen("1.txt", "r", stdin);
  std::ios::sync_with_stdio(0), std::cin.tie(0);
  int T; std::cin >> T;
  while (T --) {
    int x, n, ans = 1; std::cin >> x >> n;
    for (int i = 1; i * i <= x; ++ i) {
      if (x % i != 0) continue;
      if (x / i >= n) ans = std::max(ans, i);
      if (x / (x / i) >= n) ans = std::max(ans, x / i);
    }
    std::cout << ans << "\n";
  }
  return 0;
}

C

A 题,何时又来了?

唏,可以和解吗?

套用 A 的结论:字符串 \(s\) 合法当且仅当 \(s\) 可以被分为至少 \(n\),其中每段都包含 \(k\) 种字符至少 1 次,在顺序枚举字符的同时检查当前一段中是否已经有了 \(k\) 种字符,若满足则分一段,最后检查是否分了不少于 \(n\) 段来判断字符串合法。

然后考虑如何找到在不合法的字符串里的一个未出现过的子序列。考虑在上述过程中某不合法字符串被分为了 \(m(m<n)\) 段出现了 \(k\) 种字符的以及最后一段未出现 \(k\) 种字符的,设这 \(m\) 段中最晚出现的字符为 \(c_i(1\le i\le m)\),最后一段中 \(c_{j}\) 没有出现,则可证明:\(t = c_1c_2\cdots c_m c_j (|t| \le n)\) 没有出现。

其正确性可以考虑反证+归纳:

  • \(m=0\),则 \(c_j\) 未在 \(s\) 中出现过,显然上述结论正确。
  • 否则若 \(t\)\(s\) 中出现过,说明结尾的 \(c_j\) 一定来自于第 \(m\) 段,且该位置一定在第 \(m\) 段的 \(c_m\) 之前,而该位置之前的部分只能被分为 \(m-1\) 段出现了 \(k\) 种字符的以及最后一段未出现 \(c_m\) 的,归纳可知无法表示出 \(t\) 的前缀 \(c_1c_2\cdots c_m\),反证上述结论正确。

要求输出长度为 \(n\) 的,则在 \(t\) 后面随便补什么东西就行。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
//=============================================================
//=============================================================
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  std::ios::sync_with_stdio(0), std::cin.tie(0);
  int T; std::cin >> T;
  while (T --) {
    int n, k, m; std::cin >> n >> k >> m;
    std::string s, t; std::cin >> s;

    int cnt = 0, cnt1 = 0, cnt2[30] = {0};
    for (int i = 0; i < m; ++ i) {
      ++ cnt2[s[i] - 'a'];
      if (cnt2[s[i] - 'a'] == 1) ++ cnt1;
      if (cnt1 == k) {
        t.push_back(s[i]);
        ++ cnt;
        cnt1 = 0;
        for (int j = 0; j < 26; ++ j) cnt2[j] = 0;
      }
    }

    if (cnt >= n) {
      std::cout << "YES\n";
    } else {
      std::cout << "NO\n";
      for (int i = 0; i < k; ++ i) {
        if (!cnt2[i]) {
          if (t.size() < n) t.push_back((char) (i + 'a'));
          while (t.size() < n) t.push_back('a');
          std::cout << t << "\n";
          break;
        }
      }
    } 
  }
  return 0;
}
/*
aabbcccc 

cab

ab

cccaabb
ccbba
*/

D

期望,懒了。

E

呃呃数据结构。

发现这若干个太空港实际上将 \(1\sim n\) 划分成了若干个区间,每个区间内部的贡献都是一个公差为左侧太空港的价值,项数为与右侧太空港距离的等差数列的形式。

这东西长得太分块了哈哈,考虑将询问拆成中间的完整区间与两端点所在的不完整区间的形式。端点位于的区间可以使用 set 求前驱后继得到,等差数列求和即得不完整区间贡献;中间的完整区间考虑预处理出其贡献并存到其左端点上,再用树状数组维护区间和即可。

修改操作相当于将一个完整区间拆成两个区间,发现被影响的位置仅有原来区间的左端点与当前新加入的位置,树状数组单点修改即可。

修改和查询均为 \(O(\log n)\) 级别,总时间复杂度 \(O((m+q)\log n)\) 级别。

代码中为了方便查询操作求不完整区间(主要是两端点在同一区间内的情况)的贡献,钦定了查询的端点位置上不能有太空港,若有则找到区间内最近的非太空港位置再查询,同样使用了 set 维护。

//
/*
By:Luckyblock
*/
#include <bits/stdc++.h>
#define LL long long
#define pii std::pair<int,int>
#define mp std::make_pair
const int kN = 3e5 + 10;
//=============================================================
int n, m, q, num;
int p[kN], v[kN], id[kN];
std::set <pii> h;
std::set <int> noth;
//=============================================================
namespace BIT {
  #define lowbit(x) ((x)&(-x))
  int lim;
  LL s[kN];
  void Init(int lim_) {
    lim = lim_;
  }
  void Insert(int pos_, LL val_) {
    for (int i = pos_; i <= lim; i += lowbit(i)) {
      s[i] += val_;
    }
  }
  LL Sum(int pos_) {
    LL ret = 0;
    for (int i = pos_; i; i -= lowbit(i)) {
      ret += s[i];
    }
    return ret;
  }
  LL Query(int L_, int R_) {
    if (L_ > R_) return 0;
    return Sum(R_) - Sum(L_ - 1);
  }
  
  #undef lowbit
}
LL Calc(LL val_, LL dis_) {
  return 1ll * (dis_ + 1ll) * dis_ / 2ll * val_;
}
void Init() {
  std::cin >> n >> m >> q;
  BIT::Init(n);
  for (int i = 0; i <= n + 1; ++ i) noth.insert(i);

  for (int i = 1; i <= m; ++ i) {
    std::cin >> p[++ num];
    id[p[i]] = i;
    h.insert(mp(p[i], i)), noth.erase(p[i]);
  }
  for (int i = 1; i <= m; ++ i) std::cin >> v[i];
  for (int i = 1; i <= m; ++ i) {
    std::set <pii>::iterator it = h.lower_bound(mp(p[i], i));
    int pre = (-- it)->second;
    if (p[i] > 1) BIT::Insert(p[pre], Calc(v[pre], p[i] - p[pre] - 1));
  }
}
void Modify(int p_, int v_) {
  std::set <pii>::iterator it = h.lower_bound(mp(p_, num));
  int suf = it->second, pre = (-- it)->second;

  BIT::Insert(p[pre], Calc(v[pre], p_ - p[pre] - 1) - BIT::Query(p[pre], p[pre]));
  BIT::Insert(p_, Calc(v_, p[suf] - p_ - 1));
  
  p[++ num] = p_, v[num] = v_, id[p_] = num;
  h.insert(mp(p_, num)), noth.erase(p_);
}
LL Query(int l_, int r_) {
  if (id[l_]) l_ = *noth.lower_bound(l_);
  if (id[r_]) {
    std::set <int>::iterator it = noth.lower_bound(r_);
    r_ = *(-- it);
  }
  if (l_ > r_) return 0ll;

  int prel, prer, sufl, sufr;
  std::set <pii>::iterator it = h.lower_bound(mp(l_, num));
  sufl = it->second, prel = (-- it)->second;
  it = h.lower_bound(mp(r_, num));
  sufr = it->second, prer = (-- it)->second;
  
  if (prel == prer) {
    return Calc(v[prel], p[sufl] - l_) - Calc(v[prer], p[sufr] - (r_ + 1));
  }
  LL ret = BIT::Query(p[sufl], p[prer] - 1);
  ret += Calc(v[prel], p[sufl] - l_);
  ret += Calc(v[prer], p[sufr] - p[prer] - 1) - Calc(v[prer], p[sufr] - (r_ + 1));
  return ret;
}
//=============================================================
int main() {
  // freopen("1.txt", "r", stdin);
  std::ios::sync_with_stdio(0), std::cin.tie(0);
  Init();
  while (q --) {
    int op, x, y; std::cin >> op >> x >> y;
    if (op == 1) Modify(x, y);
    else std::cout << Query(x, y) << "\n";
  }
  return 0;
}

写在最后

学到了什么:

  • E:贡献为不相交区间形式,且修改等价于将某区间一分为二,考虑整区间与散区间的贡献。
posted @ 2024-01-29 16:38  Luckyblock  阅读(28)  评论(0编辑  收藏  举报