VIJOS-P1325 桐桐的糖果计划

JDOJ 1432 桐桐的糖果计划

https://neooj.com/oldoj/problem.php?id=1432

Description

桐桐很喜欢吃棒棒糖。他家处在一大堆糖果店的附近。 但是,他们家的区域经常出现塞车、塞人等情况,这导致他不得不等到塞的车或人走光了他才能去买到他最爱吃的棒棒糖品种。于是,他去找市长帮他修路,使得每两个糖果店之间至少有两条完全不同的路。可是市长经费有限,于是让桐桐找出哪些路被塞住后会使某些糖果店与糖果店间无法到达及最少的修路条数。你能帮助他吃到他最喜爱的糖果吗? 注:1-> 3-> 2    和  1-> 3-> 4-> 2  为不完全不同的路,即不符合题意的路。         1-> 3-> 4-> 2  和  1-> 5-> 2  为完全不同的路,即符合题意的路。

Input

输入第一行是两个数n,m(n< =5000,m< =10000) 接下来的m行,每行两个数i,j,表示i,j间有一条边连接。

Output

输出有两行。第一行为塞住后就不可以到达某些糖果店的道路条数,第二行为最少的修路条数。

Sample Input

7 7 1 2 2 3 3 4 2 5 4 5 5 6 5 7

Sample Output

3 2 
 
这道题比较容易分析,在草纸上画完图之后可以很明确地分析出来第一行就是在问整个无向图中桥的个数
桥也叫做割边,就是无向图中能够使整张图分裂成两个不连通子图的边。
第二问就是求缩点之后,需要继续添加多少条边,才能使新图变成边双联通图。
 
