树链剖分详解,看这一篇就够了

news2024/11/15 2:08:42

前置知识:

  • 树形结构
  • 链式前向星(熟练)
  • 线段树(熟练)
  • DFS序(熟练)
  • LCA(了解定义)

什么是树链剖分

树链剖分其实有两种:重链剖分和长链剖分。重链剖分就是把儿子节点最重的儿子称为重儿子,把树分成若干条重链(如图一);

图一
图一

 长链剖分则是把深度差最大的儿子称为长儿子,把树分成若干条长链(如图二):

图二

 树链剖分的应用非常广泛,一般用的都是重链剖分。

树链剖分是提高树上搜索效率的一个巧妙方法,它一定规则把树剖分成一条线性的不相交的链,对整棵树的操作,就转化为对链的操作;而从根到任何一条链只是经过\Theta (\log _2n)条链从而操作的复杂度为\Theta (\log _2n)

树链剖分的一个特别之处,就是每条链的 DFS序 是有序的可以使用线段树处理,从而高效地解决一些树上的修改和查询问题。

树链剖分和LCA

首先通过求最近公共祖先介绍树链剖分的基本概念,而且进行树链剖分时求 LCA 是必需的步骤。

LCA 的概念

  • LCA问题:在一棵有根树中,一个节点的祖先节点是指它本身或者它父节点的祖先。给定两个节点,两个点共同的祖先中距离两者最近的节点就是这两个节点的最近公共祖先。需要注意的是最近公共祖先可能是这两个节点中的某一个。
  • 求LCA的各种算法都是快速向上“跳”到祖先节点。回顾求LCA的两种方法,其思想可以概括为:①倍增法,用二进制递增直接向祖先“跳”;②Tarjan算法,用并查集合并子树,子树内的节点都指向子树的根,查询LCA时,可以从节点直接跳到它所在的子树的根,从而实现快速跳的目的。

树链剖分的基本概念:

  1. 重儿子:假设x有n个儿子节点,其中以3儿子节点的为根子树大小最大,3就是x的重儿子
  2. 轻儿子:除重儿子外的所有儿子均为轻儿子
  3. 以图三为例:1的重儿子为3,轻儿子为2;3的重儿子为6,其余的为轻儿子
  4. 轻边:x与轻儿子相连的边
  5. 重边:x与重儿子相连的边
  6. 轻链:均由轻儿子构成的一条链
  7. 重链:均由重儿子构成的一条链
图三

 

进行树链剖分时,求 LCA也是必需的步骤。 和倍增法很像,树链剖分也是“跳”到祖先节点,它的跳法比较巧妙。每条链路内的节点可以看作一个集合,并以“链头”为集;链路上的节点查询LCA时,都指向链头,从而实现快速跳的目的。特别关键的是,从根到叶子只需要经过\Theta (\log _2n)条链,那么从一个节点跳到它的LCA,只需要跳\Theta (\log _2n)条链。

如何把树剖成链,使从根到叶子经过的链更少?注意每个节点只能属于一条链。很自然的思路是观察树上节点的分布情况,如果选择那些有更多节点的分支建链,链会更长一些,从而使链的数量更少。如图四所示,图四从根节点a开始,每次选择有更多子节点的分支建链,最后形成了粗线条所示的3条链。从叶子节点到根,最多经过两条链。例如,从 h 到 a,先经过以 d 为头的链,然后就到了以 a 为头的链。

预处理节点信息


我们需要的信息如下

  • dep[X]:x节点的深度
  • fa[X]:x节点的父亲节点
  • son[X]:x节点的重儿子
  • siz[X]:x节点为根的子树大小
  • top[X]:x节点所在链的顶点

首先第一个DFS我们直接获取 dep[X],fa[X],son[X],siz[X]

void dfs1(int x) {
	siz[x] = 1;
	dep[x] = dep[f[x]] + 1;
	for (int i = head[x]; i; i = e[i].ne) {
		int dd = e[i].to;
		if (dd == f[x])continue;
		f[dd] = x;
		dfs1(dd);
		siz[x] += siz[dd];
		if (!son[x] || siz[son[x]] < siz[dd])
			son[x] = dd;
	}
}

