date: 20180820
spj: 距离NOIP还有81天

目录

STL模板:

priority_queue 的用法:重载<,struct cmp
queue 的用法
stack 的用法
vector的用法
map和set的用法
* 遍历容器中得所有元素
dequeue双端队列的用法

基础数论模板:

gcd
ex_gcd
求phi():筛选法、定义法
筛选法求质数
判断质数的一般方法
快速幂
矩阵快速幂
求逆元的方法(递推式、快速幂、ex_gcd)
卢卡斯定理的实现
组合数朴素公式
组合数递推式(杨辉三角)
组合数取模
二项式定理
n的正约数个数(唯一分解定理的推导)
ST表
裴蜀定理

高精度计算+-*

*二项式反演

*离散概率

图论模板:

floyd求最短路求最小环、图的传递闭包
spfa求最短路、dijkstra堆优化求最短路
kosaraju算法求强联通分量
tarjan求强联通分量
最小生成树
倍增LCA、tarjan LCA
并查集(启发式)
前向星存图
KM求最优匹配、匈牙利算法求最大匹配
FF算法求最大流

*最小费用最大流
树上Hash

数据结构模板:

树状数组模板(区间最大最小和)
线段树模板(区间最大最小和)
字典树(STL实现)
实数二分,整数二分答案
*线性基

字符串模板:

字符串哈希Hash(字符串必备)

*KMP算法初步

*AC自动机

说明:打*的模板未学习,历年NOIP范围内并未涉及,在本blog中不展开讲解(放代码)

正文

STL模板:

priority_queue 的用法:重载<,struct cmp

定义:

/*下面两种优先队列的定义是等价的*/
priority_queue<int> q;
priority_queue<int,vector<int>,less<int> > q;//后面有一个空格
/*这一种写法与上面不等价,恰好相反*/
priority_queue<int,vectot<int>,greater<int> > q;

重载小于号:

struct rec{
int x,y;
bool operator < (rec other){
if (x!=other.x) return x<other.x;
else return y<other.y;
}
};

struct cmp

struct cmp{
bool operator () (rec a,b){
if (a.x!=b.x) return a.x<b.x;
else return a.y<b.y;
}
};
priority_queue<rec,vector<rec>,cmp>q;

基本操作:

.top()      访问堆顶元素
.push() 插入一个新元素
.pop() 删除堆顶元素
.empty() 是否为空
.size() 堆中元素个数

queue 的用法 

单端队列

.empty()    判断队列q是否为空
.size() 访问队列q中的元素个数
.push() 会将一个元素a置入队列q中
.front() 返回队列q内的第一个元素
.back() 会返回队列q中最后一个元素
.pop() 会从队列q中移除第一个元素。

stack 的用法

.empty() 堆栈为空则返回真
.pop() 移除栈顶元素(不会返回栈顶元素的值)
.push() 在栈顶增加元素
.size() 返回栈中元素数目
.top() 返回栈顶元素

vector的用法

定义:vector<int>a;

用法:

.back();                返回a的最后一个元素
.front();           返回a的第一个元素
.push_back();           插入一个元素
a[i];           返回a的第i个元素,当且仅当a[i]存在
.erase(a.begin()+,a.begin()+);
删除a中第1个(从第0个算起)到第2个元素也就是说删除的元素从a.begin()+1算起(包括它)一直到a.begin()+(不包括它)
.pop_back();           删除a向量的最后一个元素
.clear();           清空a中的元素
.empty();            判断a是否为空,空则返回ture,不空则返回false
.swap(b);           b为向量,将a中的元素和b中的元素进行整体性交换
.find(a.begin(),a.end(),);  
在a中的从a.begin()(包括它)到a.end()(不包括它)的元素中查找10,若存在返回其在向量中的位置 遍历:
for(vector<int>::iterator it=b.begin();it!=b.end();it++)
*it就是元素值

map和set的用法

定义:

map<int,int>mp;//定义一个map
set<int>stt;//定义一个set

用法:

map

begin()          返回指向map头部的迭代器
clear() 删除所有元素
count() 返回指定元素出现的次数
empty() 如果map为空则返回true
end() 返回指向map末尾的迭代器
erase() 删除一个元素
find() 查找一个元素
insert() 插入元素
lower_bound() 返回键值>=给定元素的第一个位置
upper_bound() 返回键值>给定元素的第一个位置
size() 返回map中元素的个数

set

begin()           返回指向第一个元素的迭代器
clear() 清除所有元素
count() 返回某个值元素的个数
empty() 如果集合为空,返回true
end() 返回指向最后一个元素的迭代器
erase() 删除集合中的元素
find() 返回一个指向被查找到元素的迭代器
insert() 在集合中插入元素
lower_bound() 返回指向大于(或等于)某值的第一个元素的迭代器
upper_bound() 返回大于某个值元素的迭代器
size() 集合中元素的数目

* 遍历容器中得所有元素

//这里以vector为例
for (vector<int>::iterator it=a.begin();it!=a.end();it++)
printf("%d ",*it);

dequeue双端队列的用法

.size()         返回容器中实际数据的个数。
.pop_back() 删除最后一个数据。
.pop_front() 删除头部数据。
.erase(pos) 删除pos位置的数据,传回下一个数据的位置。
.erase(beg,end) 删除[beg,end)区间的数据,传回下一个数据的位置。
.front() 传回地一个数据。
.empty() 判断容器是否为空。
.end() 指向迭代器中的最后一个数据地址。
.back() 传回最后一个数据,不检查这个数据是否存在。
.begin() 传回迭代器重的可一个数据。
.clear() 移除容器中所有数据。

基础数论模板:

gcd

求a,b的最大公约数gcd(a,b)
int simple_gcd(int a,int b)
{
if (b==) return a;
return simple_gcd(b,a%b);
// 代码更短: return (!b)?a:simple_gcd(b,a%b);
}

ex_gcd

/*
求ax+by=gcd(a,b)的正整数解一组
证明:
易证明 a-[a/b]*b=a%b
由朴素gcd定理可知:
gcd(a,b)=gcd(b,a%b)
ax1+by1=bx2+(a%b)y2=gcd(a,b)
ax1+by1=bx2+( a-[a/b]*b)y2
解之得:x1=y2,y1=x2-[a/b]*y2;
*/ int ex_gcd(int a,int b,int &x,int &y)
{
if (b==) {
x=;y=;
return a;
}
int r=ex_gcd(b,a%b,x,y);
int t=x;x=y;y=t-a/b*y;
return r;
}

判断质数的一般方法

bool checkprime(int x)
{
for (int i=;i<=ceil(sqrt(x));i++)
if (!(x%i)) return false;
return true;
}

筛选法求质数

void getprime(int n)
{
bool prime[MAXN+]; memset(prime,true,sizeof(prime));
vector<int>a;a.clear();
for (int i=;i<=n;i++) {
if (prime[i]==false) continue;
a.push_back(i);
for (int j=i+i;j<=n;j+=i) prime[j]=false;
}
for (int i=;i<a.size();i++) printf("%d ",a[i]);//求出1-n所有质数
}

米勒罗宾判断质数:

int pow(int x,int n,int mo)
{
int ans=;
while (n) {
if (n&) ans=ans*x%mo;
x=x*x%mo;
n>>=;
}
return ans;
}
int is_prime(int x)
{
if (x==||x==) return -;
if (x==) return ;
for (int i=;i<=MAXT;i++) {
int m=rand()%(x-)+;
if (pow(m,x-,x)!=) return ;
}
return ;
}

求phi():筛选法、定义法

定义法:

