[bzoj2125]最短路

本文详细解析了仙人掌图上的最短路径问题,通过深度优先搜索找到环并建立圆方树,利用LCA算法求解两点间的最短路径。文章深入探讨了在遇到返祖边时的特殊情况处理,提供了完整的代码实现。

传送门

仙人掌最短路,圆方树!
先dfs找环,顺带求出环的长度len,记录每个点到环内深度最小的点的最短路径有没有经过返祖边
对于每个环建一个方点,都是圆方树的基本操作啦!
求答案时对于\(dist(x,y)\),先求出\(z=lca(x,y)\),然后判断\(z\)是否是方点
1、z是方点,那么x和y的最短路径经过了一个环,那么我们对于x和y就少跳一步,跳到z的儿子处。
对应的圆方树是这样的:
1460350-20190117174311214-256894192.png
原图上是这样的:
1460350-20190117174321515-663413252.png
很显然这个时候需要分类讨论了:(x,y是跳了后的x,y)
假如x和y的最短路径都经过返祖边或者都不经过返祖边,并且不知道t-x和t-y哪条是返祖边的情况下,最短距离就是\(min(abs(dep[x]-dep[y]),min(len-dep[x]+dep[y],len-dep[y]+dep[x]))\)
否则就是只有一条经过返祖边,答案就是\(min(len-(dep[x]-dep[z])-(dep[y]-dep[z]),(dep[x]-dep[z])+(dep[y]-dep[z]))\)

2、z不是方点,那么直接算就是了,答案就是\(dep[x]+dep[y]-2*dep[z]\)(x,y是读入的x,y)

代码:

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
void read(int &x) {
    char ch; bool ok;
    for(ok=0,ch=getchar(); !isdigit(ch); ch=getchar()) if(ch=='-') ok=1;
    for(x=0; isdigit(ch); x=x*10+ch-'0',ch=getchar()); if(ok) x=-x;
}
#define rg register
const int maxn=2e4+1e3;
int n,m,q,dep[maxn],top,id,f[maxn][21],dis[maxn],len[maxn],ans;bool vis[maxn*2],w[maxn*2];
struct o{int x,id;}st[maxn];
struct oo{
    int pre[maxn*2],nxt[maxn*2],h[maxn*2],v[maxn*2],cnt;
    oo(){cnt=1;}
    inline void add(int x,int y,int z)
    {
        pre[++cnt]=y,nxt[cnt]=h[x],h[x]=cnt,v[cnt]=z;
        pre[++cnt]=x,nxt[cnt]=h[y],h[y]=cnt,v[cnt]=z;
    }
}a,b;
void dfs(int x,int fa)
{
    for(rg int i=a.h[x];i;i=a.nxt[i])
        if(a.pre[i]!=fa)
        {
            if(!dep[a.pre[i]])st[++top].x=x,st[top].id=i,dep[a.pre[i]]=dep[x]+a.v[i],dfs(a.pre[i],x),top--;
            else if(dep[a.pre[i]]<dep[x])
            {
                ++id,b.add(id,x,min(dep[x]-dep[a.pre[i]],a.v[i]));
                int g=min(dep[x]-dep[a.pre[i]],a.v[i]);len[id-n]+=a.v[i];
                if(g==a.v[i])w[x]=1;vis[i]=vis[i^1]=1;int ttt=top;
                while(a.pre[i]!=st[top].x)
                {
                    int t=st[top].x,g=min(dep[t]-dep[a.pre[i]],dep[x]-dep[t]+a.v[i]);vis[st[top].id]=vis[st[top].id^1]=1;
                    if(g==dep[x]-dep[t]+a.v[i])w[t]=1;b.add(id,t,g),len[id-n]+=a.v[st[top].id],top--;
                }
                int t=st[top].x;b.add(id,t,0);len[id-n]+=a.v[st[top].id];
                vis[st[top].id]=vis[st[top].id^1]=1;top=ttt;
            }
        }
}
void dfs1(int x,int fa)
{
    for(rg int i=1;i<=20;i++)
    {
        if(dis[x]<(1<<i))break;
        f[x][i]=f[f[x][i-1]][i-1];
    }
    for(rg int i=b.h[x];i;i=b.nxt[i])
        if(b.pre[i]!=fa)f[b.pre[i]][0]=x,dis[b.pre[i]]=dis[x]+1,dep[b.pre[i]]=dep[x]+b.v[i],dfs1(b.pre[i],x);
}
int lca(int x,int y)
{
    if(dis[x]>dis[y])swap(x,y);
    int poor=dis[y]-dis[x];
    for(rg int i=20;i>=0;i--)if(poor&(1<<i))y=f[y][i];
    for(rg int i=20;i>=0;i--)if(f[y][i]!=f[x][i])x=f[x][i],y=f[y][i];
    return x==y?x:f[x][0];
}
int jump(int x,int z)
{
    int poor=dis[x]-dis[z]-1;
    for(rg int i=20;i>=0;i--)if(poor&(1<<i))x=f[x][i];
    return x;
}
int main()
{
    read(n),read(m),read(q);id=n;
    for(rg int i=1,x,y,z;i<=m;i++)read(x),read(y),read(z),a.add(x,y,z);
    dep[1]=1,dfs(1,0);for(int i=2;i<=a.cnt;i+=2)if(!vis[i])b.add(a.pre[i],a.pre[i^1],a.v[i]);
    memset(dep,0,sizeof dep),dfs1(1,0);
    for(rg int i=1,x,y,z,l,r;i<=q;i++)
    {
        read(x),read(y);z=lca(x,y);ans=0;
        if(z>n)
        {
            l=jump(x,z),r=jump(y,z);
            ans=dep[x]-dep[l]+dep[y]-dep[r];
            if((w[l]&&w[r])||(!w[l]&&!w[r]))ans+=min(abs(dep[l]-dep[r]),min(len[z-n]-dep[l]+dep[r],len[z-n]-dep[r]+dep[l]));
            else ans+=min(len[z-n]-(dep[l]-dep[z])-(dep[r]-dep[z]),(dep[l]-dep[z])+(dep[r]-dep[z]));
        }
        else ans=dep[x]+dep[y]-dep[z]*2;
        printf("%d\n",ans);
    }
}