接下来获取top[X]

我们处理的方式:优先对重儿子处理,重儿子处理结束后再处理轻儿子(新开链)

void dfs2(int x, int tv) {
	top[x] = tv;
	if (son[x])dfs2(son[x], tv);
	for (int i = head[x]; i; i = e[i].ne) {
		int dd = e[i].to;
		if (dd == f[x] || dd == son[x])continue;
		dfs2(dd, dd);
	}
}

现在,我们按照倍增法求LCA的思路推导,看看能不能转化到树链剖分上。

经过剖分得到重链后,该怎么求 x,y 的LCA呢?有两种情况:

  1. 当两个节点在同一条重链上的时候。重链上的节点都是祖先和后代的关系,所以位置就在深度最浅的那个节点上。
  2. x和y不在同一条重链上。让x和y沿着这一条重链向上条,直到位于同一条重链,此时直接向上跳即可。

好了,现在我们掌握了精髓,代码如下(是不是很简单?):

for (int i = 1; i <= m; ++i) {
		int x, y;
		scanf("%d%d", &x, &y);
		while (top[x] != top[y]) {
			if (dep[top[x]] >= dep[top[y]])x = f[top[x]];
			else y = f[top[y]];
		}
		printf("%d\n", dep[x] < dep[y] ? x : y);
	}

完整代码(洛谷P3379)

//来自于https://www.luogu.com.cn/problem/P3379
#include <cstdio>
#include <iostream>
using namespace std;
struct edge {
	int to, ne;
} e[1000005];
int n, m, s, ecnt, head[500005], dep[500005], siz[500005], son[500005], top[500005], f[500005];
void add(int x, int y) {
	e[++ecnt].to = y;
	e[ecnt].ne = head[x];
	head[x] = ecnt;
}
void dfs1(int x) {
	siz[x] = 1;
	dep[x] = dep[f[x]] + 1;
	for (int i = head[x]; i; i = e[i].ne) {
		int dd = e[i].to;
		if (dd == f[x])continue;
		f[dd] = x;
		dfs1(dd);
		siz[x] += siz[dd];
		if (!son[x] || siz[son[x]] < siz[dd])
			son[x] = dd;
	}
}
void dfs2(int x, int tv) {
	top[x] = tv;
	if (son[x])dfs2(son[x], tv);
	for (int i = head[x]; i; i = e[i].ne) {
		int dd = e[i].to;
		if (dd == f[x] || dd == son[x])continue;
		dfs2(dd, dd);
	}
}
int main() {
	scanf("%d%d%d", &n, &m, &s);
	for (int i = 1; i < n; ++i) {
		int x, y;
		scanf("%d%d", &x, &y);
		add(x, y);
		add(y, x);
	}
	dfs1(s);
	dfs2(s, s);
	for (int i = 1; i <= m; ++i) {
		int x, y;
		scanf("%d%d", &x, &y);
		while (top[x] != top[y]) {
			if (dep[top[x]] >= dep[top[y]])x = f[top[x]];
			else y = f[top[y]];
		}
		printf("%d\n", dep[x] < dep[y] ? x : y);
	}
}

典型应用

关于重链,还有一个重要特征没有提到:一条重链内部节点的DFS序(时间戳)是连续的。也就是说,如果用DFS序标记这条重链上的节点,那么这条重链就变成一段连续的数字。把这段连续的数字看作“线段”,线段内的区间问题用线段树处理正合适。

根据上述讨论,能够用数据结构(一般是线段树)维护重链,从而高效解决一些树上的问题,如以下问题。

  1. 修改点x到点y的路径上各点的权值。
  2. 查询点x到点y的路径上节点权值之和。
  3. 修改点x子树上各点的权值。
  4. 查询点x子树上所有节点的权值之和。