void getprime(int n)
{
memset(prime,true,sizeof(prime));a.clear();
for (int i=;i<=n;i++) {
if (prime[i]==false) continue;
a.push_back(i);
for (int j=i+i;j<=n;j+=i) prime[j]=false;
}
}
int phi(int x)
{
getprime(x);
int p=,ph=x;
for (;;){
int nowprime=a[p];
if (nowprime>x||p==a.size()-) break;
if (x%nowprime==) ph=ph*(nowprime-)/nowprime;
p++;
}
return ph;
}

筛选法:

void getphi(int x)
{
memset(phi,,sizeof(phi));
phi[]=;
for (int i=;i<=x;i++)
if (phi[i]==)
{
for (int j=i;j<=x;j+=i){
if (phi[j]==) phi[j]=j;
phi[j]=phi[j]/i*(i-);
}
}
}

快速幂

递归式写法

const int mo=;
int pow(int x,int n)
{
if (n==) return ;
if (n==) return x;
int t=pow(x,n/)%mo;
t=t*t%mo;
if (n%) t=t*x%mo;
return t;
}

非递归式写法(加快乘)

//wrong code fixed by lukelin
int mul(int x,int n,int p){
int ans=;
for (;n;n>>=,x=(x<<)%p)
if (n&) ans=(ans+x)%p;
return ans;
} int fpow(int x,int n,int p){
int ans=;
for (;n;n>>=,x=x*x%p)
if (n&) ans=(ans*x)%p;
return ans;
}

矩阵快速幂

const int mo=1e9+;
const int MatixWide=;
struct Matix{
int m[MatixWide+][MatixWide+];
Matix operator * (const Matix &t) const {
Matix ans;
memset(ans.m,,sizeof(ans.m));
for (int i=;i<=MatixWide;i++)
for (int j=;j<=MatixWide;j++) {
int sum=;
for (int k=;k<=MatixWide;k++)
sum=(sum+m[i][k]*t.m[k][j]%mo)%mo;
ans.m[i][j]=sum;
}
return ans;
}
};
Matix pow(Matix x,int n)
{
Matix ans;
memset(ans.m,,sizeof(ans.m));
ans.m[][]=ans.m[][]=;n--;
while (n) {
if (n&) ans=ans*x;
x=x*x;
n>>=;
}
return ans;
}

求逆元的方法(递推式、快速幂、ex_gcd)

快速幂求逆元:

/*
基于费马小定理pow(a,p-2)%p 就是a在mod p意义下的逆元
*/
int pow(int x,int n,int m)
{
if (n==) return ;
if (n==) return x;
int t=pow(x,n/,m)%m;
t=t*t%m;
if (n%) t=t*x%m;
return t;
}
int inv(int x,int m)
{
return pow(x,m-,m)%m;
}

扩展欧几里得求逆元

/*
求ax=1(mod p)
就是求ax%p=1,所以ax ax=kp+1,就是ax-kp=1,
当p为质数的时候gcd(a,-p)=1,求此时x的值就是逆元
注意模到正数就行
*/ int ex_gcd(int a,int b,int &x,int &y)
{
if (b==) {
x=;y=;
return a;
}
int r=ex_gcd(b,a%b,x,y);
int t=x;x=y;y=t-a/b*y;
return r;
}
int inv(int a,int m)
{
int x,y;
int g=ex_gcd(a,m,x,y);
return (x+m)%m;
}

筛选法求逆元

/*
inv[i]=(p-p/i)*inv[p%i]%p
证明: 设t=p/i,k=p%i
那么 t*i+k=0(mod p)
同时除以i*k得: -t/k=1/i
所以 1/i=-t*1/k=-p/i*inv[p%i]
所以 把1/i(inv[i])调成正数为 inv[i]=(p-p/i)*inv[p%i]%p
*/ void getinv(int x,int p)
{
memset(inv,,sizeof(inv));
inv[]=;
for (int i=;i<=x;i++)
inv[i]=(long long)(p-p/i)%p*inv[p%i]%p;
}

组合数朴素公式

其中 n!=1*2*...*n,特别的,0!=1;

卢卡斯定理的实现

long long lucas(int m,int n)
{
if(n<m) return ;
else if(n<p) return s[n]*inv[m]*inv[n-m]%p;
else return lucas(m/p,n/p)*lucas(m%p,n%p)%p;
}
void getready()
{
inv[]=inv[]=s[]=s[]=;
for(int i=;i<=n;i++) {
s[i]=s[i-]*i%p;
inv[i]=(p-p/i)*inv[p%i]%p;
}
for(int i=;i<=n;i++) inv[i]=inv[i-]*inv[i]%p;
}

组合数递推式(杨辉三角)

    for (int i=;i<=n;i++) c[i][i]=c[i][]=;
for (int i=;i<=n;i++)
for (int j=;j<i;j++)
c[i][j]=c[i-][j-]+c[i-][j];

组合数取模

上至lucas定理或者暴力乘以inv(m!)和inv((n-m)!)公式法求解
二项式定理

n的正约数个数(唯一分解定理的推导)

/*
唯一分解定理:n=p1^a1+p2^a2+p3^a3+...+pk^ak
正约数个数为(a1+1)(a2+1)...(ak+1)个
如果是质数那么约数一定是2
特判1的约数为1
*/
int getnum(int x)
{
getprime(x);
if (x==||x==) return ; //其实0的约数没有意义
if (prime[x]) return ;
int p=,ans=,cnt;
for (;;){
int nowprime=a[p];
if (nowprime>x||p==a.size()-) break;
cnt=;
while (x%nowprime==) cnt++,x/=nowprime;
ans*=cnt;
p++;
}
return ans;
}

ST表

# include <bits/stdc++.h>
/*
ST表:倍增思想
MAX[i][j]第i个数后面2^j-1个数(含i是2^j)最大值
初始化:MAX[i][0]=a[i]
dp:MAX[i][j]=max(MAX[i][j-1],MAX[i+2^(j-1)][j-1])
求值令k=log2(r-l+1)
ans=max(MAX[l][k],MAX[r+1-2^k][k])
*/
using namespace std;
const int MAXN=;
int a[MAXN],MAX[MAXN][],n,m;
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
void init()
{
memset(MAX,,sizeof(MAX));
for (int i=;i<=n;i++) MAX[i][]=a[i];
for (int j=;j<=;j++)
for (int i=;i+(<<j)-<=n;i++)
MAX[i][j]=max(MAX[i][j-],MAX[i+(<<(j-))][j-]);
}
int query(int l,int r)
{
int k=log2(r-l+);
return max(MAX[l][k],MAX[r-(<<k)+][k]);
}
int main()
{
n=read(),m=read();
for (int i=;i<=n;i++) a[i]=read();
init(); int l,r;
while (m) {
l=read(),r=read();
printf("%d\n",query(l,r));
m--;
}
return ;
}

裴蜀定理

ax+by=c(x ∈N* ,y∈N*) 有正整数解的充要条件:gcd(a,b)∣c

设  s=gcd(a,b) ,显然s∣a ,并且s∣b

又因为x,y∈Z

所以s∣ax,s∣by

显然要使得之前的式子成立,则必须满足 c 是 a 和 b 的公约数的倍数

又因为 x 和 y 是正整数

所以 c 必然是 a,b 最大公约数的倍数。

因此,证得该定理成立

所以 对于给定a,b的f(x)=ax+by(x ∈N* ,y∈N*)最小值为c=gcd(a,b)

扩展到多元就是累计的gcd{ai} 就是最小值。