我们选择使用tarjan算法,那么问题就是,如何在使用tarjan求有向图强连通分量的基础上来求割边?
附带我写的tarjan模板,供下一步理解和参考。
void tarjan(int x)//tarjan算法模板(链式前向星)
{
z[++top]=x;
v[x]=;
inz[x]=;
for(int i=head[x];i;i=nxt[i])
{
int y=to[i];
if(v[y]==)
tarjan(to[i]),low[x]=min(low[x],low[to[i]);
else if(inz[to[i]]==)
low[x]=min(low[x],deep[to[i]]);
}
if(deep[x]==low[x])
{
ans++;
int t;
do
{
t=z[top--];
inz[t]=;
f[ans][++f[ans][]]=t;
}while(t!=x)
}
}

说明一下,deep数组表示时间戳,即节点x被搜索的次序编号。

low数组表示x能通过其他边回到的x的祖先,标记的祖先最小的时间戳。

深搜回溯到当前节点之后,当deep[x]==low[x],以x为根的搜索子树上的所有节点为一个强连通分量。

z数组模拟数据结构栈,inz数组表示元素是否在栈中,v数组表示元素是否被搜索过。

f数组记录每一个强连通分量里元素的个数和所有元素。

好了,下面我们介绍用tarjan算法求割边。

假如low[y]>deep[x]的时候,说明y不仅不能到达x的祖先,也不能通过另外一条边直接直接直接到达x,说明它们之间的e(x,y)便是割边。

注意处理重边。

处理重边的方式有两种,依据重边有用没用而定(而有用没用依题目而定)(等于没说)

假如重边没有用,在tarjan的时候加一个参数记录它的父亲,注意是参数,当y遇到父亲节点时不拓展回去,保证重边不会被遍历。

假如重边有用,那么增加一个参数记录边的编号,y不能通过这条边访问其父亲节点,但是却可以通过重边访问x,保证重边会被遍历到。

那么好了,我们如何来判定重边?

这里要引入对偶边的判定。

位运算异或,请小伙伴们自行补习。

void tarjan(int x,int pre)
{
v[x]=true;
deep[x]=low[x]=++tot;
for(int i=head[x];i!=-;i=next[i])
{
int y=to[i];
if( (i^)==pre ) continue;
if(!deep[y])
{
tarjan(y,i);
low[x]=min(low[x],low[y]);
if(low[y]>deep[x]) //割边;
}
else low[x]=min(low[x],deep[y]);
}
}

好了我们来针对一下这道题,第一问通过刚才的讲解已经没什么难度了。

来看第二问,这里注意我们是求缩点后需要加多少条边。

这里我们需要画图理解。

缩点后的无向图会变成一棵树,统计度为2的那些双联通分量就是叶子结点(双向边)。

然后ans=(t+1)/2;

好了讲了这么多相信大家也会对这个有一定的理解,细节的实现可以自己慢慢调。下面的代码仅供参考

#include<iostream>
#include<cstdio>
#include<cstring>
#define maxn 10010
using namespace std;
int n,m,num,head[maxn],low[maxn],deep[maxn],topt;
int top,z[maxn],inz[maxn],ans,sum,belong[maxn],rudu[maxn];
struct node
{
int v,pre;
}e[maxn*];
void add(int from,int to)
{
e[num].v=to;
e[num].pre=head[from];
head[from]=num++;
}
void tarjan(int x,int fa)
{
low[x]=deep[x]=++topt;
z[++top]=x;inz[x]=;
for(int i=head[x];i!=-;i=e[i].pre)
{
int v=e[i].v;
if(i==(fa^))
continue;
if(deep[v]==)
{
tarjan(v,i);
low[x]=min(low[x],low[v]);
if(low[v]>deep[x])ans++;
}
else if(inz[v])
low[x]=min(low[x],deep[v]);
}
if(low[x]==deep[x])
{
sum++;
while(x!=z[top])
{
belong[z[top]]=sum;
inz[z[top]]=;
top--;
}
belong[z[top]]=sum;
inz[z[top]]=;
top--;
}
}
int main()
{
scanf("%d%d",&n,&m);
int u,v;
memset(head,-,sizeof(head));
for(int i=;i<=m;i++)
{
scanf("%d%d",&u,&v);
add(u,v);
add(v,u);
}
for(int i=;i<=n;i++)
if(deep[i]==)
tarjan(i,-);
printf("%d\n",ans);
ans=;
for(int u=;u<=n;u++)
for(int i=head[u];i!=-;i=e[i].pre)
{
int v=e[i].v;
if(belong[u]!=belong[v])
rudu[belong[u]]++;
}
for(int i=;i<=sum;i++)
if(rudu[i]==)
ans++;
printf("%d\n",(ans+)/);
}

VIJOS-P1325 桐桐的糖果计划的更多相关文章

  1. Vijos P1325桐桐的糖果计划

    > P1325桐桐的糖果计划 标签:**图结构 强连通分量** 描述 桐桐很喜欢吃棒棒糖.他家处在一大堆糖果店的附近. 但是,他们家的区域经常出现塞车.塞人等情况,这导致他不得不等到塞的车或人走 ...

  2. 桐桐的糖果计划(vijos 1325)

    背景 桐桐是一个快乐的小朋友,他生活中有许多许多好玩的事,让我们一起来看看吧…… 描述 桐桐很喜欢吃棒棒糖.他家处在一大堆糖果店的附近. 但是,他们家的区域经常出现塞车.塞人等情况,这导致他不得不等到 ...

  3. vijos1325 桐桐的糖果计划

    Description 桐桐是一个快乐的小朋友,他生活中有许多许多好玩的事,让我们一起来看看吧-- 桐桐很喜欢吃棒棒糖.他家处在一大堆糖果店的附近. 但是,他们家的区域经常出现塞车.塞人等情况,这导致 ...

  4. 桐桐的贸易--WA

    问题 A: 桐桐的贸易 时间限制: 1 Sec  内存限制: 64 MB提交: 15  解决: 2[提交][状态][讨论版] 题目描述 桐桐家在Allianceance城,好友ROBIN家在Horde ...

  5. 桐桐的数学游戏(N皇后)

    题目描述 相信大家都听过经典的“八皇后”问题吧?这个游戏要求在一个8×8的棋盘上放置8个皇后,使8个皇后互相不攻击(攻击的含义是有两个皇后在同一行或同一列或同一对角线上). 桐桐对这个游戏很感兴趣,也 ...

  6. 【2018寒假集训 Day1】【位运算】桐桐的运输方案

    桐桐的运输方案(transp) [问题描述] 桐桐有 N 件货物需要运送到目的地,它们的重量和价值分别记为: 重量:W1,W2,…,Wn: 价值:V1,V2,…,Vn: 已知某辆货车的最大载货量为 X ...

  7. Java实现桐桐的数学难题

    桐桐的数学难题 题目描述 今天数学课上,桐桐学习了质数的知识:一个正整数如果只能被1和它本身整除,那么这个整数便是质数.桐桐就想:任意一个正整数是否都能分解成若干个质数相乘的形式呢?输入一个正整数n( ...

  8. Vijos P1325桐桐的糖果计划(有向图双连通分量)

    /*重边不能删 不能删 不能删...*/ #include<iostream> #include<cstdio> #include<cstring> #define ...

  9. 【年终糖果计划】跟风领一波糖果 candy.one 领取教程

    糖果领取网址(较为稳定):https://candy.one/i/1474564 用微信和QQ打开的朋友请复制到其他浏览器打开 糖果领取网址(较为稳定):https://candy.one/i/147 ...

随机推荐

  1. 修改官方发行openstack镜像的cloud-init登录方式为账号密码登录

    openstack使用的镜像多为qcow2格式,各个发行商也开源了针对openstack制作的镜像.但是这些镜像的登录方式都是注入用户名和密码的方式,就是说不能够直接通过账号和密码登录.那么如何将一个 ...

  2. Linux应试技巧

    前言:此文是为了CSP-S第二轮认证所用系统NOI-Linux的写的,但其他的Linux系统也可以按照相同或类似的方法进行配置. 配置NOI-Linux 我大约是一个月以前由于比赛的原因才开始接触NO ...

  3. Zuul中聚合Swagger的坑

    每个服务都有自己的接口,通过Swagger来管理接口文档.在服务较多的时候我们希望有一个统一的入口来进行文档的查看,这个时候可以在zuul中进行文档的聚合显示. 下面来看下具体的整合步骤以及采坑记录. ...

  4. 5G最新套餐以及对应限速标准

    原文: http://news.mydrivers.com/1/654/654529.htm 再过两天,国内的5G就要正式运营了,中国移动.联通.电信的5G预约用户亿元超过千万,三家运营商的5G套餐费 ...

  5. 基于Django的Rest Framework框架的认证组件

    0|1一.认证组件的作用 在一个程序中有一些功能是需要登录才能使用的,原生Django中的auth组件可以用来解决这个认证问题,drf框架中也有对应的认证组件来解决这个问题. models.py   ...

  6. HTML+css基础 css的几种形式 css选择器的两大特性

    3.外联样式 css选择器的两大特性 1.继承性:所有跟文本字体有关的属性都会被子元素继承.且权重是0000. 2.层叠性:就是解决选择器权重大小的一种能力,就是看那个选择器的权重大.谁的权重大听谁的 ...

  7. 【测试方法】Web测试中bug定位基本方法

    知识总结:Web测试中bug定位基本方法 涉及知识点:测试方法 在web测试过程中,经常会遇到页面中内容或数据显示错误,甚至不显示,第一反应就是BUG,没错,确实是BUG.进一步了解这个BUG的问题出 ...

  8. Autoware 培训笔记 No. 4——寻迹

    1. 前言 好多初创公司公布出来的视频明显都是寻迹的效果,不是说寻迹不好,相反可以证明,寻迹是自动技术开始的第一步. 自动驾驶寻迹:一种能够自动按照给定的路线(通常是采用不同颜色或者其他信号标记来引导 ...

  9. MySQL for OPS 06:备份恢复

    写在前面的话 人在河边走,湿鞋是早晚是事情,操作服务器,数据库也一样.谁也不知道自己哪一天控制不住自己就手贱.这时候有两个东西能救我们,一是备份,二是 bin log,bin log 前面讲了,但是 ...

  10. AspNetCore.Identity详解1——入门使用

    今年在面试的时候被问到单点登录的知识,当时支支吾吾不知该如何作答,于是面试失败.回到住所便开始上网查找资料,但苦于难于找到详尽的demo,总是无法入门.又由于我正在学习了解asp.net core,里 ...