其中,问题(1)是“树上差分”问题,详见前面的“倍增+差分”解法。树上差分只能解决简单的修改问题,对问题(3)这样的修改整棵子树的问题,树上差分就行不通了。

1.重链的DFS序(时间戳)

前面给出的dfs2()函数,是先DFS重儿子,再DFS轻儿子。如果在dfs2()函数的第1句用编号i d [ x ]记录节点x的DFS序,即id[x] =++num,对每个节点重新编号的结果如图4.50所示。

观察到以下现象。

部节点的编号是有序的。重链{ a , b , e , j , q }的DFS序是1,2,3,4,5;重链d,p的DFS序是7,8;重链c,f的DFS序是10,11

图4.50 重链的DFS序

(2)每棵子树上的所有节点的编号也是有序的。例如,以e为根的子树{ e , i , j , q },其DFS序是3,4,5,6

下面是关键内容-用线段树处理重链。由于每条重链内部的节点是有序的,可以按DFS序把它们安排在一棵线段树上。把每条重链看作一个连续的区间,对一条重链内部的修改和查询,用线段树处理;若一条路径跨越了多条重链,简单地跳过两条重链之间的轻边即可。

提示

重链内部用线段树,重链之间跳过。

如图4.51所示,先建一棵线段树,然后把线段树的节点看作DFS序(时间戳)。DFS序对应了原来那棵树的节点。同一条重链的节点,在线段树上是连续的。

图4.51 用线段树重建树链

2.修改x到y的最短路径上的节点权值

x到y的最短路径经过LCA ( x , y ),这实际上是一个查找LCA ( x , y )的过程。可以借助重链修改路径上的节点权值,步骤如下。

  1. 令x的链头的深度更深,即t o p [ x ] \geq t o p [ y ]。从x开始向上走,先沿着x所在的重链向上走,修改这一段路径上的节点。
  2. 到达x的链头后,跳过一条轻边,到达上一条重链。
  3. 继续执行步骤(1)和步骤(2),直到x和y位于同一条重链上,再修改此时两点之间的节点。

例如,修改从p到q的路径上所有节点权值之和,步骤如下。

  1. 从p走到它的链头t o p [ p ] = d,修改p和d的权值。
  2. 跳到b。
  3. b和q在同一条重链上,修改从b到q的权值。

用线段树处理上述过程,仍以修改从p到q的路径上节点权值之和为例。

  1. 从p跳到链头d,p和d属于同一条重链,用线段树修改对应的[7,8]区间。
  2. 从d穿过轻边(b,d),到达b所在的重链。
  3. b和q在同一条重链上,用线段树修改对应区间[2,5]

3.查询x到y的路径上所有节点权值之和

查询与修改的过程几乎一样,以查询p到q的路径上节点权值之和为例。

  1. 从p跳到链头d,p和d属于同一条重链,用线段树查询对应的[7,8]区间;
  2. 从d穿过轻边(b,d),到达b所在的重链;
  3. b和q在同一条重链上,用线段树查询对应区间[2,5]。

4.修改x的子树上各点的权值,查询x的子树上节点权值之和

每棵子树上的所有节点的DFS序是连续的,也就是说,每棵子树对应了一个连续的区间。

例题——洛谷P3384 轻重链剖分

题目描述

如题,已知一棵包含 NN 个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作:

  • 1 x y z,表示将树从 xx 到 yy 结点最短路径上所有节点的值都加上 zz。

  • 2 x y,表示求树从 xx 到 yy 结点最短路径上所有节点的值之和。

  • 3 x z,表示将以 xx 为根节点的子树内所有节点值都加上 zz。

  • 4 x 表示求以 xx 为根节点的子树内所有节点值之和

输入格式

第一行包含 44 个正整数 N,M,R,PN,M,R,P,分别表示树的结点个数、操作个数、根节点序号和取模数(即所有的输出结果均对此取模)。

接下来一行包含 NN 个非负整数,分别依次表示各个节点上初始的数值。

接下来 N-1N−1 行每行包含两个整数 x,yx,y,表示点 xx 和点 yy 之间连有一条边(保证无环且连通)。

