题解 \(by\;zj\varphi\)

一道树剖的题

先对于原图求出一棵最小生成树,求出来的这棵树中的边定为树边,其它边叫非树边

那么对于一条非树边,它要成为最小生成树上的边,权值只能为连接它两个端点的树上路径最小值减 \(1\),因为,他和这些树边构成了一个环,最小生成树要在环上去掉最大的一条边,所以只要这条边比树上路径最大值小即可

而对于树边,发现它要比它参与构成的所有环中的非树边权值要小,因为非树边在其环中权职一定为最大的,若树边比它大,就不会在最小生成树中了

对于这两种操作,我们都可以用树剖来做,就是一个板子

注意:维护最小值的线段树维护的是边下放到它终点后的点权,记得在 \(lca\) 处不要多修改

Code
#include<bits/stdc++.h>
#define ri register signed
#define p(i) ++i
using namespace std;
namespace IO{
char buf[1<<21],*p1=buf,*p2=buf;
#define gc() p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++
template<typename T>inline void read(T &x) {
ri f=1;x=0;register char ch=gc();
while(ch<'0'||ch>'9') {if (ch=='-') f=0;ch=gc();}
while(ch>='0'&&ch<='9') {x=(x<<1)+(x<<3)+(ch^48);ch=gc();}
x=f?x:-x;
}
}
using IO::read;
namespace nanfeng{
#define FI FILE *IN
#define FO FILE *OUT
template<typename T>inline T cmax(T x,T y) {return x>y?x:y;}
template<typename T>inline T cmin(T x,T y) {return x>y?y:x;}
static const int N=7e4+7,M=1e5+7,INF=2e9+7;
int first[N],fa[N],W[N],hs[N],siz[N],dfn[N],top[N],bc[N],ans[M],dep[N],vis[M],tot,t=1,n,m,opt,num;
struct edge{int v,nxt,w;}e[N<<1];
struct edgei{int u,v,w,id;}G[M],AS[M];
inline int cmp(edgei e1,edgei e2) {return e1.w<e2.w;}
inline void add(int u,int v,int w) {
e[t].v=v,e[t].w=w,e[t].nxt=first[u],first[u]=t++;
e[t].v=u,e[t].w=w,e[t].nxt=first[v],first[v]=t++;
}
int find(int x) {return fa[x]==x?x:fa[x]=find(fa[x]);}
inline void kruskal() {
sort(G+1,G+m+1,cmp);
for (ri i(1);i<=m;p(i)) {
int u=G[i].u,v=G[i].v,w=G[i].w;
if (find(u)==find(v)) continue;
vis[G[i].id]=1;
fa[find(u)]=find(v);
add(u,v,w);
}
}
struct Segmx{
#define ls(x) (x<<1)
#define rs(x) (x<<1|1)
#define up(x) T[x].mx=cmax(T[ls(x)].mx,T[rs(x)].mx);
struct segmenttree{int mx;}T[N<<2];
void build(int x,int l,int r) {
if (l==r) return (void)(T[x].mx=W[bc[l]]);
int mid(l+r>>1);
build(ls(x),l,mid);
build(rs(x),mid+1,r);
up(x);
}
int querymx(int x,int l,int r,int lt,int rt) {
if (l<=lt&&rt<=r) return T[x].mx;
int mid(lt+rt>>1),res(0);
if (l<=mid) res=cmax(res,querymx(ls(x),l,r,lt,mid));
if (r>mid) res=cmax(res,querymx(rs(x),l,r,mid+1,rt));
return res;
}
}Tx;
struct Segmn{
#define ls(x) (x<<1)
#define rs(x) (x<<1|1)
struct segmenttree{int mn,lz;segmenttree(){mn=INF;}}T[N<<2];
inline void down(int x) {
if (!T[x].lz) return;
T[ls(x)].mn=cmin(T[x].mn,T[ls(x)].mn);
T[rs(x)].mn=cmin(T[x].mn,T[rs(x)].mn);
T[ls(x)].lz=T[rs(x)].lz=1;
T[x].lz=0;
}
void update(int x,int k,int l,int r,int lt,int rt) {
if (l<=lt&&rt<=r) {
T[x].mn=cmin(T[x].mn,k);
return (void)(T[x].lz=1);
}
down(x);
int mid(lt+rt>>1);
if (l<=mid) update(ls(x),k,l,r,lt,mid);
if (r>mid) update(rs(x),k,l,r,mid+1,rt);
}
int querymn(int x,int k,int l,int r) {
if (l==r) return T[x].mn;
int mid(l+r>>1);
down(x);
if (k<=mid) return querymn(ls(x),k,l,mid);
else return querymn(rs(x),k,mid+1,r);
}
}Tn;
void dfs1(int x,int f) {
fa[x]=f;
siz[x]=1;
for (ri i(first[x]),v;i;i=e[i].nxt) {
if ((v=e[i].v)==f) continue;
W[v]=e[i].w,dep[v]=dep[x]+1;
dfs1(v,x);
siz[x]+=siz[v];
if (siz[v]>siz[hs[x]]) hs[x]=v;
}
}
void dfs2(int x,int tp) {
bc[dfn[x]=p(tot)]=x;
top[x]=tp;
if (hs[x]) dfs2(hs[x],tp);
for (ri i(first[x]),v;i;i=e[i].nxt) {
if ((v=e[i].v)==fa[x]||v==hs[x]) continue;
dfs2(v,v);
}
}
inline int solve(int u,int v,int w) {
ri res(0);
while(top[u]!=top[v]) {
if (dep[top[u]]<dep[top[v]]) swap(u,v);
res=cmax(res,Tx.querymx(1,dfn[top[u]],dfn[u],1,n));
Tn.update(1,w,dfn[top[u]],dfn[u],1,n);
u=fa[top[u]];
}
if (dep[u]<dep[v]) swap(u,v);
if (dfn[u]>dfn[v]) {
res=cmax(res,Tx.querymx(1,dfn[v]+1,dfn[u],1,n));
Tn.update(1,w,dfn[v]+1,dfn[u],1,n);
}
return res-1;
}
inline int main() {
// FI=freopen("nanfeng.in","r",stdin);
// FO=freopen("nanfeng.out","w",stdout);
read(n),read(m),read(opt);
for (ri i(1);i<=n;p(i)) fa[i]=i;
for (ri i(1);i<=m;p(i))
read(G[i].u),read(G[i].v),read(G[i].w),G[i].id=i,AS[i]=G[i];
kruskal();
dfs1(1,0),dfs2(1,1);
Tx.build(1,1,n);
for (ri i(1);i<=m;p(i)) {
int u=AS[i].u,v=AS[i].v,w=AS[i].w;
if (!dfn[u]) continue;
if (vis[i]) continue;
ans[i]=solve(u,v,w);
}
for (ri i(1);i<=m;p(i)) {
int u=AS[i].u,v=AS[i].v,w=AS[i].w,lw=dep[u]<dep[v]?v:u;
if (!dfn[u]) continue;
if (vis[i]) ans[i]=Tn.querymn(1,dfn[lw],1,n)-1;
}
for (ri i(1);i<=m;p(i)) printf("%d ",ans[i]==INF-1?-1:ans[i]);
return 0;
}
}
int main() {return nanfeng::main();}