# include <bits/stdc++.h>
using namespace std;
int gcd(int a,int b)
{
return b==?a:gcd(b,a%b);
}
int main()
{
int n,ans=,a; scanf("%d",&n);
while (n--) {
scanf("%d",&a);
a=(a>)?a:-a;
ans=gcd(ans,a);
}
printf("%d\n",ans);
return ;
}

高精度计算

/*
Hint:支持正大整数a,b加法,减法(a>b),
支持大整数除以long long范围的实数
支持long long 转化为大整数
大整数转化为 unsigned long long
比大小
< : a<b true : false;
> : a>b true :false
<=:a<=b true:false
>=a>=b true :false
!= a!=b true :false
== a==b true :false
类型是lint ,输入值为lint,输出值为lint
支持位数为L=100005
*/
# include<bits/stdc++.h>
# define LL long long
# define Rint register int
using namespace std;
const int L=;
char s[L];
struct lint{
int len,num[L];
inline lint operator + (const lint &t) const{
lint ans;
memset(ans.num,,sizeof(ans.num));
if (len>t.len) ans.len=len;
else ans.len=t.len;
for (Rint i=;i<=ans.len;i++) {
ans.num[i]+=num[i]+t.num[i];
ans.num[i+]+=ans.num[i]/;
ans.num[i]%=;
}
if (ans.num[ans.len+]>) ans.len++;
return ans;
}
inline lint operator * (const lint &t) const{
lint ans;
memset(ans.num,,sizeof(ans.num));
for (Rint i=;i<=len;i++)
for (Rint j=;j<=t.len;j++)
ans.num[i+j-]+=num[i]*t.num[j];
for (Rint i=;i<=len+t.len;i++) {
ans.num[i+]+=ans.num[i]/;
ans.num[i]%=;
}
if (ans.num[len+t.len]>)
ans.len=len+t.len;
else ans.len=len+t.len-;
return ans;
}
inline lint operator - (const lint &t) const {
int nm=,rs=;
lint ans;
for (Rint i=;i<=len+;i++) {
nm=num[i]-t.num[i]-rs; rs=;
while (nm<) rs++,nm+=;
ans.num[i]=nm;
}
int i;
for (i=len;i>=;i--) if (ans.num[i]>) break;
ans.len=i;
return ans;
}
inline lint operator / (const int &x) {
lint a;
a.len=len;
int rest=;
for (int i=len;i>=;i--){
rest=rest*+num[i];
a.num[i]=rest/x;
rest%=x;
}
while (!a.num[a.len]&&a.len>) a.len--;
return a;
}
inline lint read() {
lint a; memset(a.num,,sizeof(a.num));
scanf("%s",s);
a.len=strlen(s);
for (Rint i=;i<=a.len;i++)
a.num[i]=s[a.len-i]-'';
return a;
}
inline int cmp(lint a,lint b) {
if (a.len<b.len) return ;
else if (a.len>b.len) return -;
for (Rint i=a.len;i>=;i--)
if (a.num[i]>b.num[i]) return -;
else if (a.num[i]<b.num[i]) return ;
return ;
}
inline bool operator < (const lint &t) const{
if (len>t.len) return ;
else if (len<t.len) return ;
for (Rint i=len;i>=;i--)
if (num[i]<t.num[i]) return ;
else if (num[i]>t.num[i]) return ;
return ;
}
inline bool operator <= (const lint &t) const{
if (len>t.len) return ;
else if (len<t.len) return ;
for (Rint i=len;i>=;i--)
if (num[i]<t.num[i]) return ;
else if (num[i]>t.num[i]) return ;
return ;
}
inline bool operator > (const lint &t) const{
if (len>t.len) return ;
else if (len<t.len) return ;
for (Rint i=len;i>=;i--)
if (num[i]<t.num[i]) return ;
else if (num[i]>t.num[i]) return ;
return ;
}
inline bool operator >= (const lint &t) const{
if (len>t.len) return ;
else if (len<t.len) return ;
for (Rint i=len;i>=;i--)
if (num[i]<t.num[i]) return ;
else if (num[i]>t.num[i]) return ;
return ;
}
inline bool operator ==(const lint &t) const{
if (len>t.len||len<t.len) return ;
for (Rint i=len;i>=;i--)
if (num[i]!=t.num[i]) return ;
return ;
}
inline bool operator !=(const lint &t) const{
if (len>t.len||len<t.len) return ;
for (Rint i=len;i>=;i--)
if (num[i]!=t.num[i]) return ;
return ;
}
inline void swap(lint &a,lint &b)
{
lint t=a; a=b; b=t;
}
inline void print(lint x)
{
for (Rint i=x.len;i>=;i--) printf("%d",x.num[i]);
putchar('\n');
}
inline lint change(LL x) {
lint a; memset(a.num,,sizeof(a.num));
if (x==) { a.num[]=; a.len=; return a;}
a.len=;
while (x>) a.num[++a.len]=x%,x/=;
return a;
}
inline unsigned LL rechange(lint x) {
unsigned LL a=0ll;
for (Rint i=x.len;i>=;i--) a=a*+x.num[i];
return a;
}
}R;
void work1()
{
lint a=R.read(),b=R.read(),c;
printf("a-b=");
if (a<b) { R.swap(a,b); printf("-");}
R.print(a-b);
c=a+b; printf("a+b="); R.print(c);
c=a*b; printf("a*b="); R.print(c);
printf("a<b = "); printf((a<b)?"Yes\n":"No\n");
printf("a<=b = "); printf((a<=b)?"Yes\n":"No\n");
printf("a>b = "); printf((a>b)?"Yes\n":"No\n");
printf("a>=b = "); printf((a>=b)?"Yes\n":"No\n");
printf("a==b = "); printf((a==b)?"Yes\n":"No\n");
printf("a!=b = "); printf((a!=b)?"Yes\n":"No\n");
}
void work2()
{
unsigned LL a,b;
scanf("%lld",&a);
lint u=R.change(a);
b=R.rechange(u);
printf("%lld\n",b);
}
int main()
{
work1();
work2();
return ;
}

离散化:

void lisanhua(int *d,int len)
{
tmp.clear();
for (int i=;i<=len;i++) tmp.push_back(d[i]);
sort(tmp.begin(),tmp.end());
vector<int>::iterator it=unique(tmp.begin(),tmp.end());
for (int i=;i<=len;i++)
d[i]=lower_bound(tmp.begin(),it,d[i])-tmp.begin()+;
}

图论模板:

floyd求最短路求最小环、图的传递闭包

最短路:

# include <bits/stdc++.h>
# define Rint register int
using namespace std;
const int MAXN=;
int mp[MAXN][MAXN];
int main()
{
int n,m,s; scanf("%d%d%d",&n,&m,&s);
for (Rint i=;i<=MAXN;i++)
for (Rint j=;j<=MAXN;j++)
if (i!=j) mp[i][j]=INT_MAX/;
int INF=mp[][];
int u,v,w;
for (Rint i=;i<=m;i++) {
scanf("%d%d%d",&u,&v,&w);
mp[u][v]=min(w,mp[u][v]);
}
for (Rint k=;k<=n;k++)
for (Rint i=;i<=n;i++)
for (Rint j=;j<=n;j++)
if (k!=i&&i!=j&&j!=k)
mp[i][j]=min(mp[i][j],mp[i][k]+mp[k][j]);
for (Rint i=;i<=n;i++)
if (mp[s][i]==INF) printf("2147483647 ");
else printf("%d ",mp[s][i]);
printf("\n");return ;
}

最小环

