Hashed String
Code
Hashed String
// Hashed String {{{
class HashedString {
static const int M = (1LL << 61) - 1;
static const int B;
static vector<int> pow;
int N;
vector<int> p_hash;
__int128 mul(int a, int b) { return (__int128)a * b; }
int mod_mul(int a, int b) { return mul(a, b) % M; }
public:
explicit HashedString(string const& s) {
N = size(s);
while (size(pow) < size(s) + 1) pow.push_back(mod_mul(pow.back(), B));
p_hash.resize(size(s) + 1);
p_hash[0] = 0;
for (int i = 0; i < size(s); i++)
p_hash[i + 1] = (mul(p_hash[i], B) + s[i]) % M;
}
int get_hash(int l, int r) {
int raw_val = p_hash[r + 1] - mod_mul(p_hash[l], pow[r - l + 1]);
return (raw_val + M) % M;
}
int prefix(int len) { return get_hash(0, len-1); }
int suffix(int len) { return get_hash(N-len, N-1); }
int whole() { return get_hash(0, N-1); }
int from(int l, int len) {
int r = l+len-1;
r = min(r, N-1);
return get_hash(l, r);
}
int to(int r, int len) {
int l = r-len+1;
l = max(l, 0LL);
return get_hash(l, r);
}
};
vector<int> HashedString::pow{1};
mt19937 rng((uint32_t)chrono::steady_clock::now().time_since_epoch().count());
const int HashedString::B = uniform_int_distribution<int>(0, M - 1)(rng);
//}}}