题意与分析

一条很有趣的题目。给一个无向图,问它是否无环,且可以在上面找到一条线,使所有的顶点要么在线上要么不在线上但在与线相连的边上。

那么首先要确定所有点联系在一起。这个可以同判环一起处理:如果建图新加入的点同原先的点含有同一个祖先,那它肯定是环没跑了。然后遍历所有节点,看看是否拥有同一个祖先。这样就完成了两个任务。

接下来需要一点分析:我们可以证明,这条线(如果存在)一定是树的直径,或者是与树的直径长度相等(在端点差一个点那边分的叉)。为什么?如果这条线不是树的直径,那么长度一定小于直径,且树的直径与它的交点一定至少会延伸出两个点,那么这就一定会翻车,这条线一定不会满足条件。所以如果有一条线满足这个条件,那它必须得是树的直径。然后就是之前知识学习的地方,先找树的直径(这里需要记录端点,我没有采用栈的方法记录,采用了一种比较简单的方法解决),然后判断非树直径的点是否度数为1即可。

这题综合考察了树的几个性质,非常适合学习/复习。比如说我竟然忘了并查集怎么判环- -

代码

/*
* Filename: poj3310.cpp
* Date: 2018-11-05
*/
#include <iostream>
#include <cstring>
#include <vector> #define INF 0x3f3f3f3f
#define PB push_back
#define MP make_pair
#define fi first
#define se second
#define rep(i,a,b) for(repType i=(a); i<=(b); ++i)
#define per(i,a,b) for(repType i=(a); i>=(b); --i)
#define ZERO(x) memset(x, 0, sizeof(x))
#define MS(x,y) memset(x, y, sizeof(x))
#define ALL(x) (x).begin(), (x).end() #define QUICKIO \
ios::sync_with_stdio(false); \
cin.tie(0); \
cout.tie(0);
#define DEBUG(...) fprintf(stderr, __VA_ARGS__), fflush(stderr) using namespace std;
typedef int repType; const int MAXN=105;
vector<int> G[MAXN]; int pa[MAXN],n;
int find_pa(int x)
{
return pa[x]==x?x:pa[x]=find_pa(pa[x]);
}
bool union_pa(int x,int y)
{
int fx=find_pa(x),
fy=find_pa(y);
if(fx!=fy) pa[fx]=fy;
else return false;
return true;
} bool judge_cnt()
{
int cnt=0;
rep(i,1,n)
if(pa[find_pa(i)]==i) cnt++;
return cnt==1;
} int dep[MAXN];
void dfs(int x)
{
rep(i,0,int(G[x].size())-1)
if(dep[G[x][i]]==-1)
{
dep[G[x][i]]=dep[x]+1;
dfs(G[x][i]);
}
}
bool on_road[MAXN];
bool on_road_tmp[MAXN];
int maxdep=0;
void dfs2(int now, int ndep)
{
if(ndep>=maxdep)
{
memcpy(on_road,on_road_tmp,sizeof(on_road));
maxdep=ndep;
} rep(i,0,int(G[now].size())-1)
if(dep[G[now][i]]==-1)
{
dep[G[now][i]]=dep[now]+1;
on_road_tmp[G[now][i]]=true;
dfs2(G[now][i],ndep+1);
on_road_tmp[G[now][i]]=false;
}
} int
main()
{
int kase=0;
while(cin>>n)
{
if(!n) break;
rep(i,1,n) G[i].clear();
rep(i,1,n) pa[i]=i;
int e; cin>>e;
bool has_loop=false;
rep(i,1,e)
{
int u,v; cin>>u>>v;
G[u].PB(v);
G[v].PB(u);
if(find_pa(u)!=find_pa(v)) union_pa(u,v);
else has_loop=true;
}
if(e>n-1) has_loop=true;
bool ok=true;
if(has_loop || !judge_cnt())
ok=false;
if(ok)
{
MS(dep,-1);
dep[1]=0;
dfs(1);
int pnt_id=1;
maxdep=0;
rep(i,1,n) if(dep[pnt_id]<dep[i])
{
pnt_id=i;
maxdep=dep[pnt_id];
}
ZERO(on_road_tmp);
MS(dep,-1);
dep[pnt_id]=0;
on_road_tmp[pnt_id]=true;
dfs2(pnt_id, 0);
rep(i,1,n)
if(!on_road[i]&&G[i].size()!=1)
{
ok=false; break;
}
}
if(ok) cout<<"Graph "<<++kase<<" is a caterpillar."<<endl;
else cout<<"Graph "<<++kase<<" is not a caterpillar."<<endl;
}
return 0;
}

