题目:https://loj.ac/problem/2737

相连的关系形成若干环 / 内向基环树 。如果不是只有一个环的话,就得断开一些边使得图变成若干链。边的边权是以它为出边的点的点权。

基环树的树的部分可以 DP 或者贪心,贪心就是只在分叉处断边。

对于每个环,先做出 f[ i ][ 0/1 ] 表示环上这个点 不向下 / 向下 延伸链的代价,然后在环上 DP 。

方法就是先指定 tot -> 1 的边不选, DP 一番,再制定 tot -> 1 的边选, DP 一番。

如果指定 tot -> 1 的边选,要注意不能连出一个环。所以不仅有 g[ i ][ 0/1 ] 表示 “可以向右延伸” / “无要求” , 还要有一个 [ 0/1 ] 表示 “之前有没有断过” 。不过不用有 g[ i ][ 1 ][ 1 ] ,因为 “无要求” 就是表示 i -> i+1 得断开,这样就一定 “断过” 了;所以就记 g[ i ][ 2 ] 表示 “之前断过,现在可以向右延伸” 。 g[ tot ][ 2 ] 就是指定 tot -> 1 的边选的时候加给答案的贡献。

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
int rdn()
{
int ret=;bool fx=;char ch=getchar();
while(ch>''||ch<''){if(ch=='-')fx=;ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return fx?ret:-ret;
}
ll Mn(ll a,ll b){return a<b?a:b;}
ll Mx(ll a,ll b){return a>b?a:b;}
const int N=1e5+; const ll INF=1e14+;
int n,hd[N],xnt,to[N<<],nxt[N<<],w[N<<],tp[N],tw[N],tot;
int cd[N],cnt,col[N],tim,dfn[N],low[N],sta[N],top;
ll ans,f[N][],g[N][]; bool vis[N],ins[N];
void add(int x,int y,int z)
{
to[++xnt]=y;nxt[xnt]=hd[x];hd[x]=xnt;w[xnt]=z;cd[x]++;
}
void tarjan(int cr)
{
dfn[cr]=low[cr]=++tim; sta[++top]=cr; ins[cr]=;
for(int i=hd[cr],v;i;i=nxt[i])
if(!dfn[v=to[i]])tarjan(v),low[cr]=Mn(low[cr],low[v]);
else if(ins[v])low[cr]=Mn(low[cr],dfn[v]);
if(dfn[cr]==low[cr])
{
cnt++; int siz=;
do{
ins[sta[top]]=; col[sta[top]]=cnt; siz++;
}while(sta[top--]!=cr);
if(siz==)col[cr]=, cnt--;
}
}
void dfs(int cr)
{
ins[cr]=;
for(int i=hd[cr],v;i;i=nxt[i])
if(!col[v=to[i]])
{
dfs(v=to[i]);
ll t0=f[cr][]+f[v][]+w[i];
ll t1=Mn(f[cr][]+f[v][],f[cr][]+f[v][]+w[i]);
f[cr][]=t0; f[cr][]=t1;
}
}
void ini_dfs(int cr,int st)
{
dfs(cr); tp[++tot]=cr;
for(int i=hd[cr],v;i;i=nxt[i])
if(col[v=to[i]])
{tw[tot]=w[i]; if(v!=st)ini_dfs(v,st);}
}
void solve(int cr)
{
tot=; ini_dfs(cr,cr);
g[][]=tw[tot]+f[tp[]][]; g[][]=tw[tot]+f[tp[]][];//f[tp[]]!!
for(int i=;i<=tot;i++)
{
int cr=tp[i];//
g[i][]=Mn(g[i-][],g[i-][]+tw[i-])+f[cr][];
g[i][]=Mn(g[i-][],g[i-][]+tw[i-])+f[cr][];
}
ll ret=g[tot][];
g[][]=f[tp[]][]; g[][]=f[tp[]][]; g[][]=g[][]=INF;
for(int i=;i<=tot;i++)
{
int cr=tp[i];
g[i][]=Mn(g[i-][],g[i-][]+tw[i-])+f[cr][];
g[i][]=Mn(g[i-][],g[i-][]+tw[i-])+f[cr][];
g[i][]=Mn(g[i-][],g[i-][]+tw[i-])+f[cr][];
}
ans+=Mn(ret,g[tot][]);
}
int main()
{
n=rdn();
for(int i=,d,z;i<=n;i++)
{ d=rdn(); z=rdn(); add(d,i,z);}
bool fg=;for(int i=;i<=n;i++)if(cd[i]!=){fg=;break;}
if(!fg)
{
int cr=;while(!vis[cr]){vis[cr]=;cr=to[hd[cr]];}
for(int i=;i<=n;i++)if(!vis[i]){fg=;break;}
if(!fg){puts("");return ;}
}
for(int i=;i<=n;i++)if(!dfn[i])tarjan(i);
for(int i=;i<=n;i++)
if(!ins[i]&&col[i])solve(i);
printf("%lld\n",ans); return ;
}

LOJ 2737 「JOISC 2016 Day 3」电报 ——思路+基环树DP的更多相关文章

  1. Loj #2731 「JOISC 2016 Day 1」棋盘游戏

    Loj 2731 「JOISC 2016 Day 1」棋盘游戏 JOI 君有一个棋盘,棋盘上有 \(N\) 行 \(3\) 列 的格子.JOI 君有若干棋子,并想用它们来玩一个游戏.初始状态棋盘上至少 ...

  2. LOJ 2736 「JOISC 2016 Day 3」回转寿司 ——堆+分块思路

    题目:https://loj.ac/problem/2736 如果每个询问都是 l = 1 , r = n ,那么每次输出序列的 n 个数与本次操作的数的最大值即可.可以用堆维护. 不同区间的询问,可 ...

  3. LOJ #2731. 「JOISC 2016 Day 1」棋盘游戏(dp)

    题意 JOI 君有一个棋盘,棋盘上有 \(N\) 行 \(3\) 列 的格子.JOI 君有若干棋子,并想用它们来玩一个游戏.初始状态棋盘上至少有一个棋子,也至少有一个空位. 游戏的目标是:在还没有放棋 ...

  4. [LOJ#2732] 「JOISC 2016 Day 2」雇佣计划

    参考博文 (不过个人感觉我讲的稍微更清楚一点) 题目就是让我们求图中满足数值大于等于B的连通块数量 然后我们可以尝试转换为求连通块两端所产生的“谷”的数量,显然一个连通块对谷可以贡献2的答案,最终答案 ...

  5. LOJ #2831. 「JOISC 2018 Day 1」道路建设 线段树+Link-cut-tree

    用 LCT 维护颜色相同连通块,然后在线段树上查一下逆序对个数就可以了. code: #include <cstdio> #include <algorithm> #inclu ...

  6. 题解 「JOISC 2016 Day 3」电报

    题目传送门 题目大意 给出一个\(n\)个点\(n\)条边的图,每个点有且仅有一个出边,改变每条边都会有对应的花费.求最小的花费使得整个图强连通. 思路 很显然,最后的图就是一个环.那我们要求的答案实 ...

  7. 「JOISC 2016 Day 1」棋盘游戏

    「JOISC 2016 Day 1」棋盘游戏 先判无解:第1,3行有连续的空格或四个角有空格. 然后可以发现有解的情况第1,3行可以在任意时间摆放. 对于某一列,若第2行放有棋子,那么显然可以把棋盘分 ...

  8. loj 2392「JOISC 2017 Day 1」烟花棒

    loj 答案显然满足二分性,先二分一个速度\(v\) 然后显然所有没有点火的都会往中间点火的人方向走,并且如果两个人相遇不会马上点火,要等到火快熄灭的时候才点火,所以这两个人之后应该在一起行动.另外有 ...

  9. 「JOISC 2016 Day 3」回转寿司

    https://loj.ac/problem/2736 题解 挺有意思的题. 考虑这种操作不好直接维护,还有时限比较长,所以考虑分块. 考虑一个操作对整个块的影响,无非就是可能把最大的拿走,再把新的元 ...

随机推荐

  1. android小程序-电子钢琴-多点触控

    我在第一篇博客<android小程序-电子钢琴-滑动连续响应>中实现了一个简单地7键钢琴,这几天把它又完善了一下,增加了多点触控,按键也增加了一个低音区和一个高音区,使得又可以多弹一点简单 ...

  2. python 一次创建多级目录

    python 一次创建多级目录沙漠骆驼:qq音乐import osos.mkdirs('/home/user/app')

  3. Maven Speed Up

    收录架构 proxy代理仓库 不支持仓库搜索功能 收录版本 所有版本 更新时间 每24小时更新一次 使用说明 一.在maven软件中使用 以Maven 3.5.2为例: 打开maven配置文件 ./a ...

  4. Day4作业及默写

    1,写代码,有如下列表,按照要求实现每一个功能 li = ["alex", "WuSir", "ritian", "barry&q ...

  5. ipython output logging:使用日志记录输出

    通常使用ipython的%logstart日志功能时,仅开启输入的记录. 例如在ipython中开启%logstart后,记录的日志文件内容如下: #!/usr/bin/env python # 20 ...

  6. 在线播放Video/PDF/JPG

    Label1.Text = Play(url, , ); public string Play(string url, int width, int height) { string strTmp = ...

  7. 使用Redis做为MySQL的缓存

    OS: Ubuntu 16.04.4 x64 更新并安装必要的工具 apt update && apt upgrade -y && apt dist-upgrade - ...

  8. HDU 6140 17多校8 Hybrid Crystals(思维题)

    题目传送: Hybrid Crystals Problem Description > Kyber crystals, also called the living crystal or sim ...

  9. [转]Peer-to-Peer Communication Across Network Address Translators

    Peer-to-Peer Communication Across Network Address Translators Bryan Ford Massachusetts Institute of ...

  10. 【转载】 AI会议的总结(by南大周志华)

    原文地址: https://blog.csdn.net/LiFeitengup/article/details/8441054 最近在查找期刊会议级别的时候发现这篇博客,应该是2012年之前的内容,现 ...