首页 > 其他分享 >P5329 [SNOI2019] 字符串 题解

P5329 [SNOI2019] 字符串 题解

时间:2024-09-25 10:24:38浏览次数:14  
标签:kMod return int 题解 P5329 a1 b1 hs SNOI2019

Description

给出一个长度为 \(n\) 的由小写字母组成的字符串 \(a\),设其中第 \(i\) 个字符为 \(a_i\ (1\leq i\leq n)\)。

设删掉第 \(i\) 个字符之后得到的字符串为 \(s_i\),请按照字典序对 \(s_1,s_2,……,s_n\) 从小到大排序。若两个字符串相等,则认为编号小的字符串字典序更小。

Solution

容易发现对于一个极长的字符相等的连续段删掉任何一个字符都是一样的,所以先把这些连续段缩掉。

然后从前往后扫,如果 \(s_i<s_{i+1}\),则 \(i\) 一定比 \([i+1,n]\) 的所有数字典序大,就放到末尾。否则 \(i\) 一定比 \([i+1,n]\) 的所有数字典序小,放到开头。

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

Code

#include <bits/stdc++.h>

// #define int int64_t

using u64 = uint64_t;

const int kMaxN = 1e5 + 5, kMaxL = 1e6 + 5, kMod = 1e9 + 7;

int n;
int len[kMaxL], sum[kMaxL];
u64 pw[kMaxL];
std::string s[kMaxN];
std::vector<int> id[kMaxN], f[kMaxN];
std::vector<u64> hs[kMaxN];

inline int add(int x, int y) { return (x + y >= kMod ? x + y - kMod : x + y); }
inline int sub(int x, int y) { return (x >= y ? x - y : x - y + kMod); }
inline void inc(int &x, int y) { (x += y) >= kMod ? x -= kMod : x; }
inline void dec(int &x, int y) { (x -= y) < 0 ? x += kMod : x; }

void prework() {
  pw[0] = 1;
  for (int i = 1; i <= 1e6; ++i) pw[i] = 13331ull * pw[i - 1];
  for (int i = 1; i <= n; ++i) {
    hs[i].resize(len[i] + 1);
    for (int j = 1; j <= len[i]; ++j)
      hs[i][j] = 13331ull * hs[i][j - 1] + s[i][j];
  }
}

int getpos(int x, int p) { // 去掉 p 后第 x 个字符的位置
  if (x < p) return x;
  else return x + 1;
}

u64 gethash(std::vector<u64> &hs, int l, int r) {
  return hs[r] - hs[l - 1] * pw[r - l + 1];
}

u64 gethash(std::vector<u64> &hs, int l, int r, int p) { // [l, r] 去掉 p 的哈希值
  if (p < l || p > r) return gethash(hs, l, r);
  else return gethash(hs, l, p - 1) * pw[r - p] + gethash(hs, p + 1, r);
}

bool cmp(int a1, int b1, int a2, int b2) { // s[a1] 去掉 b1 位置的字符是否 <= s[a2] 去掉 b2 位置的字符
  int L = 0, R = std::min(len[a1] - 1, len[a2] - 1) + 1, res = 0;
  while (L + 1 < R) {
    int mid = (L + R) >> 1;
    if (gethash(hs[a1], 1, getpos(mid, b1), b1) == gethash(hs[a2], 1, getpos(mid, b2), b2)) L = res = mid;
    else R = mid;
  }
  if (res == std::min(len[a1] - 1, len[a2] - 1)) {
    if (res == len[a1] - 1) return 1;
    else return s[a1][getpos(res + 1, b1)] == '.';
  } else {
    return s[a1][getpos(res + 1, b1)] <= s[a2][getpos(res + 1, b2)];
  }
}

void dickdreamer() {
  std::cin >> n;
  for (int i = 1; i <= n; ++i) {
    std::cin >> s[i];
    s[i] = " " + s[i] + ".";
    len[i] = (int)s[i].size() - 1;
  }
  prework();
  for (int i = 1; i <= n; ++i) {
    id[i].resize(len[i] + 1);
    int l = 1, r = len[i], lst = 0;
    for (int j = 1; j <= len[i] - 1; ++j) {
      if (s[i][j] < s[i][j + 1]) {
        for (int k = j; k >= lst + 1; --k) id[i][r--] = k;
        lst = j;
      } else if (s[i][j] > s[i][j + 1]) {
        for (int k = lst + 1; k <= j; ++k) id[i][l++] = k;
        lst = j;
      }
    }
    for (int j = lst + 1; j <= len[i]; ++j) id[i][l++] = j;
  }
  f[1].resize(len[1] + 1);
  for (int i = 1; i <= len[1]; ++i) f[1][i] = 1;
  for (int i = 2; i <= n; ++i) {
    f[i].resize(len[i] + 1);
    for (int j = 1; j <= len[i - 1]; ++j) sum[j] = add(sum[j - 1], f[i - 1][j]);
    int p = 0;
    for (int j = 1; j <= len[i]; ++j) {
      for (; p < len[i - 1] && cmp(i - 1, id[i - 1][p + 1], i, id[i][j]); ++p) {}
      f[i][j] = sum[p];
    }
  }
  int ans = 0;
  for (int i = 1; i <= len[n]; ++i) inc(ans, f[n][i]);
  std::cout << ans << '\n';
}

