//Achen
#include<bits/stdc++.h>
#define For(i,a,b) for(int i=(a);i<=(b);i++)
#define Rep(i,a,b) for(int i=(a);i>=(b);i--)
#define Formylove return 0
const int N=2e6+,mod=;
typedef long long LL;
typedef double db;
using namespace std;
int n,m,Q; template<typename T> void read(T &x) {
char ch=getchar(); T f=; x=;
while(ch!='-'&&(ch<''||ch>'')) ch=getchar();
if(ch=='-') f=-,ch=getchar();
for(;ch>=''&&ch<='';ch=getchar()) x=x*+ch-''; x*=f;
} LL ans,Hdis[N];
int isnode[N];
struct VMtree{
int ecnt,fir[N],nxt[N],to[N]; LL val[N];
void add(int u,int v,LL w) {
nxt[++ecnt]=fir[u]; fir[u]=ecnt; to[ecnt]=v; val[ecnt]=w;
nxt[++ecnt]=fir[v]; fir[v]=ecnt; to[ecnt]=u; val[ecnt]=w;
//printf("%d %d %lld\n",u,v,w);
}
LL f[N];
int que[N];
void dfs(int x,int fa) {
f[x]=;
for(int i=fir[x];i;i=nxt[i]) if(to[i]!=fa)
dfs(to[i],x);
if(x<=n) {
for(int i=fir[x];i;i=nxt[i]) if(to[i]!=fa) {
int y=to[i];
ans=max(ans,f[y]+val[i]+f[x]); f[x]=max(f[x],f[y]+val[i]);
}
}
else {
int ql=,qr=;
for(int i=fir[x];i;i=nxt[i]) if(to[i]!=fa) {
int y=to[i];
f[x]=max(f[x],f[y]+val[i]);
while(ql<=qr&&Hdis[y]-Hdis[que[ql]]>Hdis[x]-(Hdis[y]-Hdis[que[ql]])) ql++;
if(ql<=qr) ans=max(ans,Hdis[y]-Hdis[que[ql]]+f[que[ql]]+f[y]);
while(ql<=qr&&f[y]-Hdis[y]>=f[que[qr]]-Hdis[que[qr]]) qr--;
que[++qr]=y;
}
for(int i=fir[x];i;i=nxt[i]) if(to[i]!=fa) {
int y=to[i];
while(ql<=qr&&Hdis[x]-(Hdis[que[ql]]-Hdis[y])>Hdis[que[ql]]-Hdis[y]) ql++;
if(ql<=qr) ans=max(ans,Hdis[x]-(Hdis[que[ql]]-Hdis[y])+f[que[ql]]+f[y]);
}
} if(isnode[x]) ans=max(ans,f[x]);
}
}V; int tot,dfn[N],a[N];
bool cmp(const int &A,const int &B) { return dfn[A]<dfn[B]; }
struct Tree {
int ecnt,fir[N],nxt[N],to[N]; LL val[N];
void add(int u,int v,LL w) {
nxt[++ecnt]=fir[u]; fir[u]=ecnt; to[ecnt]=v; val[ecnt]=w;
nxt[++ecnt]=fir[v]; fir[v]=ecnt; to[ecnt]=u; val[ecnt]=w;
//printf("%d %d %lld\n",u,v,w);
} int R[N],f[N][]; LL H[N];
void lca(int &cc,int x,int y) {
if(R[x]<R[y]) swap(x,y);
Rep(i,,) if(R[f[x][i]]>=R[y])
x=f[x][i];
if(x==y) { a[++cc]=x; return ; }
Rep(i,,) if(f[x][i]!=f[y][i])
x=f[x][i],y=f[y][i];
if(f[x][]>n) {
a[++cc]=x; a[++cc]=y;
}
a[++cc]=f[x][]; return ;
} int dfk,sz[N];
int dfs(int x,int fa) {
sz[x]=;
f[x][]=fa;
R[x]=R[fa]+;
dfn[x]=++dfk;
For(i,,) f[x][i]=f[f[x][i-]][i-];
for(int i=fir[x];i;i=nxt[i]) if(to[i]!=fa) {
H[to[i]]=H[x]+val[i];
dfs(to[i],x);
sz[x]+=sz[to[i]];
}
} bool in(int a,int b) { return dfn[a]>=dfn[b]&&dfn[a]<dfn[b]+sz[b]; }
int sta[N],top;
void solve() {
int cnt,sz,rt=;
read(cnt); sz=cnt;
For(i,,cnt) { read(a[i]); isnode[a[i]]=; }
sort(a+,a+cnt+,cmp);
For(i,,cnt-) {
lca(sz,a[i],a[i+]);
}
sort(a+,a+sz+,cmp);
cnt=unique(a+,a+sz+)-(a+);
For(i,,cnt) {
if(!rt||R[a[i]]<R[rt]) rt=a[i];
while(top&&!in(a[i],sta[top])) top--;
if(top) V.add(sta[top],a[i],H[a[i]]-H[sta[top]]);
sta[++top]=a[i];
} while(top) top--;
ans=; V.dfs(rt,);
printf("%lld\n",ans);
For(i,,cnt) { isnode[a[i]]=; V.fir[a[i]]=; } V.ecnt=;
}
}T; struct Cactus {
int ecnt,fir[N],nxt[N],to[N],val[N],vis[N];
void add(int u,int v,int w) {
nxt[++ecnt]=fir[u]; fir[u]=ecnt; to[ecnt]=v; val[ecnt]=w;
nxt[++ecnt]=fir[v]; fir[v]=ecnt; to[ecnt]=u; val[ecnt]=w;
} int dfk,dfn[N],low[N],sta[N],top;
void tarjan(int x) {
dfn[x]=low[x]=++dfk;
for(int i=fir[x];i;i=nxt[i]) if(!vis[i]) {
sta[++top]=i;
vis[i]=vis[i^]=;
if(!dfn[to[i]]) {
tarjan(to[i]);
low[x]=min(low[x],low[to[i]]);
if(low[to[i]]>=dfn[x]) {
LL hc=,nc=,cc=;
T.add(x,++tot,);
Rep(j,top,) {
hc+=val[sta[j]]; cc++;
if(sta[j]==i) break;
}
Hdis[tot]=hc;
while(top) {
int j=sta[top--];
if(to[j]!=x) {
if(cc==) T.add(tot,to[j],hc);
else T.add(tot,to[j],min(hc-nc,nc));
Hdis[to[j]]=nc;
}
nc+=val[j];
if(j==i) break;
}
}
}
else low[x]=min(low[x],dfn[to[i]]);
}
} void init() { ecnt=; tot=n; }
}C; struct hash {
int ecnt;
struct edge{ int u,v,w; }e[N];
vector<int>vc[N];
void ins(int u,int v,int w) {
if(u>v) swap(u,v);
int hs=(u*+v)%mod,up=vc[hs].size();
For(i,,up-) {
int ec=vc[hs][i];
if(e[ec].u==u&&e[ec].v==v) { e[ec].w=min(e[ec].w,w); return; }
}
e[++ecnt]=(edge){u,v,w}; vc[hs].push_back(ecnt);
} void add() {
For(i,,ecnt) C.add(e[i].u,e[i].v,e[i].w);
}
}H; int main() {
//freopen("1.in","r",stdin);
//freopen("1.out","w",stdout);
read(n); read(m);
For(i,,m) {
int u,v,w;
read(u); read(v); read(w);
C.add(u,v,w);
H.ins(u,v,w);
}
C.init(); H.add();
C.tarjan();
T.dfs(,);
read(Q);
For(i,,Q) T.solve();
Formylove;
}