「日常训练」Caterpillar(POJ-3310)的更多相关文章

  1. 「日常训练」ZgukistringZ(Codeforces Round #307 Div. 2 B)

    题意与分析(CodeForces 551B) 这他妈哪里是日常训练,这是日常弟中弟. 题意是这样的,给出一个字符串A,再给出两个字符串B,C,求A中任意量字符交换后(不限制次数)能够得到的使B,C作为 ...

  2. 「日常训练」 Fire!(UVA-11624)

    与其说是训练不如说是重温.重新写了Java版本的代码. import java.util.*; import java.math.*; import java.io.BufferedInputStre ...

  3. 「日常训练」COMMON 约数研究(HYSBZ-1968)

    题意与分析 感谢https://www.cnblogs.com/Leohh/p/7512960.html的题解.这题话说原来不在我的训练范围,正好有个同学问我,我就拿来做做.数学果然不是我擅长的啊,这 ...

  4. 「日常训练」Mike and Feet(Codeforces Round #305 Div. 2 D)

    题意 (Codeforces 548D) 对一个有$n$个数的数列,我们要求其连续$x(1\le x\le n)$(对于每个$x$,这样的连续group有若干个)的最小数的最大值. 分析 这是一道用了 ...

  5. 「日常训练」「小专题·图论」 Frogger (1-1)

    题意 分析 变形的dijkstra. 分析题意之后补充. 代码 // Origin: // Theme: Graph Theory (Basic) // Date: 080518 // Author: ...

  6. 「日常训练」 Mike and Fun (CFR305D2B)

    题意(CodeForces 548B) 每次对01矩阵中的一位取反,问每次操作后,单列中最长连续1的长度. 分析 非常非常简单,但是我当时训练的时候WA了四次...无力吐槽了,人间 不值得.jpg 代 ...

  7. 「日常训练」Common Subexpression Elimination(UVa-12219)

    今天做的题目就是抱佛脚2333 懂的都懂. 这条题目干了好几天,最后还是参考别人的代码敲出来了,但是自己独立思考了两天多,还是有收获的. 思路分析 做这条题我是先按照之前的那条题目(The SetSt ...

  8. 「日常训练」Magic Stones(CodeForces-1110E)

    题意 给定两个数组c和t,可以对c数组中的任何元素变换\(c_i\)​成\(c_{i+1}+c_{i-1}-c_i\)​,问c数组在若干次变换后能否变换成t数组. 分析 这种魔法题目我是同样的没做过. ...

  9. 「日常训练」Jongmah(Codeforces-1110D)

    题意 你有n个数字,范围[1, m],你可以选择其中的三个数字构成一个三元组,但是这三个数字必须是连续的或者相同的,每个数字只能用一次,问这n个数字最多构成多少个三元组? 分析 根据官方Editori ...

随机推荐

  1. 2019.1.4 SSH框架整合步骤(一)

    SSH整合 1.三大框架整合原理 Spring与Struts2整合就是将Action对象交给Spring容器负责创建 Spring与Hibernate整合就是将sessionFactory交给Spri ...

  2. SQL Server遍历所有表统计行数

    DECLARE CountTableRecords CURSOR READ_ONLY FOR SELECT sst.name, Schema_name(sst.schema_id) FROM sys. ...

  3. ListView 中嵌套 GridView

    1.主布局文件 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:andr ...

  4. 【题解】洛谷P1065 [NOIP2006TG] 作业调度方案(模拟+阅读理解)

    次元传送门:洛谷P1065 思路 简单讲一下用到的数组含义 work 第i个工件已经做了几道工序 num 第i个工序的安排顺序 finnish 第i个工件每道工序的结束时间 need 第i个工件第j道 ...

  5. Extjs header column 自定义排序规则

    Extjs 的表格自带排序功能,这个功能在大部分情况下能够满足我们的需求,但是在某种情况下,例如IP排序,默认情况下,按照字符串进行排序, 此时我们需要自定义排序规则,这个时候就需要我们重写方法了, ...

  6. 解决ssh连接linux系统特别慢的问题

    新安装的centos系统,发现ssh连接很慢,因为是测试环境,对安全的要求不高,所以完全可以更快的连接,下面一起来解决这个问题. 一.分析主要原因: 1.SSH的反向DNS解析会消耗大量时间 2.GS ...

  7. Vue--- 一点车项目 连接数据库 数据使用

    Vue--- 一点车项目  连接数据库 数据使用 后台服务器 返回数据 处理 created 这个钩子在实例被创建之后被调用: async created(){ // 分类 catelist { le ...

  8. JavaScript document对象

    1.document对象是window对象的子对象,可直接使用,多用于获取HTML页面元素 2.document对象属性 a) alinkColor活动链接颜色 b) linkColor文本链接颜色 ...

  9. IPC进程间通信---共享内存

    共享内存 共享内存:共享内存就是分配一块能被其它进程访问的内存.每个共享内存段在内核中维护着一个内部结构shmid_ds, 该结构定义在头文件linux/shm.h中,其结构如下: struct sh ...

  10. Linux 学习第三天

    一.常用命令 1.diff A.diff -q 源文件 目标文件  (快速比较文件是否相同) 2.ifconfig.nmcli (查看配置信息) 命令输入注意: Windows 查看网卡配置信息输入命 ...