int32_t main() {
#ifdef ORZXKR
  freopen("in.txt", "r", stdin);
  freopen("out.txt", "w", stdout);
#endif
  std::ios::sync_with_stdio(0), std::cin.tie(0), std::cout.tie(0);
  int T = 1;
  // std::cin >> T;
  while (T--) dickdreamer();
  // std::cerr << 1.0 * clock() / CLOCKS_PER_SEC << "s\n";
  return 0;
}

标签:kMod,return,int,题解,P5329,a1,b1,hs,SNOI2019
From: https://www.cnblogs.com/Scarab/p/18430804

相关文章

  • CF164D Minimum Diameter 题解
    最小点覆盖模板题。思路考虑二分直径\(x\)。我们将距离\(>x\)的点对连一条边,那么每一条边的两端至少有一端需要被删掉。这是最小点覆盖的定义。那么就是判断最小点覆盖是否小于等于\(k\)。发现这个问题并不好用一些多项式复杂度的做法解决。考虑暴搜。每一次我们把度......
  • 20240924 模拟赛 T4 题解
    Description这是一道交互题。有一棵\(n\)个节点的树,现在要求你通过若干次询问得到这棵树的每一条边连接哪两个点。每次询问你需要指定\(n\)个整数\(d_1,d_2,\ldots,d_n\),满足\(-1\leqd_i\leqn\),其中\(1\leqi\leqn\)。每次询问交互库会返回给你一个长度为\(n\)的......
  • P4780 Phi的反函数 题解
    因为\(\varphi(x)\)的值只与\(x\)的不同质因子有关,又因为\(2^{31}\)之内的数的质因子个数不超过\(10\),所以容易枚举\(10\)个位置上填入的质因子打出朴素的暴力,简单剪枝后得到\(20\)分。注意需要先判掉\(x=n+1\)的情况。考虑优化:因为\(\varphi\)的值只与质因......
  • Codeforces Round 959 (Div. 1 + Div. 2) C. Hungry Games题解
    CodeforcesRound959(Div.1+Div.2)C.HungryGames题解题目链接大致题意:给定一个长度为n的数组,并且给出一对l,r表示一个区间,如果∑i......
  • 力扣题解2207
    大家好,欢迎来到无限大的频道。今日继续给大家带来力扣题解。题目描述(中等)​:字符串中最多数目的子序列给你一个下标从 0 开始的字符串 text 和另一个下标从 0 开始且长度为 2 的字符串 pattern ,两者都只包含小写英文字母。你可以在 text 中任意位置插入 一个......
  • 【洛谷】P2261 [CQOI2007] 余数求和 的题解
    【洛谷】P2261[CQOI2007]余数求和的题解洛谷传送门题解这还是蓝题,这还是省选qaq题目看着很简单,但是真的很考验思路,思路对了,代码不到555分钟写完。刚开始做的时......
  • 题解:SP1741 TETRIS3D - Tetris 3D
    题意维护一个\(D\timesS\)的平面,每个点有一个高度。要求支持一个操作:查询一个矩形区域的最大值,并将该区域更新为最大值加上给定的数。分析发现\(D,S\leq10^3\),考虑使用二维线段树维护。二维线段树,顾名思义,就是在普通线段树的每一个节点上维护一棵线段树。在本题中,外层节......
  • 题解:P10950 太鼓达人
    分析显然答案包含长度为\(K\)的所有\(01\)串,每个串和前一个的重叠长度为\(K-1\),所以每个串对长度的贡献为\(1\)。因此该串的长度为所有\(01\)串的个数,即\(2^K\)。考虑第二个如何解决。发现每个位置的状态只有\(0\)和\(1\),考虑爆搜。显然直接搜的复杂度为\(O(2^......