結果

問題 No.1018 suffixsuffixsuffix
ユーザー tempura_pptempura_pp
提出日時 2020-04-03 23:01:23
言語 C++17
(gcc 12.3.0 + boost 1.83.0)
結果
WA  
実行時間 -
コード長 3,665 bytes
コンパイル時間 1,607 ms
コンパイル使用メモリ 130,348 KB
実行使用メモリ 7,112 KB
最終ジャッジ日時 2023-09-16 04:15:32
合計ジャッジ時間 6,368 ms
ジャッジサーバーID
(参考情報)
judge14 / judge12
このコードへのチャレンジ(β)

テストケース

テストケース表示
入力 結果 実行時間
実行使用メモリ
testcase_00 AC 1 ms
4,380 KB
testcase_01 AC 1 ms
4,380 KB
testcase_02 AC 1 ms
4,376 KB
testcase_03 AC 1 ms
4,380 KB
testcase_04 AC 2 ms
4,380 KB
testcase_05 AC 2 ms
4,376 KB
testcase_06 WA -
testcase_07 WA -
testcase_08 AC 2 ms
4,380 KB
testcase_09 AC 42 ms
5,116 KB
testcase_10 AC 44 ms
5,144 KB
testcase_11 AC 48 ms
5,120 KB
testcase_12 AC 45 ms
5,104 KB
testcase_13 AC 47 ms
5,192 KB
testcase_14 AC 80 ms
6,696 KB
testcase_15 WA -
testcase_16 WA -
testcase_17 WA -
testcase_18 AC 92 ms
6,740 KB
testcase_19 AC 19 ms
4,380 KB
testcase_20 AC 19 ms
4,376 KB
testcase_21 AC 19 ms
4,384 KB
testcase_22 AC 19 ms
4,380 KB
testcase_23 AC 20 ms
4,376 KB
testcase_24 AC 86 ms
6,704 KB
testcase_25 AC 100 ms
7,024 KB
testcase_26 AC 92 ms
6,720 KB
testcase_27 AC 99 ms
6,704 KB
testcase_28 AC 98 ms
6,704 KB
testcase_29 AC 25 ms
4,380 KB
testcase_30 AC 25 ms
4,376 KB
testcase_31 AC 24 ms
4,380 KB
testcase_32 AC 25 ms
4,376 KB
testcase_33 AC 24 ms
4,376 KB
testcase_34 AC 2 ms
4,384 KB
testcase_35 AC 20 ms
4,380 KB
testcase_36 AC 25 ms
4,380 KB
testcase_37 AC 107 ms
7,028 KB
権限があれば一括ダウンロードができます

ソースコード

diff #

#include<iostream>
#include<string>
#include<algorithm>
#include<vector>
#include<iomanip>
#include<math.h>
#include<complex>
#include<queue>
#include<deque>
#include<stack>
#include<map>
#include<set>
#include<bitset>
#include<functional>
#include<assert.h>
#include<numeric>
using namespace std;
#define REP(i,m,n) for(int i=(int)(m) ; i < (int) (n) ; ++i )
#define rep(i,n) REP(i,0,n)
using ll = long long;
constexpr int inf=1e9+7;
constexpr ll longinf=1LL<<60 ;
constexpr ll mod=1e9+7 ;

struct SuffixArray {
  vector< int > SA;
  const string s;

  SuffixArray(const string &str) : s(str) {
    SA.resize(s.size());
    iota(begin(SA), end(SA), 0);
    sort(begin(SA), end(SA), [&](int a, int b) {
      return s[a] == s[b] ? a > b : s[a] < s[b];
    });
    vector< int > classes(s.size()), c(s.begin(), s.end()), cnt(s.size());
    for(int len = 1; len < s.size(); len <<= 1) {
      for(int i = 0; i < s.size(); i++) {
        if(i > 0 && c[SA[i - 1]] == c[SA[i]] && SA[i - 1] + len < s.size() && c[SA[i - 1] + len / 2] == c[SA[i] + len / 2]) {
          classes[SA[i]] = classes[SA[i - 1]];
        } else {
          classes[SA[i]] = i;
        }
      }
      iota(begin(cnt), end(cnt), 0);
      copy(begin(SA), end(SA), begin(c));
      for(int i = 0; i < s.size(); i++) {
        int s1 = c[i] - len;
        if(s1 >= 0) SA[cnt[classes[s1]]++] = s1;
      }
      classes.swap(c);
    }
  }

  int operator[](int k) const {
    return SA[k];
  }

  size_t size() const {
    return s.size();
  }

  bool lt_substr(const string &t, int si = 0, int ti = 0) {
    int sn = (int) s.size(), tn = (int) t.size();
    while(si < sn && ti < tn) {
      if(s[si] < t[ti]) return true;
      if(s[si] > t[ti]) return false;
      ++si, ++ti;
    }
    return si >= sn && ti < tn;
  }

  int lower_bound(const string &t) {
    int low = -1, high = (int) SA.size();
    while(high - low > 1) {
      int mid = (low + high) / 2;
      if(lt_substr(t, SA[mid])) low = mid;
      else high = mid;
    }
    return high;
  }

  pair< int, int > lower_upper_bound(string &t) {
    int idx = lower_bound(t);
    int low = idx - 1, high = (int) SA.size();
    t.back()++;
    while(high - low > 1) {
      int mid = (low + high) / 2;
      if(lt_substr(t, SA[mid])) low = mid;
      else high = mid;
    }
    t.back()--;
    return {idx, high};
  }

  void output() {
    for(int i = 0; i < size(); i++) {
      cout << i << ": " << s.substr(SA[i]) << endl;
    }
  }
};


int main(){
    cin.tie(nullptr);
    ios::sync_with_stdio(false);
    ll n,k,q;
    cin>>n>>k>>q;
    string S;
    cin>>S;
    vector<int> A(n);
    A[0] = S.size();
    int i = 1, j = 0;
    while (i < S.size()) {
        while (i+j < S.size() && S[j] == S[i+j]) ++j;
        A[i] = j;
        if (j == 0) { ++i; continue;}
        int k = 1;
        while (i+k < S.size() && k+A[k] < j) A[i+k] = A[k], ++k;
        i += k; j -= k;
    }
    REP(i,1,n){
        if(A[i]==n-i && n%i==0){
            k*=n/i;
            n=i;
            S.resize(i);
            break;
        }
    }
    if(k>=2)S = S + S;
    SuffixArray sa(S);
    int cur = 0;
    ll sum = 0;
    vector<ll> ans(q);
    if(k==1){
        rep(i,q){
            ll t;
            cin>>t;
            ans[i]=sa[t-1];
        }
    }
    else {
        rep(i,q){
            ll t;
            cin>>t;
            while(1){
                ll x = sa[cur]>=n?1:k-1;
                if(t<=sum+x)break;
                sum += x;
                ++cur;
            }
            if(sa[cur]<n)ans[i]= sa[cur] + n*(sum-t+k-1);
            else ans[i]=sa[cur];
        }
    }
    rep(i,q)cout<<ans[i]+1<<" \n"[i+1==q];
    return 0;
}
0