Templet
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Templet相关的知识,希望对你有一定的参考价值。
KMP
void getnext(){
int q, k;
next1[0] = 0;
for(q = 1, k = 0; q < m; q++){
while(k > 0 && sh[q] != sh[k])
k = next1[k-1];
if(sh[q] == sh[k])
k++;
next1[q] = k;
}
}
int kmp(){
int i, q;
int ans = 0;
for(i = 0, q = 0; i < n; i++){
while(q > 0 && sh[q] != ch[i])
q = next1[q-1];
if(sh[q] == ch[i])
q++;
if(q == m)
ans++;
}
return ans;
}
字典树
typedef struct Trie{
Trie *next[MAX];
int v;
}Trie;
Trie *root;
void createTrie(char *str){
int len = strlen(str);
Trie *p = root, *q;
for(int i = 0; i < len; i++){
int id = str[i] - ‘0‘;
if(p->next[id] == NULL){
q =(Trie *)malloc(sizeof(Trie));
q->v = 1;
for(int j = 0; j < MAX; j++){
q->next[j] = NULL;
}
p->next[id] = q;
p = p->next[id];
}
else{
p->next[id]->v++;
p = p->next[id];
}
}
p->v = -1;
}
int findTrie(char *str){
int len = strlen(str);
Trie *p = root;
for(int i = 0; i < len; i++){
int id = str[i] -‘0‘;
p = p->next[id];
if(p == NULL)
return 0;
if(p->v == -1)
return -1;
}
return -1;
}
void delete_(Trie *g){
if(g==NULL)
return;
for(int i = 0; i < MAX; i++){
if(g->next[i] != NULL)
delete_(g->next[i]);
}
free(g);
return;
}
AC自动机
struct node{
node *next[26];
node *fail;
int sum;
}*root, *q[10000000]
void createTrie(char *str){
node *p = root, *q;
int len = strlen(str);
for(int i = 0; i < len; i++){
int id = str[i] - ‘a‘;
if(p->next[id] == NULL){
q = (struct node *)malloc(sizeof(struct node));
for(int j = 0; j < 26; j++)
q->next[j] = 0;
q->sum = 0;
q->fail = 0;
p->next[id] = q;
}
p = p->next[id];
}
p->sum++;
}
void build_fail_pointer(){
head = 0,tail = 1;
q[head] = root;
node *p, *temp;
while(head<tail){
temp = q[head++];
for(int i = 0; i < 26; i++){
if(temp->next[i]){
if(temp == root)
temp->next[i]->fail = root;
else{
p = temp->fail;
while(p){
if(p->next[i]){
temp->next[i]->fail = p->next[i];
break;
}
p = p->fail;
}
if(p == NULL)
temp->next[i]->fail = root;
}
q[tail++] = temp->next[i];
}
}
}
}
void ac_automation(char *str){
node *p = root;
int len = strlen(str);
for(int i = 0; i < len; i++){
int id = str[i] - ‘a‘;
while(!p->next[id] && p != root)
p = p->fail;
p = p->next[id];
if(!p)
p = root;
node *temp = p;
while(temp != root){
if(temp->sum >= 0){
cnt += temp->sum;
temp->sum = -1;
}
else break;
temp = temp->fail;
}
}
}
主席树
void Build(int& o, int l, int r){
o = ++ tot;
sum[o] = 0;
if(l == r) return;
int m = (l + r) >> 1;
Build(ls[o], l, m);
Build(rs[o], m + 1, r);
}
void update(int& o, int l, int r, int last, int p){
o = + +tot;
ls[o] = ls[last];
rs[o] = rs[last];
sum[o] = sum[last] + 1;
if(l == r) return;
int m = (l + r) >> 1;
if(p <= m) update(ls[o], l, m, ls[last], p);
else update(rs[o], m + 1, r, rs[last], p);
}
int query(int ss, int tt, int l, int r, int k){
if(l == r) return l;
int m = (l + r) >> 1;
int cnt = sum[ls[tt]] - sum[ls[ss]];
if(k <= cnt) return query(ls[ss], ls[tt], l, m, k);
else return query(rs[ss], rs[tt], m + 1, r, k - cnt);
}
int main(){
scanf("%d", &T);
while(T--){
scanf("%d%d", &n, &q);
for(int i = 1; i <= n; i ++) scanf("%d", a + i), b[i] = a[i];
sort(b + 1, b + n + 1);
sz = unique(b + 1, b + n + 1) - (b + 1);
tot = 0;
Build(rt[0],1, sz);
//for(int i = 0; i <= 4 * n; i ++)
//printf("%d,rt = %d,ls = %d, rs = %d, sum = %d\n", i, rt[i], ls[i], rs[i], sum[i]);
for(int i = 1; i <= n; i ++)
a[i] = lower_bound(b + 1, b + sz + 1, a[i]) - b;
for(int i = 1; i <= n; i ++)
update(rt[i], 1, sz, rt[i - 1], a[i]);
//for(int i = 0; i <= 5 * n; i++)
//printf("%d,rt = %d, ls = %d, rs = %d, sum = %d\n", i, rt[i], ls[i], rs[i], sum[i]);
while(q --)work();
}
return 0;
}
SA
int A[MAX_N];
int rev[MAX_N*2], sa[MAX_N *2];
int n,k;
int rank1[MAX_N], tmp[MAX_N];
bool compare_sa(int i, int j){
if(rank1[i] != rank1[j])
return rank1[i] < rank1[j];
int ri = i + k <= n ? rank1[i+k] : -1;
int rj = j + k <= n ? rank1[j+k] : -1;
return ri < rj;
}
void construct_sa(int sh[],int n,int *sa){
for(int i = 0; i <= n; i++){
sa[i] = i;
rank1[i] = i < n ? sh[i] : -1;
}
for(k = 1; k <= n; k *= 2){
sort(sa, sa + n + 1, compare_sa);
tmp[sa[0]] = 0;
for( int i = 1; i <= n; i++){
tmp[sa[i]] = tmp[sa[i-1]] + (compare_sa(sa[i-1], sa[i])? 1 : 0);
}
for(int i = 0; i <= n; i++){
rank1[i] = tmp[i];
}
}
}
void solve(){
reverse_copy(A, A+n, rev);
construct_sa(rev, n, sa);
int p1;
for(int i = 0; i < n; i++){
p1 = n - sa[i];
if(p1 >= 1 && n - p1 >=2)
break;
}
int m = n-p1;
reverse_copy(A+p1, A+n, rev);
reverse_copy(A+p1, A+n, rev+m);
construct_sa(rev, m*2, sa);
int p2;
for(int i = 0; i <= 2*n; i++){
p2 = p1 + m -sa[i];
if(p2 - p1 >= 1 && n - p2 >= 1)
break;
}
reverse(A, A+p1);
reverse(A+p1, A+p2);
reverse(A+p2, A+n);
for(int i = 0; i < n; i++)
printf("%d\n", A[i]);
}
SA—LCP
void construct_lcp(string S, int *sa, int *lcp){
for(int i = 0; i <= n; i++)
rank1[sa[i]] = i;
int h = 0;
lcp[0] = 0;
for(int i = 0; i < n; i++){
int j = sa[rank1[i] - 1];
if(h > 0) h--;
for(; j + h < n && i + h < n; h++)
if(S[j+h] != S[i+h])
break;
lcp[rank1[i]-1] = h;
}
}
中国剩余定理
ll China(ll *m,ll *a)
{
ll M=1,d,y,x=0;
for(int i=0;i<n;i++)
M*=m[i];
for(int i=0;i<n;i++){
ll w=M/m[i];
gcd(m[i],w,d,d,y);
x=(x+y*w*a[i])%M;
}
return (x+M)%M;
}
扩展欧几里得算法
void ex_gcd(ll a,ll b,ll &d,ll &x,ll &y)
{
if(b==0){
d=a;
x=1,y=0;
}
else{
gcd(b,a%b,d,y,x);
y-=(a/b)*x;
}
}
最短路
struct edge{int from, to, cost};
void shortpath(int s){
for(int i = 0; i < V; i++) d[i] = INF;
d[s] = 0;
while(1){
bool update = false;
for(int i = 0; i < E; i++){
edge e = es[i];
if(d[e.from] != INF && d[e.to] > d[e.from] + e.cost){
d[e.to] = d[e.from] + e.cost;
update = true;
}
}
if(!update) break;
}
}
bool find_negative_loop(){
memset(d,0,sizeof(d));
for(int i = 0; i < n; i++)
{
for(int j = 0; j < m+w+m; j++)
{
edge e = ed[j];
if(d[e.to] > d[e.from] + e.cost)
{
d[e.to] = d[e.from] + e.cost;
if(i == n-1)
return true;
}
}
}
return false;
}
void floyd(int m){//}O(m^3)
for(int i = 0; i < m; i++)
for(int k = 0; k < m; k++)
for(int j = 0; j < m; j++)
d[k][j] = min(d[k][j], d[k][i] + d[i][j]);
}
struct edge{
int to,cost;
edge(){}
edge(int to, int cost) : to(to), cost(cost){}
};
vector<edge> g[Max];
typedef pair<int, int> P;
void dijkstra(int s){
priority_queue<P, vector<P>, greater<P> > que;
memset(d[s], 0x3f3f, Max * sizeof(int));
d[s][s] = 0;
que.push(P(0, s));
while(que.size())
{
P p = que.top(); que.pop();
int v = p.second;
if(d[s][v] < p.first) continue;
for(int i = 0; i < g[v].size(); i++)
{
edge e = g[v][i];
if(d[s][e.to] > d[s][v] + e.cost)
{
d[s][e.to] = d[s][v] + e.cost;
que.push(P(d[s][e.to], e.to));
}
}
}
return ;
}
生成树
struct edge{
int x, y, d;
const bool operator < (const edge& o) const { return d < o.d; }
}zhen[500002];
void initialize(){
for(int i = 1; i < N; i++) par[i] = i, _rank[i] = 0;
}
int find(int x){
if(x == par[x]) return x;
return par[x] = find(par[x]);
}
void unit(int x, int y){
int xx = find(x), yy = find(y);
if(xx == yy) return;
if(_rank[xx] < _rank[yy]) par[xx] = yy;
else
par[yy] = xx;
if(_rank[xx] == _rank[yy]) _rank[x]++;
}
bool is_thesame(int x, int y){ return find(x) == find(y); }
void kruskal(){
sort(ch, ch + E, cmp);
for(int i = 0; i < E; i++)
if(isthesame(ch[i].x, n + ch[i].y))
ans += ch[i].cost;
}
Int d[maxn][maxn], mincost[maxn];
bool used[maxn];
int prim(){
for(int i = 0; i < 102; i++)
{
mincost[i] = INF;
used[i] = false;
}
mincost[0] = 0;
int ans = 0;
while(true)
{
int flag = -1;
for(int i = 0; i < n; i++)
if(!used[i] && (flag == -1 || mincost[i]< mincost[flag]))
flag = i;
if(flag == -1)
break;
used[flag] = true;
ans += mincost[flag];
for(int i = 0; i < n; i++)
mincost[i] = min(mincost[i], d[flag][i]);
}
return ans;
}
快速幂
ll mod_pow(ll x, ll n, ll mod){
ll res = 1;
while(n>0){
if(n&1) res = res * x % mod;
x = x * x % mod;
n >>= 1;
}
return res;
}
尺取
int res = n+1;
int s = 0, t = 0, sum = 0;
while(1){
while(t<n && sum < S){//长度最短为S
sum+=a[t++];
}
if(sum < S) break;
res = min(res, t-s);
sum -= a[s++];
}
if(res > n)
res = 0;
线段树
long long ch[100010];
long long sum;
long long Stree[100010<<2],lazy[100010<<2];
void build(int root, int l, int r){
lazy[root] = 0;
if(l == r)
Stree[root] = ch[l];
else{
int mid = (l+r)/2;
build(root*2, l, mid);
build(root*2+1, mid + 1, r);
Stree[root] = Stree[root*2] + Stree[root*2+1];
}
}
void pushdown(int root,int num){
if(lazy[root]){
Stree[root*2] += 1LL*(num-(num>>1))*lazy[root];
Stree[root*2+1] += 1LL*(num>>1)*lazy[root];
lazy[root*2] += lazy[root];
lazy[root*2+1] += lazy[root];
lazy[root] = 0;
}
}
void add(int root, int L, int R, int l, int r, long long ad){
if(L>r || R < l) return;
if(R<=r && L>=l){
Stree[root] += 1LL*(R-L+1)*ad;
lazy[root] += ad;
return ;
}
pushdown(root,R-L+1);
int mid = (L+R) >> 1;
add(root*2, L, mid, l, r, ad);
add(root*2+1, mid+1, R, l, r, ad);
Stree[root] = Stree[root*2] + Stree[root*2+1];
}
void query(int root, int L, int R, int l, int r){
if(L>r || R < l) return;
if(l <= L && R <= r){
sum += Stree[root];
return ;
}
pushdown(root,R-L+1);
int mid = (L+R) >> 1;
query(root*2, L, mid, l, r);
query(root*2+1, mid+1, R, l, r);
}
BIT_Tree
int lowbit(int i)
{
return i&(-i);
}
void update(int i, int val)
{
while(i <= n)
{
a[i] += val;
i += lowbit(i);
}
}
int sum(int i)
{
int sum = 0;
while(i > 0)
{
sum += a[i];
i -= lowbit(i);
}
return sum;
}
以上是关于Templet的主要内容,如果未能解决你的问题,请参考以下文章