NOIP 模拟 $17\; \rm weight$的更多相关文章

  1. NOIP 模拟 $17\; \rm 时间机器$

    题解 \(by\;zj\varphi\) 一道贪心的题目 我们先将节点和电阻按左边界排序,相同的按右边界排序 对于每一个节点,我们发现,选取左边界小于等于它的电阻中右边界大于它且最接近的它的一定是最优 ...

  2. NOIP 模拟 $17\; \rm 世界线$

    题解 \(by\;zj\varphi\) 此题经简单观察可发现,一个点的贡献就是这个点所能到的点减去它的出度 那么我们就可以暴力搜索,但是显然会超时,所以我们可以使用一个黑科技 \(\rm bitse ...

  3. NOIP模拟17.9.22

    NOIP模拟17.9.22 前进![问题描述]数轴的原点上有一只青蛙.青蛙要跳到数轴上≥

  4. NOIP模拟17.9.21

    NOIP模拟17.9.21 3 58 145 201 161.5 样例输出21.6 数据规模及约定对于40% 的数据,N <= 20对于60% 的数据,N <= 1000对于100% 的数 ...

  5. NOIP模拟 17.8.20

    NOIP模拟17.8.20 A.阶乘[题目描述]亲爱的xyx同学正在研究数学与阶乘的关系,但是他喜欢颓废,于是他就制作了一个和阶乘有关系的数学游戏:给出两个整数 n,m,令 t = !n,每轮游戏的流 ...

  6. NOIP模拟 17.8.18

    NOIP模拟17.8.18 A.小菜一碟的背包[题目描述]Blice和阿强巴是好朋友但萌萌哒Blice不擅长数学,所以阿强巴给了她一些奶牛做练习阿强巴有 n头奶牛,每头奶牛每天可以产一定量的奶,同时也 ...

  7. NOIP模拟 17.8.15

    NOIP模拟17.8.15 A 债务文件名 输入文件 输出文件 时间限制 空间限制debt.pas/c/cpp debt.in debt.out 1s 128MB[题目描述]小 G 有一群好朋友,他们 ...

  8. NOIP模拟 17.8.17

    NOIP模拟17.8.17 A 小 G 的字符串文件名 输入文件 输出文件 时间限制 空间限制str.pas/c/cpp str.in str.out 1s 128MB[题目描述]有一天,小 L 给小 ...

  9. NOIP模拟 17.8.16

    NOIP模拟17.8.16 A 债务文件名 输入文件 输出文件 时间限制 空间限制debt.pas/c/cpp debt.in debt.out 1s 128MB[题目描述]小 G 有一群好朋友,他们 ...

随机推荐

  1. 小程序textarea设置maxlength后不是你想的那样简单

    可能很多小伙伴们.看见这个标题后; 觉得作者是一个标题党. textarea设置maxlength后, 限制用户输入的字符呗! 还能怎么样呢? 恭喜你,说对了一半. 之前我也一直是这样想的. 知道今天 ...

  2. 20道Java实习生笔试面试选择题(内附答案解析)

    ​1.以下对继承的描述错误的是(A) A.Java中的继承允许一个子类继承多个父类 B.父类更具有通用性,子类更具体 C.Java中的继承存在的传递性 D.当实例化子类时会递归调用父类中的构造方法 解 ...

  3. 初步了解Unix系统的I/O模式

    I/O模式 对于一次IO访问(以read举例),数据会先被拷贝到操作系统内核的缓冲区中,然后才会从操作系统内核的缓冲区拷贝到应用程序的地址空间. 当一个read操作发生时,它会经历两个阶段: 等待数据 ...

  4. 「ARC103D」 Distance Sums

    「ARC103D」 Distance Sums 传送门 水题. 首先如果让你求树上的节点 \(i\) 到其它所有节点的距离和,这是非常简单的,这就是非常常规的换根 \(\texttt{DP}\). 那 ...

  5. python08篇 发邮件和异常处理

    一.发邮件 import yamail smtp = yamail.SMTP(host='smtp.qq.com', # 改成自己邮箱的服务器即可,其他的比如smtp.163.com user='65 ...

  6. Spark编程基础_RDD编程

    RDD(Resilient Distributed Dataset)叫做弹性分布式数据集,是Spark中最基本的数据抽象,它代表一个不可变.可分区.里面的元素可并行计算的集合.RDD具有数据流模型的特 ...

  7. 求数组的子数组之和的最大值IV

    在之前的基础上又安排了二维数组的,在课上一开始是理解错要求了,简单的以为用循环数组就能解决,但是却忽视了子数组是否能构成矩形,之后课下和同学们讨论,主要是多重遍历,但是我还是没搞明白怎么构成新的二维数 ...

  8. C++11 左值引用和右值引用与引用折叠和完美转发

    1.左值与右值 最感性的认识. 当然,左值也是可以在右边的. 左值是可以被修改的,右值不能. 当然取地址也是. 生存周期一般左值会比右值的长,一般右值都计算时产生的无名临时对象,存在时间比较短. 下面 ...

  9. AntDesignBlazor 学习笔记

    AntDesignBlazor是基于 Ant Design 的 Blazor 实现,开发和服务于企业级后台产品.我的 Blazor Server 学习就从这里开始,有问题可以随时上 Blazor 中文 ...

  10. javascript里面的document.getElementById

    一.getElementById:获取对 ID 标签属性为指定值的第一个对象的引用,它有 value 和 length 等属性 1.获取当前页面的值input标签值:var attr1=documen ...