-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathrolling_hash.cpp
More file actions
116 lines (101 loc) · 2.66 KB
/
rolling_hash.cpp
File metadata and controls
116 lines (101 loc) · 2.66 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
template<typename T_string = string>
struct string_hash{
int N;
string s;
const int p = 31; // lowercase
//const int p = 53; // lower + upper
const int start = 'a';
const int64_t MOD = 1e9 + 9;
vector<int64_t> hsh;
vector<int64_t> rev;
string_hash(T_string str){
N = str.size();
hsh.resize(N);
rev.resize(N);
s = str;
ll yet = 0;
ll pp = 1;
for(int i = 0; i < N; i++){
(yet += pp * (s[i] - start + 1)) %= MOD;
(pp *= p) %= MOD;
hsh[i] = yet;
}
pp = 1;
yet = 0;
for(int i = N-1; i >= 0; i--){
(yet += pp * (s[i] - start + 1)) %= MOD;
(pp *= p) %= MOD;
rev[i] = yet;
}
}
int64_t find_hash(int l,int r){
if(l <= r){
assert(l >= 0 && r < N);
int64_t res = ((hsh[r] - (l == 0 ? 0 : hsh[l-1])) * inv(power(p,l))) % MOD;
return res < 0 ? res + MOD : res;
}
else{
swap(l,r);
assert(l >= 0 && r < N);
int64_t res = ((rev[l] - (r == N-1 ? 0 : rev[r+1])) * inv(power(p,N-r-1))) % MOD;
return res < 0 ? res + MOD : res;
}
}
int64_t find_hash_scaled(int l,int r){ // Does not divide, scales the string to N
if(l <= r){
assert(l >= 0 && r < N);
int64_t res = ((hsh[r] - (l == 0 ? 0 : hsh[l-1])) * power(p,N-r-1)) % MOD;
return res < 0 ? res + MOD : res;
}
else{
swap(l,r);
assert(l >= 0 && r < N);
int64_t res = ((rev[l] - (r == N-1 ? 0 : rev[r+1])) * power(p,l)) % MOD;
return res < 0 ? res + MOD : res;
}
}
int64_t power(int64_t x, int64_t y){
int64_t res = 1;
x = x % MOD;
if (x == 0) return 0;
while (y > 0){
if (y & 1)
res = (res*x) % MOD;
y = y >> 1;
x = (x*x) % MOD;
}
return res;
}
int64_t inv(int64_t x){
x %= MOD;
assert(x != 0);
return power(x,MOD-2);
}
int64_t divi(int64_t x,int64_t y){
x %= MOD;
y %= MOD;
return (x * inv(y)) % MOD;
}
bool palindrome(int l = 0,int r = -1){
if(r == -1){
r = N-1;
}
return find_hash_scaled(l,r) == find_hash_scaled(r,l);
}
};
template<typename T_string = string>
bool same(const string_hash<T_string> &v1, int l1,int r1, const string_hash<T_string> &v2,int l2,int r2){ // v1[l1, ... ,r1] == v2[l2 , ... ,r2]
if(max(l1,r1) - min(l1,r1) != max(l2,r2) - min(l2,r2)){
return false;
}
return v1.find_hash_scaled(l1,r1) == v2.find_hash_scaled(l2,r2);
}
template<typename T_string = string>
int64_t concat(string_hash<T_string> &v1, int l1,int r1, string_hash<T_string> &v2,int l2,int r2){ // concatenates v1[l1, ... ,r1] and v2[l2, ... ,r2]
assert(v1.MOD == v2.MOD);
assert(v1.p == v2.p);
int64_t res = v2.find_hash(l2,r2);
(res *= v1.power(v1.p,max(r1,l1)-min(r1,l1)+1)) %= v1.MOD;
return (res + v1.find_hash(l1,r1)) % v1.MOD;
}
// hash(s) = s[0] + p*s[1] + p^2*s[2] + ... + p^(n-1) s[n-1]