接下来 MM 行每行包含若干个正整数,每行表示一个操作。

输出格式

输出包含若干行,分别依次表示每个操作 22 或操作 44 所得的结果(对 PP 取模)。

输入输出样例

输入 #1复制

5 5 2 24
7 3 7 8 0 
1 2
1 5
3 1
4 1
3 4 2
3 2 2
4 5
1 5 1 3
2 1 3

输出 #1复制

2
21

说明/提示

【数据规模】

对于 30\%30% 的数据: 1 \leq N \leq 101≤N≤10,1 \leq M \leq 101≤M≤10;

对于 70\%70% 的数据: 1 \leq N \leq {10}^31≤N≤103,1 \leq M \leq {10}^31≤M≤103;

对于 100\%100% 的数据: 1\le N \leq {10}^51≤N≤105,1\le M \leq {10}^51≤M≤105,1\le R\le N1≤R≤N,1\le P \le 2^{30}1≤P≤230。所有输入的数均在 int 范围内。

【样例说明】

树的结构如下:

各个操作如下:

故输出应依次为 22 和 2121。

代码如下:
//线段树与树链剖分的结合
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#define Rint register int
#define mem(a,b) memset(a,(b),sizeof(a))
#define Temp template<typename T>
using namespace std;
typedef long long LL;
Temp inline void read(T &x){
    x=0;T w=1,ch=getchar();
    while(!isdigit(ch)&&ch!='-')ch=getchar();
    if(ch=='-')w=-1,ch=getchar();
    while(isdigit(ch))x=(x<<3)+(x<<1)+(ch^'0'),ch=getchar();
    x=x*w;
}

#define mid ((l+r)>>1)
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
#define len (r-l+1)

const int maxn=200000+10;
int n,m,r,mod;
//见题意 
int e,beg[maxn],nex[maxn],to[maxn],w[maxn],wt[maxn];
//链式前向星数组,w[]、wt[]初始点权数组 
int a[maxn<<2],laz[maxn<<2];
//线段树数组、lazy操作 
int son[maxn],id[maxn],fa[maxn],cnt,dep[maxn],siz[maxn],top[maxn]; 
//son[]重儿子编号,id[]新编号,fa[]父亲节点,cnt dfs_clock/dfs序,dep[]深度,siz[]子树大小,top[]当前链顶端节点 
int res=0;
//查询答案 

inline void add(int x,int y){//链式前向星加边 
    to[++e]=y;
    nex[e]=beg[x];
    beg[x]=e;
}
//-------------------------------------- 以下为线段树 
inline void pushdown(int rt,int lenn){
    laz[rt<<1]+=laz[rt];
    laz[rt<<1|1]+=laz[rt];
    a[rt<<1]+=laz[rt]*(lenn-(lenn>>1));
    a[rt<<1|1]+=laz[rt]*(lenn>>1);
    a[rt<<1]%=mod;
    a[rt<<1|1]%=mod;
    laz[rt]=0;
}

inline void build(int rt,int l,int r){
    if(l==r){
        a[rt]=wt[l];
        if(a[rt]>mod)a[rt]%=mod;
        return;
    }
    build(lson);
    build(rson);
    a[rt]=(a[rt<<1]+a[rt<<1|1])%mod;
}

inline void query(int rt,int l,int r,int L,int R){
    if(L<=l&&r<=R){res+=a[rt];res%=mod;return;}
    else{
        if(laz[rt])pushdown(rt,len);
        if(L<=mid)query(lson,L,R);
        if(R>mid)query(rson,L,R);
    }
}