写完花了1.5h找bug发现是数组开小了。

其实是因为太长实在不想看一直各种磨蹭各种骚扰其他同学。

无脑圆方树+虚树+单调队列优化dp。

如果两个人的lca是方点要把方点的两个圆儿子加进去。

http://uoj.ac/problem/87

uoj#87. mx的仙人掌的更多相关文章

  1. UOJ.87.mx的仙人掌(圆方树 虚树)(未AC)

    题目链接 本代码10分(感觉速度还行..). 建圆方树,预处理一些东西.对询问建虚树. 对于虚树上的圆点直接做:对于方点特判,枚举其所有儿子,如果子节点不在该方点代表的环中,跳到那个点并更新其val, ...

  2. SHOI2008 cactus仙人掌图 和 UOJ87 mx的仙人掌

    cactus仙人掌图 题目描述 如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人掌图(cactus).所谓简单回路就是指在图上不重复经过任何一 ...

  3. Solution -「UOJ #87」mx 的仙人掌

    \(\mathcal{Description}\)   Link.   给出含 \(n\) 个结点 \(m\) 条边的仙人掌图.\(q\) 次询问,每次询问给出一个点集 \(S\),求 \(S\) 内 ...

  4. UOJ #86 mx的组合数 (数位DP+NTT+原根优化)

    题目传送门 matthew99神犇的题解讲得非常清楚明白,跪烂Orzzzzzzzzzzzzz 总结一下,本题有很多重要的突破口 1.Lucas定理 看到n,m特别大但模数特别小时,容易想到$lucas ...

  5. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  6. UOJ#23. 【UR #1】跳蚤国王下江南 仙人掌 Tarjan 点双 圆方树 点分治 多项式 FFT

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ23.html 题目传送门 - UOJ#23 题意 给定一个有 n 个节点的仙人掌(可能有重边). 对于所有 ...

  7. UOJ#290. 【ZJOI2017】仙人掌 仙人掌,Tarjan,计数,动态规划,树形dp,递推

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ290.html 题解 真是一道好题! 首先,如果不是仙人掌直接输出 0 . 否则,显然先把环上的边删光. ...

  8. uoj#158. 【清华集训2015】静态仙人掌

    http://uoj.ac/problem/158 预处理dfs序,询问转为区间1的个数,用可持久化bitset预处理出所有可能的修改对应哪些位置,然后用一个bitset维护当前每个点的状态,修改时可 ...

  9. uoj#290. 【ZJOI2017】仙人掌(数数+仙人掌+树形dp)

    传送门 这图可以说是非常形象了2333 模拟赛的时候打了个表发现为一条链的时候答案是\(2^{n-2}\)竟然顺便过了第一个点 然后之后订正的时候强联通分量打错了调了一个上午 首先不难发现我们可以去掉 ...