# include <bits/stdc++.h>
# define Rint register int
using namespace std;
const int MAXN=;
int mp[MAXN][MAXN],dist[MAXN][MAXN];
int main()
{
int n,m,s; scanf("%d%d%d",&n,&m,&s);
for (Rint i=;i<=MAXN;i++)
for (Rint j=;j<=MAXN;j++)
if (i!=j) dist[i][j]=mp[i][j]=INT_MAX/;
int INF=mp[][];
int u,v,w;
for (Rint i=;i<=m;i++) {
scanf("%d%d%d",&u,&v,&w);
mp[u][v]=mp[v][u]=dist[u][v]=dist[v][u]=min(w,mp[u][v]);
}
int ans=INT_MAX;
for (Rint k=;k<=n;k++) {
for (Rint i=;i<=k-;i++)
for (Rint j=i+;j<=k-;j++)
ans=min(ans,dist[i][j]+mp[k][i]+mp[j][k]);
for (Rint i=;i<=n;i++)
for (Rint j=;j<=n;j++)
dist[i][j]=min(dist[i][j],dist[i][k]+dist[k][j]);
}
printf("%d\n",ans);
return ;
}

传递闭包

# include <bits/stdc++.h>
# define Rint register int
using namespace std;
bool mp[][];
int n,m;
int main()
{
scanf("%d%d",&n,&m);
memset(mp,false,sizeof(mp));
for (int i=;i<=n;i++) mp[i][i]=true;
int u,v;
for (int i=;i<=m;i++)
scanf("%d%d",&u,&v),mp[u][v]=true;
for (Rint k=;k<=n;k++)
for (Rint i=;i<=n;i++)
for (Rint j=;j<=n;j++)
mp[i][j]=mp[i][j]|(mp[i][k]&&mp[k][j]);
for (Rint i=;i<=n;i++) {
for (Rint j=;j<=n;j++)
if (mp[i][j]) printf("Y "); else printf("N ");
printf("\n");
}
return ;
}

spfa求最短路、dijkstra堆优化求最短路

spfa求最短路

# include <bits/stdc++.h>
using namespace std;
queue<int>q;
struct rec {
int pre,to,w;
};
const int MAXN=,MAXM=,inf=;
int N,M,S,tot=;
int head[MAXN],d[MAXN];
rec a[MAXM];
void adde(int u,int v,int w)
{
a[++tot].pre=head[u];
a[tot].to=v;
a[tot].w=w;
head[u]=tot;
}
void spfa(int s)
{
for (int i=;i<=N;i++) d[i]=inf;
d[s]=;
q.push(s);
while (!q.empty()) {
int v0=q.front();q.pop();
for (int i=head[v0];i!=;i=a[i].pre){
if (d[v0]<d[a[i].to]-a[i].w) {
d[a[i].to]=d[v0]+a[i].w;
q.push(a[i].to);
}
}
}
}
int main()
{
scanf("%d%d%d",&N,&M,&S);
int u,v,w;
for (int i=;i<=M;i++){
scanf("%d%d%d",&u,&v,&w);
adde(u,v,w);
}
spfa(S);
for (int i=;i<=N;i++)
printf("%d ",d[i]);
printf("\n");
return ;
}

dijktra求最短路

/*
思想是每次找与当前便利点u最近的元素用它来更新子节点
*/
#include <bits/stdc++.h>
using namespace std;
const int MAXM=,MAXN=;
int head[MAXN],d[MAXN],n,m,s,tot=,INF;
bool vis[MAXN];
struct record{
int pre,to,w;
}a[MAXM];
void adde(int u,int v,int w)
{
a[++tot].pre=head[u];
a[tot].to=v;
a[tot].w=w;
head[u]=tot;
}
struct rec{
int id,lenth;
bool operator < (const rec a)const{
if (lenth!=a.lenth) return lenth>a.lenth;
else return id>a.id;
}
};
priority_queue<rec>q;
void dijkstra(int s)
{
memset(vis,false,sizeof(vis));
memset(d,,sizeof(d));INF=d[];d[s]=;
rec Node; Node.id=s; Node.lenth=; q.push(Node);
while (! q.empty()){
rec Node=q.top(); q.pop();
int u=Node.id;
if (vis[u]==true) continue;
vis[u]=true;
for (int i=head[u];i!=;i=a[i].pre){
int v=a[i].to;
if (d[v]-a[i].w>d[u]) {
d[v]=a[i].w+d[u];
rec N; N.id=v;N.lenth=d[v];
q.push(N);
}
}
}
}
int main()
{
scanf("%d%d%d",&n,&m,&s);
for (int i=;i<=m;i++) {
int u,v,w; scanf("%d%d%d",&u,&v,&w);
adde(u,v,w);
}
dijkstra(s);
for (int i=;i<=n;i++)
if (d[i]==INF) printf("2147483647 ");
else printf("%d ",d[i]);
return ;
}

求次短路