inline void update(int rt,int l,int r,int L,int R,int k){
    if(L<=l&&r<=R){
        laz[rt]+=k;
        a[rt]+=k*len;
    }
    else{
        if(laz[rt])pushdown(rt,len);
        if(L<=mid)update(lson,L,R,k);
        if(R>mid)update(rson,L,R,k);
        a[rt]=(a[rt<<1]+a[rt<<1|1])%mod;
    }
}
//---------------------------------以上为线段树 
inline int qRange(int x,int y){
    int ans=0;
    while(top[x]!=top[y]){//当两个点不在同一条链上 
        if(dep[top[x]]<dep[top[y]])swap(x,y);//把x点改为所在链顶端的深度更深的那个点
        res=0;
        query(1,1,n,id[top[x]],id[x]);//ans加上x点到x所在链顶端 这一段区间的点权和
        ans+=res;
        ans%=mod;//按题意取模 
        x=fa[top[x]];//把x跳到x所在链顶端的那个点的上面一个点
    }
    //直到两个点处于一条链上
    if(dep[x]>dep[y])swap(x,y);//把x点深度更深的那个点
    res=0;
    query(1,1,n,id[x],id[y]);//这时再加上此时两个点的区间和即可
    ans+=res;
    return ans%mod;
}

inline void updRange(int x,int y,int k){//同上 
    k%=mod;
    while(top[x]!=top[y]){
        if(dep[top[x]]<dep[top[y]])swap(x,y);
        update(1,1,n,id[top[x]],id[x],k);
        x=fa[top[x]];
    }
    if(dep[x]>dep[y])swap(x,y);
    update(1,1,n,id[x],id[y],k);
}

inline int qSon(int x){
    res=0;
    query(1,1,n,id[x],id[x]+siz[x]-1);//子树区间右端点为id[x]+siz[x]-1 
    return res;
}

inline void updSon(int x,int k){//同上 
    update(1,1,n,id[x],id[x]+siz[x]-1,k);
}

inline void dfs1(int x,int f,int deep){//x当前节点,f父亲,deep深度 
    dep[x]=deep;//标记每个点的深度 
    fa[x]=f;//标记每个点的父亲 
    siz[x]=1;//标记每个非叶子节点的子树大小 
    int maxson=-1;//记录重儿子的儿子数 
    for(Rint i=beg[x];i;i=nex[i]){
        int y=to[i];
        if(y==f)continue;//若为父亲则continue 
        dfs1(y,x,deep+1);//dfs其儿子 
        siz[x]+=siz[y];//把它的儿子数加到它身上 
        if(siz[y]>maxson)son[x]=y,maxson=siz[y];//标记每个非叶子节点的重儿子编号 
    }
}

inline void dfs2(int x,int topf){//x当前节点,topf当前链的最顶端的节点 
    id[x]=++cnt;//标记每个点的新编号 
    wt[cnt]=w[x];//把每个点的初始值赋到新编号上来 
    top[x]=topf;//这个点所在链的顶端 
    if(!son[x])return;//如果没有儿子则返回 
    dfs2(son[x],topf);//按先处理重儿子,再处理轻儿子的顺序递归处理 
    for(Rint i=beg[x];i;i=nex[i]){
        int y=to[i];
        if(y==fa[x]||y==son[x])continue;
        dfs2(y,y);//对于每一个轻儿子都有一条从它自己开始的链 
    }
}

int main(){
    read(n);read(m);read(r);read(mod);
    for(Rint i=1;i<=n;i++)read(w[i]);
    for(Rint i=1;i<n;i++){
        int a,b;
        read(a);read(b);
        add(a,b);add(b,a);
    }
    dfs1(r,0,1);
    dfs2(r,r);
    build(1,1,n);
    while(m--){
        int k,x,y,z;
        read(k);
        if(k==1){
            read(x);read(y);read(z);
            updRange(x,y,z);
        }
        else if(k==2){
            read(x);read(y);
            printf("%d\n",qRange(x,y));
        }
        else if(k==3){
            read(x);read(y);
            updSon(x,y);
        }
        else{
            read(x);
            printf("%d\n",qSon(x));
        }
    }
}

习题 

洛谷上的所有习题icon-default.png?t=N7T8https://www.luogu.com.cn/problem/list?keyword=&tag=228&page=1

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1680068.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【NR学习一】NR中的带宽、子载波间隔、PRB数量、FFT点数与采样率之间的关系