随机推荐

  1. 每天一个Linux命令(52)telnet命令

        执行telnet指令开启终端机阶段作业,并登入远端主机.     (1)用法:     用法:  telnet [参数] [主机]     (2)功能:     功能:  telnet命令通常 ...

  2. HNOI2019梦游记

    \(Day_0\) 十点半开始睡觉,开始了八个小时的不眠之夜,整晚都没睡着,这状态明天肯定挂了 \(Day_1\) 开局一条鱼,计算几何只会\(20\) 还是\(T2\)的\(20\)纯暴力好打,\( ...

  3. HAproxy 配置参数详解

    HAproxy 配置参数详解 /etc/haproxy/haproxy.cfg # 配置文件 ----------------------------------------------------- ...

  4. nodejs/REPL环境命令行操作命令

    1,输入node 进入node[REPL]环境 2,按两次[ctrl+c]退出node[REPL]环境 3,上箭头会查找上次输入的命令 4,cls清屏 5,tab键会自动补全路径 6,REPL环境

  5. Qt配置USBCAN通信

    周立功为CAN通信提供了动态库:官方提供了很多相关动态库和lib等,如图 ,其中kerneldlls里还有很多动态库,还有一个配置文件.其实这么多的文件,如果我们只用到USBCAN2通信,只需要ker ...

  6. maven 一个简单项目 —— maven权威指南学习笔记(三)

    目标: 对构建生命周期 (build  lifecycle),Maven仓库 (repositories),依赖管理 (dependency management)和项目对象模型 (Project O ...

  7. sg函数的应用

    刚刚接触到sg函数突然感觉到原来可以这么好用,sg函数应该算是博弈论中比较经典的东西了.下面来说说sg函数: 从网上搜集资料终于能看懂了下面解释来自http://www.cnblogs.com/cj6 ...

  8. CentOS防火墙iptables-config的相关配置参数详解

    默认/etc/sysoncifg/iptables-config的配置内容: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 2 ...

  9. Java -- JDBC 批处理

    两种批处理方式: 采用Statement.addBatch(sql)方式实现批处理: •优点:可以向数据库发送多条不同的SQL语句. •缺点: •SQL语句没有预编译. •当向数据库发送多条语句相同, ...

  10. Python DB API 连接数据库

    Python DB API Mysql,Oracle,SqlServer 不关闭,会浪费资源.