# include <iostream>
# include <cstdio>
# include <queue>
# include <algorithm>
# define int long long
using namespace std;
const int MAXN=;
const int MAXM=*;
struct A{ int pre,to,w;}a[MAXM];
struct B{ int u,v,w;};
vector<B>E;
int head[MAXN],d1[MAXN],d2[MAXN],tot=,n,m;
bool vis[MAXN];
inline int read()
{
int X=,w=;char c=;
while (!(c>=''&&c<='')) w|=c=='-',c=getchar();
while (c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
inline void write(int x)
{
if (x<) { putchar('-');x=-x;}
if (x>) write(x/);
putchar(x%+'');
}
inline void writeln(int x){write(x);putchar('\n');}
inline void adde(int u,int v,int w)
{
a[++tot].pre=head[u];
a[tot].to=v;
a[tot].w=w;
head[u]=tot;
}
struct node{ int id,val;};
struct cmp{
bool operator () (node a,node b) const{
return a.val>b.val;
}
};
priority_queue<node,vector<node>,cmp>q;
void dijkstra(int s,int *dist)
{ for(int i=;i<=n;i++) dist[i]=0x7f7f7f7f7f,vis[i]=false;
dist[s]=;
node Tmp;
Tmp.id=s,Tmp.val=;
q.push(Tmp);
while (!q.empty()) {
node u=q.top();q.pop();
if (vis[u.id]) continue;
vis[u.id]=true;
for (int i=head[u.id];i;i=a[i].pre) {
int v=a[i].to;
if (dist[v]-a[i].w>dist[u.id]) {
dist[v]=a[i].w+dist[u.id];
Tmp.id=v,Tmp.val=dist[v];
q.push(Tmp);
}
}
}
}
signed main()
{
n=read();m=read();
for (int i=;i<=m;i++) {
int u=read(),v=read(),w=read();
B Tmp1,Tmp2;
Tmp1.u=u;Tmp1.v=v;Tmp1.w=w;
Tmp2.u=v;Tmp2.v=u;Tmp2.w=w;
E.push_back(Tmp1);
E.push_back(Tmp2);
adde(u,v,w); adde(v,u,w);
}
dijkstra(,d1);
dijkstra(n,d2);
int shortest=d1[n];
int Ans=0x7f7f7f7f7f;
for (int i=;i<E.size();i++)
if (d1[E[i].u]+d2[E[i].v]+E[i].w!=shortest)
Ans=min(Ans,d1[E[i].u]+d2[E[i].v]+E[i].w);
writeln(Ans);
return ;
}

kosaraju算法求强联通分量

/*
https://www.cnblogs.com/ljc20020730/p/9359951.html
反图:设G为原图,F图为反图。
step1:对G遍历记录每个点的退出顺序dfn[]
对于下面这个图遍历序列为1 2 3 5 (5 3 2 )1 3(4 1)
括号括起来的是返回值
dfn=5 3 2 4 1
step2:对反图F 按照dfn的反序对反图进行dfs遍历
访问过程:1 4 (return)2 5 3 (return)
F图如下:
得出两个强联通分量就是14 ; 2 5 3;
在最小环处理的问题上要注意两个问题
1.图的强联通问题:必须保证图的联通性如果不行,那么反复dfs知道找到所有点为止
2.单个的点不是环及dfs2中得出cnt=1的情况要舍去
*/
# include <cstdio>
# include <cstring>
# include <iostream>
using namespace std;
struct rec{
int pre,to;
};
const int MAXN=;
int n,m,tot1=,tot2=;
int dfn[MAXN],head1[MAXN],head2[MAXN];
bool vis[MAXN];
rec a[MAXN],b[MAXN];
int adde1(int u,int v) //正图前向星
{
tot1++;
a[tot1].pre=head1[u];
a[tot1].to=v;
head1[u]=tot1;
}
int adde2(int u,int v)//反图前向星
{
tot2++;
b[tot2].pre=head2[u];
b[tot2].to=v;
head2[u]=tot2;
}
int dfs1(int u) //求dfn序
{
vis[u]=true;
for (int i=head1[u];i!=;i=a[i].pre)
{
int v=a[i].to;
if (vis[v]==true) continue;
dfs1(v);
}
dfn[++dfn[]]=u;
}
int dfs2(int u) //求环
{
vis[u]=true;
for (int i=head2[u];i!=;i=b[i].pre)
{
int v=b[i].to;
if (vis[v]==true) continue;
dfs2(v);
}
printf("%d ",u);
}
int main()
{
scanf("%d%d",&n,&m);
for (int i=;i<=m;i++)
{
int u,v;
scanf("%d%d",&u,&v);
adde1(u,v); adde2(v,u);//正图A,反图B;
}
memset(dfn,,sizeof(dfn));
memset(vis,false,sizeof(vis));
for (int i=;i<=n;i++)
if (vis[i]==false) dfs1(i); //得出dfn
/*
printf("*****************************************************\n");
printf("调试输出dfn[]: "); for (int i=1;i<=dfn[0];i++) printf("%d ",dfn[i]);
for (int i=1;i<=first[0];i++) printf("%d ",first[i]);
printf("\n");
printf("*****************************************************\n");
*/
memset(vis,false,sizeof(vis));
int cnt=;
for (int i=dfn[];i>;i--) //反序搞一搞求强联通块
{
if (vis[dfn[i]]==true) continue;
cnt++;
printf("强联通块# %d :",cnt);
dfs2(dfn[i]); printf("\n");
}
return ;
}

tarjan求强联通分量

# include <bits/stdc++.h>
using namespace std;
const int MAXN=,MAXM=;
bool ins[MAXN];
int low[MAXN],dfn[MAXN],head[MAXN],tot=,tt=,n,m;
struct rec{
int pre,to;
}a[MAXM];
stack<int>s;
void adde(int u,int v)
{
a[++tot].pre=head[u];
a[tot].to=v;
head[u]=tot;
}
void trajan(int u)
{
tt++; low[u]=dfn[u]=tt;
s.push(u); ins[u]=true;
for (int i=head[u];i;i=a[i].pre) {
int v=a[i].to;
if (! dfn[v]) {
trajan(v);
low[u]=min(low[u],low[v]);
} else
if (ins[v]) low[u]=min(low[u],dfn[v]);
}
if (dfn[u]==low[u]) {
while (true) {
int v=s.top();s.pop();ins[v]=false;
printf("%d ",v);
if (u==v) break;
}
printf("\n");
}
}
int main()
{
scanf("%d%d",&n,&m);
int u,v;
for (int i=;i<=m;i++)
scanf("%d%d",&u,&v),adde(u,v);
memset(ins,false,sizeof(ins));
memset(dfn,,sizeof(dfn));
for (int i=;i<=n;i++) if (!dfn[i]) trajan(i);
return ;
}

最小生成树

# include<bits/stdc++.h>
using namespace std;
const int MAXN=,MAXM=;
struct rec{
int u,v,w;
};
int tot=;
rec a[MAXM];
int f[MAXN];
int n,m;
bool cmp(rec x,rec y)
{
return x.w<y.w;
}
int father(int x)
{
if (f[x]==x) return x;
f[x]=father(f[x]);
return f[x];
}
void kruskal()
{
sort(a+,a++tot,cmp);
for (int i=;i<=n;i++) f[i]=i;
int cnt=,ans=;
for (int i=;i<=m;i++){
int fx=father(a[i].u);
int fy=father(a[i].v);
if (fx==fy) continue;
f[fx]=fy; cnt++;
ans+=a[i].w;
if (cnt==n-) break;
}
if (cnt==n-) printf("%d\n",ans);
else printf("无解");
}
int main()
{
scanf("%d%d",&n,&m);
tot=;
for (int i=;i<=m;i++)
{
int u,v,w;
scanf("%d%d%d",&u,&v,&w);
a[++tot].u=u;
a[tot].v=v;
a[tot].w=w;
}
kruskal();
return ;
}

倍增LCA、tarjan LCA

倍增lca

# include <iostream>
# include <cstdio>
# include <cstring>
using namespace std;
struct rec{
int pre,to;
};
int tot,H,n,m,s;
const int MAXN=*;
int head[MAXN],dep[MAXN],g[MAXN][];
bool vis[MAXN];
rec a[MAXN];
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
inline void adde(int u,int v)
{
a[++tot].pre=head[u];
a[tot].to=v;
head[u]=tot;
}
inline void dfs(int u,int depth)
{
vis[u]=true;
dep[u]=depth;
for (register int i=head[u];i!=;i=a[i].pre) {
int v=a[i].to; if (vis[v]) continue;
g[v][]=u;dfs(v,depth+);
}
}
inline void init()
{
memset(g,,sizeof(g)); memset(vis,false,sizeof(vis));
dfs(s,); g[s][]=s;
for (register int j=;j<=;j++)
for (register int i=;i<=n;i++)
g[i][j]=g[g[i][j-]][j-];
}
inline int LCA(int u,int v)
{
if (dep[u]<dep[v]) swap(u,v);
for (register int i=;i>=;i--)
if (dep[g[u][i]]>=dep[v]) u=g[u][i];
if (u==v) return u;
for (register int i=;i>=;i--)
if (g[u][i]!=g[v][i]) u=g[u][i],v=g[v][i];
return g[u][];
}
int main()
{
n=read();m=read();s=read(); int u,v;
for (register int i=;i<=n-;i++)
{u=read();v=read(); adde(u,v);adde(v,u);}
init();
while (m--){
u=read();v=read();
printf("%d\n",LCA(u,v));
}
return ;
}

tarjanLCA

/*
1.任选一个点为根节点,从根节点开始。
2.遍历该点u所有子节点v,并标记这些子节点v已被访问过。
3.若是v还有子节点,返回2,否则下一步。
4.合并v到u上。
5.寻找与当前点u有询问关系的点v。
6.若是v已经被访问过了,则可以确认u和v的最近公共祖先为v被合并到的父亲节点a。
*/
# include <iostream>
# include <cstdio>
# include <cstring>
# include <algorithm>
using namespace std;
const int MAXN=,MAXM=;
struct rec{
int pre,to,ans;
};
int head[MAXN],qhead[MAXN],f[MAXN];
bool vis[MAXN];
rec a[MAXM],qes[MAXM];
int n,m,q,tot=;
void adde(int u,int v)
{
tot++;
a[tot].to=v;
a[tot].pre=head[u];
head[u]=tot;
}
void qadde(int u,int v,int x)
{
qes[x].to=v;
qes[x].pre=qhead[u];
qhead[u]=x;
}
int father(int x)
{
if (f[x]==x) return x;
f[x]=father(f[x]);
return f[x];
}
void calc(int x,int y)
{
int fx=father(x);
int fy=father(y);
f[fy]=fx;
}
void tarjan(int u)
{
vis[u]=true;
for(int i=head[u];i!=;i=a[i].pre)
{
int v=a[i].to;
if (vis[v]==false) {
tarjan(v);
calc(u,v);
}
}
for (int i=qhead[u];i!=;i=qes[i].pre)
{
int v=qes[i].to;
if (vis[v]==false) continue;
qes[i].ans=father(v);
if (i%==) qes[i-].ans=qes[i].ans;
else qes[i+].ans=qes[i].ans;
}
}
int main()
{
int n,m,s;
scanf("%d%d%d",&n,&m,&s);
memset(f,,sizeof(f));
memset(a,,sizeof(a));
for (int i=;i<=n-;i++) {
int u,v;
scanf("%d%d",&u,&v);
adde(u,v);adde(v,u);
f[i]=i;
}
f[n]=n;
for (int i=;i<=m;i++){
int u,v;
scanf("%d%d",&u,&v);
qadde(u,v,*i-);qadde(v,u,*i);
}
memset(vis,false,sizeof(vis));
tarjan(s);
for (int i=;i<=m;i++) printf("%d\n",qes[*i].ans);
return ;
}

并查集(启发式)

# include <bits/stdc++.h>
using namespace std;
const int MAXN=;
int fa[MAXN];
int n,m;
int father(int x)
{
if (fa[x]==x) return x;
fa[x]=father(fa[x]);
return fa[x];
}
void calc(int x,int y)
{
int fx=father(x),fy=father(y);
fa[fx]=fy;
}
bool check(int x,int y)
{
int fx=father(x),fy=father(y);
if (fx==fy) return true;
else return false;
}
int main()
{
scanf("%d%d",&n,&m);
for (int i=;i<=n;i++) fa[i]=i;
int u,v,ch;
for (int i=;i<=m;i++) {
scanf("%d%d%d",&ch,&u,&v);
if (ch==) calc(u,v);
else {
if (check(u,v)) printf("Y\n");
else printf("N\n");
}
}
return ;
}

前向星存图

/*
加边操作
*/
void adde(int u,int v,int w)
{
a[++tot].pre=head[u];
a[tot].to=v;
a[tot].w=w;
head[u]=tot;
}
/*
遍历操作
*/
for (int i=head[u];i;i=a[i].pre)
{
int v=a[i].to;
}

KM求最优匹配、匈牙利算法求最大匹配

匈牙利算法求最大匹配

# include <iostream>
# include <algorithm>
# include <cstdio>
# include <cstring>
# include <cmath>
using namespace std;
const int MAXN=;
int pre[MAXN],n,m,k,ans=;
bool vis[MAXN],mp[MAXN][MAXN];
bool find(int u)
{
for (int i=;i<=m;i++)
if ((!vis[i])&&(mp[u][i])) {
vis[i]=true;
if (pre[i]==-||find(pre[i])){
pre[i]=u;
return true;
}
}
return false;
}
void solve()
{
memset(pre,-,sizeof(pre));
for (int i=;i<=n;i++){
memset(vis,false,sizeof(vis));
if (find(i)) ans++;
}
}
int main()
{
scanf("%d%d%d",&n,&m,&k);
memset(mp,false,sizeof(mp));
int u,v;
for (int i=;i<=k;i++) scanf("%d%d",&u,&v),mp[u][v]=true;
solve();
printf("%d\n",ans);
return ;
}

KM算法求最优匹配

# include <bits/stdc++.h>
using namespace std;
const int N=,M=;
bool X[N],Y[N];
int mp[N][M],pre[N],lx[N],ly[M],n,m,k;
bool find(int u)
{
X[u]=true;
for (int v=;v<=m;v++)
if ((!Y[v])&&(lx[u]+ly[v]==mp[u][v])){
Y[v]=true;
if (pre[v]==-||find(pre[v])) {
pre[v]=u;
return true;
}
}
return false;
}
int solve()
{
memset(pre,-,sizeof(pre));
memset(lx,,sizeof(lx));
memset(ly,,sizeof(lx));
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
lx[i]=max(lx[i],mp[i][j]);
for (int w=;w<=n;w++){
while (true) {
memset(X,false,sizeof(X));
memset(Y,false,sizeof(Y));
int d=INT_MAX;
if (find(w)) break;
for (int i=;i<=n;i++) if (X[i])
for (int j=;j<=m;j++) if (!Y[j])
d=min(d,lx[i]+ly[j]-mp[i][j]);
for (int i=;i<=n;i++) if (X[i]) lx[i]-=d;
for (int i=;i<=m;i++) if (Y[i]) ly[i]+=d;
}
}
int sum=;
for (int i=;i<=n;i++) sum+=lx[i];
for (int i=;i<=m;i++) sum+=ly[i];
return sum;
}
int main()
{
scanf("%d%d%d",&n,&m,&k);
int u,v,w;
memset(mp,-,sizeof(mp));
while (k--) scanf("%d%d%d",&u,&v,&w),mp[u][v]=w;
printf("%d\n",solve());
return ;
}

FF算法求最大流

// luogu-judger-enable-o2
# include <bits/stdc++.h>
# define Rint register int
using namespace std;
const int MAXN=,MAXM=;
struct rec{
int pre,to,w;
}a[*MAXM];
int head[MAXN],n,m,s,t,ans=,tot=;
bool vis[MAXN];
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
inline void adde(int u,int v,int w)
{
a[++tot].pre=head[u];
a[tot].to=v;
a[tot].w=w;
head[u]=tot;
}
inline int dfs(int u,int f)
{
if (u==t||f==) return f;
vis[u]=true;
for (Rint i=head[u];i;i=a[i].pre){
int v=a[i].to,d;
if (vis[v]) continue;
if ((a[i].w>)&&(d=dfs(v,min(f,a[i].w)))>){
a[i].w-=d;
if (i%) a[i+].w+=d; else a[i-].w+=d;
return d;
}
}
return ;
}
inline void FF()
{
while (true) {
memset(vis,false,sizeof(vis));
int f=dfs(s,INT_MAX);
if (f==) break;
ans+=f;
}
}
int main()
{
n=read();m=read();s=read();t=read();
int u,v,w;
for (Rint i=;i<=m;i++){
u=read();v=read();w=read();
adde(u,v,w); adde(v,u,);
}
ans=;
FF();
printf("%d\n",ans);
return ;
}

qwq

*最小费用最大流

树上Hash

https://www.cnblogs.com/AseanA/p/7633174.html

如何进行树上hash,设f[i]为以i为节点的hash值,则

f[i]=sigma(f[j]*prime[j])  j为son[i]

其中prime为预处理出来的素数表···注意f[j]需要进行排序····

然后对于两颗待判定的树,将两颗树分别以树上每一个节点为根节点求hash值··将根节点的hash值储存起来排序然后两颗树一一比对··如果完全一样则两棵树就一样

数据结构模板:

树状数组模板(区间最大最小和)

单点修改区间求和:

# include <bits/stdc++.h>
using namespace std;
const int MAXN=;
int c[MAXN],n,m;
int lowbit(int x)
{
return x&(-x);
}
void update(int x,int y)
{
while (x<=n) {
c[x]+=y;
x+=lowbit(x);
}
}
int query(int x)
{
int ret=;
while (x>) {
ret+=c[x];
x-=lowbit(x);
}
return ret;
}
int main()
{
scanf("%d%d",&n,&m);
memset(c,,sizeof(c));
for (int i=;i<=n;i++) {
int t; scanf("%d",&t);
update(i,t);
}
for (int i=;i<=m;i++) {
int ch,x,y;
scanf("%d%d%d",&ch,&x,&y);
if (ch==) update(x,y);
else printf("%d\n",query(y)-query(x-));
}
return ;
}

区间修改,单点求值

# include <bits/stdc++.h>
using namespace std;
const int MAXN=;
int c[MAXN],n,m;
int lowbit(int x)
{
return x&(-x);
}
void update(int x,int y)
{
while (x<=n) {
c[x]+=y;
x+=lowbit(x);
}
}
int query(int x)
{
int ret=;
while (x>) {
ret+=c[x];
x-=lowbit(x);
}
return ret;
}
int main()
{
scanf("%d%d",&n,&m);
memset(c,,sizeof(c));
int last=;
for (int i=;i<=n;i++) {
int t; scanf("%d",&t);
update(i,t-last);
last=t;
}
for (int i=;i<=m;i++) {
int ch,l,r,opx;
scanf("%d",&ch);
if (ch==) {
scanf("%d%d%d",&l,&r,&opx);
update(l,opx);
update(r+,-opx);
} else if (ch==) {
scanf("%d",&l);
printf("%d\n",query(l));
}
}
return ;
}

区间修改区间求值

    /*
c1[i]表示维护原序列a[i]的差分数列
c2[i]表示维护(i-1)*c1[i]
显然a[i]=c1[1]+c1[2]+...+c1[i];
a[1]+a[2]+a[3]+..a[i]=(c1[1])+(c1[1]+c1[2])+...+(c1[1]+c1[2]+...+c1[i])
=i*(c1[1]+c1[2]+c1[3]+...+c1[i])-(0*c1[1]+1*c1[2]+2*c1[3]+...+(i-1)*c1[i])
维护数组(i-1)*c1[i]的前缀和
a[1]+a[2]+a[3]+..a[i]=i*c1.query(i)-c2.query(i)
a[1]+a[2]+a[3]+..a[r]=r*c1.query(r)-c2.query(r)
a[1]+a[2]+a[3]+..a[l-1]=(l-1)*c1.query(l-1)-c2.query(l-1)
a[l]+a[l+1]+...+a[r-1]+a[r]=(a[1]+a[2]+...+a[r])-(a[1]+a[2]+...+a[l-1])
= r*c1.query(r)-c2.query(r)-((l-1)*c1.query(l-1)-c2.query(l-1))
*/
# include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll MAXN=;
ll n,m;
struct qwq{
ll c[MAXN];
ll lowbit(ll x)
{
return x&(-x);
}
void update(ll x,ll y)
{
while (x<=n) {
c[x]+=y;
x+=lowbit(x);
}
}
ll query(ll x)
{
ll ret=;
while (x>) {
ret+=c[x];
x-=lowbit(x);
}
return ret;
}
}c1,c2;
int main()
{
scanf("%lld%lld",&n,&m);
ll last=;
for (ll i=;i<=n;i++) {
ll t; scanf("%lld",&t);
c1.update(i,t-last);
c2.update(i,(i-)*(t-last));
last=t;
}
for (ll i=;i<=m;i++) {
ll ch,l,r,opx;
scanf("%lld",&ch);
if (ch==) {
scanf("%lld%lld%lld",&l,&r,&opx);
c1.update(l,opx);
c1.update(r+,-opx);
c2.update(l,(l-)*opx);
c2.update(r+,-r*opx);
} else if (ch==) {
scanf("%lld%lld",&l,&r);
ll sum1=(l-)*c1.query(l-)-c2.query(l-);
ll sum2=r*c1.query(r)-c2.query(r);
printf("%lld\n",sum2-sum1);
}
}
return ;
}

线段树模板(区间最大最小和)

区间修改(加)区间求和(lazy标记)

# include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MAXN=;
int n,m,opl,opr,opx;
ll f[MAXN*],a[MAXN],s[MAXN*],ans;
void build (int x,int l,int r)
{
if (l==r) {
f[x]=a[l];
return;
}
int mid=(l+r)/;
build (*x,l,mid);
build (*x+,mid+,r);
f[x]=f[*x]+f[*x+];
}
void down(int x,int l,int r)
{
int mid=(l+r)/;
f[*x]+=s[x]*(mid-l+);
f[*x+]+=s[x]*(r-mid);
s[*x]+=s[x];
s[*x+]+=s[x];
s[x]=;
}
void update(int x,int l,int r)
{
if (opl<=l&&r<=opr) {
f[x]+=(r-l+)*opx;
s[x]+=opx;
return;
}
if (s[x]>) down(x,l,r);
int mid=(l+r)/;
if (opl<=mid) update(*x,l,mid);
if (opr>mid) update(*x+,mid+,r);
f[x]=f[*x]+f[*x+];
}
void query(int x,int l,int r)
{
if (opl<=l&&r<=opr) {
ans+=f[x];
return;
}
int mid=(l+r)/;
if (s[x]>) down(x,l,r);
if (opl<=mid) query(*x,l,mid);
if (opr>mid) query(*x+,mid+,r);
}
int main()
{
scanf("%d%d",&n,&m);
for (int i=;i<=n;i++) scanf("%d",&a[i]);
build(,,n);
int ch;
for (int i=;i<=m;i++) {
scanf("%d",&ch);
if (ch==) scanf("%d%d%d",&opl,&opr,&opx),update(,,n);
else if (ch==) scanf("%d%d",&opl,&opr),ans=,query(,,n),printf("%lld\n",ans);
}
return ;
}

LCT之树链剖分

/*
如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作:
操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z
操作2: 格式: 2 x y 表示求树从x到y结点最短路径上所有节点的值之和
操作3: 格式: 3 x z 表示将以x为根节点的子树内所有节点值都加上z
操作4: 格式: 4 x 表示求以x为根节点的子树内所有节点值之和
*/
# include <bits/stdc++.h>
# define MAXN
using namespace std;
typedef long long ll;
int n,m,r,p,val[MAXN],b[MAXN];
struct Tree{
ll c[MAXN];
int lowbit(int x){ return x&(-x);}
void update(int x,int y){
while (x<=n){
c[x]+=y;
x+=lowbit(x);
}
}
ll query(int x){
ll ret=;
while (x>){
ret+=c[x];
x-=lowbit(x);
}
return ret;
}
}c1,c2;
struct Edge{
int pre,to;
}a[*MAXN];
int head[MAXN],tot=;
void adde(int u,int v)
{
a[++tot].pre=head[u];
a[tot].to=v;
head[u]=tot;
}
ll getsum(int l,int r)
{
return (ll) c1.query(r)*r-c2.query(r)-(l-)*c1.query(l-)+c2.query(l-);
}
int f[MAXN],dep[MAXN],son[MAXN],size[MAXN];
void dfs1(int u,int fa,int depth)//f[],dep[],son[],size[]
{
f[u]=fa;dep[u]=depth;size[u]=;
for (int i=head[u];i;i=a[i].pre)
{
int v=a[i].to; if (v==fa) continue;
dfs1(v,u,depth+);
size[u]+=size[v];
if (size[son[u]]<size[v]) son[u]=v;
}
}
int w[MAXN],cntw=,top[MAXN],old[MAXN];
void dfs2(int u,int tp) //w[],top[],old[]
{
w[u]=++cntw;top[u]=tp;
old[cntw]=u;
if (son[u]!=) dfs2(son[u],tp);
for (int i=head[u];i;i=a[i].pre)
{
int v=a[i].to; if (v==f[u]||v==son[u]) continue;
dfs2(v,v);
}
}
void change(int u,int v,int d) //此处u,v都为老编号
{
int f1=top[u],f2=top[v];
while (f1!=f2){
if (dep[f1]<dep[f2]) swap(f1,f2),swap(u,v);
c1.update(w[f1],d);
c1.update(w[u]+,-d);
c2.update(w[f1],d*(w[f1]-));
c2.update(w[u]+,-d*w[u]);
u=f[f1];
f1=top[u];
}
if (dep[u]<dep[v]) swap(u,v);
c1.update(w[v],d);
c1.update(w[u]+,-d);
c2.update(w[v],d*(w[v]-));
c2.update(w[u]+,-d*w[u]);
}
ll lca(int u,int v)
{
int f1=top[u],f2=top[v];
ll ret=0ll;
while (f1!=f2){
if (dep[f1]<dep[f2]) swap(f1,f2),swap(u,v);
//w[f1]~w[u]
ret=ret+getsum(w[f1],w[u]);
u=f[f1];
f1=top[u];
}
if (dep[u]<dep[v]) swap(u,v);
//w[v]~w[u]
ret=ret+getsum(w[v],w[u]);
return ret%p;
}
int main()
{
scanf("%d%d%d%d",&n,&m,&r,&p);
for (int i=;i<=n;i++) scanf("%d",&val[i]);
int u,v;
for (int i=;i<=n-;i++) {
scanf("%d%d",&u,&v);
adde(u,v); adde(v,u);
}
dfs1(r,,);
dfs2(r,);
for (int i=;i<=n;i++) b[i]=val[old[i]];
for (int i=;i<=n;i++) c1.update(i,b[i]-b[i-]),c2.update(i,(b[i]-b[i-])*(i-));
int ch,x,y,z;
for (int i=;i<=m;i++) {
scanf("%d%d",&ch,&x);
if (ch==) scanf("%d%d",&y,&z),change(x,y,z);
else if (ch==) scanf("%d",&y),printf("%lld\n",lca(x,y)%p);
else if (ch==) {
scanf("%d",&z);
int l=w[x],r=w[x]+size[x]-;
c1.update(l,z); c1.update(r+,-z);
c2.update(l,z*(l-)); c2.update(r+,-z*r);
}else if (ch==) {
int l=w[x],r=w[x]+size[x]-;
ll ret=getsum(l,r);
printf("%lld\n",ret%p);
}
}
return ;
}

字典树(STL实现)
实数二分,整数二分答案
*线性基

NOIP经典基础模板总结的更多相关文章

  1. new 经典基础模板总结

    NOIP-NOI-ZJOI基础模板总结 目录 C++语言和STL库操作 重载运算符操作 /* 重载运算符 格式 如重载小于号 这里是以x递减为第一关键字比较,y递减为第二关键字比较 */ bool o ...

  2. Bootstrap 4/3 页面基础模板 与 兼容旧版本浏览器

    Bootstrap 3 与 4 差别很大,目录文件结构.所引入的内容也不同,这里说说一下 Bootstrap 引入的文件.网页模板和兼容性问题.本网站刚刚搭建好,正好发一下文章原来测试网站. Boot ...

  3. Hadoop程序基础模板

    分布式编程相对复杂,而Hadoop本身蒙上大数据.云计算等各种面纱,让很多初学者望而却步.可事实上,Hadoop是一个很易用的分布式编程框架,经过良好封装屏蔽了很多分布式环境下的复杂问题,因此,对普通 ...

  4. 【转】手摸手,带你用vue撸后台 系列四(vueAdmin 一个极简的后台基础模板)

    前言 做这个 vueAdmin-template 的主要原因是: vue-element-admin 这个项目的初衷是一个vue的管理后台集成方案,把平时用到的一些组件或者经验分享给大家,同时它也在不 ...

  5. 树状数组(二叉索引树 BIT Fenwick树) *【一维基础模板】(查询区间和+修改更新)

    刘汝佳:<训练指南>Page(194) #include <stdio.h> #include <string.h> #include <stdlib.h&g ...

  6. jsp基础模板

    jsp页面基础模板 base.jsp <%@ page language="java" contentType="text/html; charset=UTF-8& ...

  7. vue-cli3 搭建的前端项目基础模板

    基于 vue-cli3 搭建的前端模板,fork 或 clone 本仓库,即可搭建完成一个新项目的基础模板,源码地址,欢迎 star 或 fork 特性 CSS 预编译语言:less Ajax: ax ...

  8. 最短路径(最基础,经典的模板和思想):HDU-2544最短路

    题目: 最短路 Time Limit: 5000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Su ...

  9. NOIP前的模板复习和注意事项

    联赛除去今天刚好只有一个星期了,最后一个星期也很关键,要吃好睡好保持心情愉悦.当然也免不了最后的复习计划. 首先是模板,之前还有很多模板没有复习到,这些东西是一定要落实到位的. 每天往后面写一点... ...

随机推荐

  1. iOS基于B站的IJKPlayer框架的流媒体探究

    阅读数:6555 学习交流及技术讨论可新浪微博关注:极客James 一.流媒体 流媒体技术从传输形式上可以分为:渐进式下载和实施流媒体. 1.渐进式下载 它是介于实时播放和本地播放之间的一种播放方式, ...

  2. iOS--LaunchImage启动页设置及问题解决

    在Assets.xcassets中使用LaunchImage来设置启动图:   一.根据不同屏幕尺寸的需求设置不同的图片,可以参照下图: 1.点击Image.xcassets 进入图片管理,然后右击, ...

  3. 持续集成之Jenkins自动部署war包到远程服务器

    一.无war包链接的情况 无war包链接时,需先下载war包到本地,然后执行: ---------------------------------------------以下部分为转载-------- ...

  4. Docker存储驱动Device Mapper,Overlay,AUFS

    Docker存储驱动之Device Mapper简介 - BookShu - 博客园https://www.cnblogs.com/styshoo/p/6528762.html Docker存储驱动之 ...

  5. JS 获取链接中的参数

    1.获取链接全部参数,以对象的形式返回 //获取url中参数 function GetRequest() { var url = location.search; //获取url中"?&qu ...

  6. 利用js给datalist或select动态添加option选项

    <!DOCTYPE html> <html> <head> <title>鼠标点击时加载</title> <script type=& ...

  7. 【apache2】AH00543: httpd: bad user name apache

    当启动 apache 时,出现一下异常:AH00543: httpd: bad user name daemon 解决方法:            #groupadd daemon          ...

  8. StringBuilder与String有哪些区别?

    System.String具备不可修改性,在程序中这样的特性容易产生性能上的问题.针对这个问题.NET提供的StringBuilder类可以解决类似的问题. String 和 StringBuilde ...

  9. @Autowired 与@Resource的区别(详细)

    参考:@Autowired 与@Resource的区别(详细) spring不但支持自己定义的@Autowired注解,还支持几个由JSR-250规范定义的注解,它们分别是@Resource.@Pos ...

  10. 日志与python日志组件logging

    1. 日志的相关概念: (1)日志的作用: a. 开发人员进行程序调试 b. 开发人员定位程序故障的位置 c. 运维人员观察应用运行是否正常 (2)日志的等级 a. DEBUG 最详细的日志,用于问题 ...