NR中的带宽、子载波间隔、PRB数量、FFT点数与采样率之间的运算关系 在5G NR&#xff08;New Radio&#xff09;系统设计中&#xff0c;带宽&#xff08;Bandwidth&#xff09;、子载波间隔&#xff08;Subcarrier Spacing, SCS&#xff09;、资源块&#xff08;Resource Block…

汇聚荣科技:如何有效为拼多多店铺引流?

在电商竞争激烈的今天&#xff0c;为拼多多店铺引流是每个店主必须面对的挑战。有效的引流策略不仅能增加店铺曝光度&#xff0c;还能提升转化率&#xff0c;促进销量增长。 一、社交媒体营销 利用微信、微博等社交平台进行推广&#xff0c;可以通过发布产品信息、用户评价和促…

Web3Tools - 助记词生成

Web3Tools - 助记词生成工具 本文介绍了一个简单的助记词生成工具&#xff0c;使用 React 和 Material-UI 构建。用户可以选择助记词的语言和长度&#xff0c;然后生成随机的助记词并显示在页面上 功能介绍 选择语言和长度&#xff1a; 用户可以在下拉菜单中选择助记词的语言&…

【半夜学习MySQL】复合查询(含多表查询、自连接、单行/多行子查询、多列子查询、合并查询等详解)

&#x1f3e0;关于专栏&#xff1a;半夜学习MySQL专栏用于记录MySQL数据相关内容。 &#x1f3af;每天努力一点点&#xff0c;技术变化看得见 文章目录 回顾基本查询多表查询自连接子查询单行子查询多行子查询多列子查询在from子句中使用子查询合并查询 回顾基本查询 下面使用…

设计非递归算法,编程:在二叉排序树中,打印关键码a, b的公共祖先。注:例,若a是b的祖先,则a不算作公共祖先。反之亦然。

二叉排序树&#xff1a; 代码&#xff1a; #include <iostream> using namespace std;// 定义二叉树节点结构 typedef struct BTNode {char show;struct BTNode* left;struct BTNode* right; } BTNode;// 非递归插入节点的函数 BTNode* insertNode(BTNode* root, char k…

Hadoop3:HDFS副本节点选择逻辑讲解

一、副本节点选择&#xff08;机架感知&#xff09; 说明 第一个副本&#xff0c;因为我们的client可能是web页&#xff0c;也可能是shell终端。 如果是web页&#xff0c;则随机选取一个节点&#xff0c;如果是shell终端&#xff0c;则选择当前shell终端所在的节点。 节点距离最…

TiDB学习2:TiDB Sever

目录 1. TiDB Server架构 2. sql语句的解析和编译 2.1 Parse ​编辑 2.2 compile 3. 行转化为KV对(聚簇表) ​编辑4. SQL 读写相关模块 4.1 DistSQL(复杂查询) 4.2 KV(简单查询) 5. 在线DDL相关模块 6. GC机制与相关模块 7. TiDB Server的缓存 8. 热点小表缓存 9. …

财富增长新途径:副业赚钱方法全攻略

探寻财富之路:多元化赚钱途径解析 在追求财富的道路上,每个人都在以自己的方式前行。然而,正如古人所云:“君子爱财,取之有道。”今天,我将为您揭示一些新颖且实用的赚钱途径,希望能为您的财富积累之路注入新的活力。 1、视频内容的创作与分享 在这个视频内容为王的时…

Benedict Evans:Ways to think about AGI思考 AGI 的方法:

​Benedict Evans本文发布于2024 年 5 月 4 日 How do we think about a fundamentally unknown and unknowable risk, when the experts agree only that they have no idea? 当专家们一致认为他们一无所知时&#xff0c;我们如何看待根本上未知和不可知的风险&#xff1f; T…

关于 vs2019 c++20 规范里的 STL 库里模板 decay_t<T>