转载于:https://www.cnblogs.com/lcxer/p/10283615.html

源码来自:https://pan.quark.cn/s/fdd21a41d74f 正方教务管理系统成绩推送 简介 使用本项目前: 早晨睡醒看一遍教务系统、上厕所看一遍教务系统、刷牙看一遍教务系统、洗脸看一遍教务系统、吃早餐看一遍教务系统、吃午饭看一遍教务系统、睡午觉前看一遍教务系统、午觉醒来看一遍教务系统、出门前看一遍教务系统、吃晚饭看一遍教务系统、洗澡看一遍教务系统、睡觉之前看一遍教务系统 使用本项目后: 成绩更新后自动发通知到微信 以节省您宝贵的时间 测试环境 正方教务管理系统 版本 V8.0、V9.0 如果你的教务系统页面与下图所示的页面完全一致或几乎一致,则代表你可以使用本项目。 目前支持的功能 主要功能 每隔 30 分钟自动检测一次成绩是否有更新,若有更新,将通过微信推送及时通知用户。 相较于教务系统增加了哪些功能? 显示成绩提交时间,即成绩何时被录入教务系统。 显示成绩提交人姓名,即成绩由谁录入进教务系统。 成绩信息按时间降序排序,确保新的成绩始终在上方,提升用户查阅效率。 计算 计算百分制 对于没有分数仅有级别的成绩,例如”及格、良好、优秀“,可以强制显示数字分数。 显示未公布成绩的课程,即已选课但尚未出成绩的课程。 使用方法 Fork 本仓库 → 开启 工作流读写权限 → → → → → 添加 Secrets → → → → → → Name = Name,Secret = 例子 程序会自动填充 尾部的 ,因此你无需重复添加 对于部分教务系统,可能需要在 中添加 路径,如: 开启 Actions → → → 运行 程序 → → 若你的程序正常运行且未报错,那么在此之后,程序将会每隔 30 分钟自动检测一次成绩是否有更新 若你看不懂上述使用...
综合能源系统零碳优化调度研究(Matlab代码实现)内容概要:本文围绕“综合能源系统零碳优化调度研究”,提供了基于Matlab代码实现的完整解决方案,重点探讨了在高比例可再生能源接入背景下,如何通过优化调度实现零碳排放目标。文中涉及多种先进优化算法(如改进遗传算法、粒子群优化、ADMM等)在综合能源系统中的应用,涵盖风光场景生成、储能配置、需求响应、微电网协同调度等多个关键技术环节,并结合具体案例(如压缩空气储能、光热电站、P2G技术等)进行建模与仿真分析,展示了从问题建模、算法设计到结果验证的全流程实现过程。; 适合人群:具备一定电力系统、能源系统或优化理论基础,熟悉Matlab/Simulink编程,从事新能源、智能电网、综合能源系统等相关领域研究的研究生、科研人员及工程技术人员。; 使用场景及目标:①开展综合能源系统低碳/零碳调度的科研建模与算法开发;②复现高水平期刊(如SCI/EI)论文中的优化模型与仿真结果;③学习如何将智能优化算法(如遗传算法、灰狼优化、ADMM等)应用于实际能源系统调度问题;④掌握Matlab在能源系统仿真与优化中的典型应用方法。; 阅读建议:建议结合文中提供的Matlab代码与网盘资源,边学习理论模型边动手调试程序,重点关注不同优化算法在调度模型中的实现细节与参数设置,同时可扩展应用于自身研究课题中,提升科研效率与模型精度。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符  | 博主筛选后可见
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值