&#xff08;1&#xff09; 这个模板&#xff0c;在库代码里非常常见。 decay 英文是“衰弱&#xff0c;消减” 的意思&#xff0c;大概能感觉到就是要简化模板参数 T 的类型&#xff0c;去掉其上的修饰符。因为常用且复杂&#xff0c;故单独列出其源码和注释。先举例其应用场景…

JumpServer堡垒机应用(v3.10.8) 下

目录 JumpServer堡垒机简单式部署与管理(v3.10.8) 上-CSDN博客 一. 资产管理 1.1创建资产 1.2 给资产主机创建用户 1.2.1 普通账户&#xff1a; 1.2.2 特权账户&#xff1a; 1.2.3 创建用户 二. 命令过滤 2.1 创建命令组 2.2 创建命令过滤 ​编辑 三. 创建资产授权 …

《Python编程从入门到实践》day29

# 昨日知识点回顾 修改折线图文字和线条粗细 矫正图形 使用内置格式 # 今日知识点学习 15.2.4 使用scatter()绘制散点图并设置样式 import matplotlib.pyplot as plt import matplotlib matplotlib.use(TkAgg)plt.style.use(seaborn-v0_8) # 使用内置格式 fig, ax plt.subpl…

软考中级-软件设计师 (十一)标准化和软件知识产权基础知识

一、标准化基础知识 1.1标准的分类 根据适用的范围分类&#xff1a; 国际标准指国际化标准组织&#xff08;ISO&#xff09;、国际电工委员会&#xff08;IEC&#xff09;所制定的标准&#xff0c;以及ISO所收录的其他国际组织制定的标准。 国家标准&#xff1a;中华人民共和…

单位个人如何向期刊投稿发表文章?

在单位担任信息宣传员一职以来,我深感肩上的责任重大。每月的对外信息宣传投稿不仅是工作的核心,更是衡量我们部门成效的重要指标。起初,我满腔热血,以为只要勤勉努力,将精心撰写的稿件投至各大报社、报纸期刊的官方邮箱,就能顺利登上版面,赢得读者的青睐。然而,现实远比理想骨…

Polylang Pro插件下载:多语言网站构建的终极解决方案

在全球化的今天&#xff0c;多语言网站已成为企业拓展国际市场的重要工具。然而&#xff0c;创建和管理一个多语言网站并非易事。幸运的是&#xff0c;Polylang Pro插件的出现&#xff0c;为WordPress用户提供了一个强大的多语言解决方案。本文将深入探讨Polylang Pro插件的功能…

基于物联网的教室人数检测系统-设计说明书

设计摘要&#xff1a; 本设计基于物联网技术&#xff0c;实现了一个教室人数检测系统。系统利用STM32单片机作为中控&#xff0c;通过红外对管检测人员进出教室&#xff0c;并实时统计应到人数和实到人数&#xff0c;同时使用OLED显示屏显示相关信息。系统还通过温湿度传感器检…

用SwitchHosts模拟本地域名解析访问

一.用SwitchHosts模拟本地域名解析访问 1.下载地址 https://download.csdn.net/download/jinhuding/89313168 2.使用截图

每周一算法:恰好经过K条边的最短路

题目描述 牛站 给定一张由 M M M 条边构成的无向图&#xff0c;点的编号为 1 ∼ 1000 1\sim 1000 1∼1000 之间的整数。 求从起点 S S S 到终点 E E E 恰好经过 K K K 条边&#xff08;可以重复经过&#xff09;的最短路。 注意: 数据保证一定有解。 输入格式 第 1 …

【动态规划五】回文串问题

目录 leetcode题目 一、回文子串 二、最长回文子串 三、分割回文串 IV 四、分割回文串 II 五、最长回文子序列 六、让字符串成为回文串的最少插入次数 leetcode题目 一、回文子串 647. 回文子串 - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/…

mysql的explain

explain可以用于select&#xff0c;delete&#xff0c;insert&#xff0c;update的statement。 当explain用于statement时&#xff0c;mysql将会给出其优化器&#xff08;optimizer&#xff09;的执行计划。 通过explain字段生成执行计划表。下面来解析这个执行计划表的每一列…