\(\mathbf{POJ\;2432}\)题解

题意

给出圆上的\(N\)个点,每个点有一个经度(大于\(0\)小于\(360\));再给出\(M\)条双向边,保证边\(x y\)仅会沿圆上较短的弧连接,且不存在边连接圆上相对的两个点的情况。

求一条从点\(1\)出发最后回到点\(1\),且能环绕圆的经过点数最少的路径。

思路

边权为\(1\)的最短路,显然可以想到BFS。但由于还要满足“能环绕圆”这一条件,我们需要加一些限制。

不妨预处理出每条边的连接的两地间的经度差作为边权,顺时针边置正权,逆时针边置负权,并在BFS的状态中加入一维,表示路径权值和。

那么在BFS时,就可以用已走过的路径权值和\(dis\)来判断一条从\(1\)出发再回到\(1\)的路径是否合法了。

不难发现,若\(dis\)不为\(0\),那么一条从\(1\)出发回到\(1\)的路径一定是合法的,反之亦然。

另外,一个状态\(f(x,dis)\)一定不会出现两次,所以用一个集合(STL set)来记录已经出现过的状态,起一般BFS中标记数组的作用。

最后,在计算答案时用一个成员变量\(step\)记录经过了几个点。

使用以上算法即可通过本题。如果想进一步优化,可以把queue改为手写queue,set改为手写Hash。

代码

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
//include STL queue
#include<queue>
//include STL set
#include<set>
//include STL pair
#include<utility>
#define N 5005
#define M 25005
using namespace std;
struct pos{
int x,dis;
int step;
}tmp,fro;
struct Edge{
int nxt,to,w;
}e[M<<1];
int n,m,tot=1,head[N],a[N];
queue<pos>q;
set<pair<int,int> >cnt;
void addedge(int x,int y,int z)
{
e[++tot]=(Edge){head[x],y,z},head[x]=tot;//顺时针
e[++tot]=(Edge){head[y],x,-z},head[y]=tot;//逆时针
}
int getdis(int x,int y)//返回两地经度差
{
int z=min(abs(x-y),360-abs(x-y));
if((x+z)%360==y)
return z;//顺时针
else
return -z;//逆时针
}
int bfs()
{
fro.x=1;fro.dis=0,fro.step=0;
q.push(fro);
while(!q.empty())
{
fro=q.front(),q.pop();
int x=fro.x,dis=fro.dis,step=fro.step;
cnt.insert(make_pair(x,dis));
for(int i=head[x],y;i;i=e[i].nxt)
{
y=e[i].to;
if(y==1 && dis+e[i].w)
return step+1;
if(cnt.find(make_pair(y,dis+e[i].w))!=cnt.end())
continue;
tmp.x=y,tmp.dis=dis+e[i].w,tmp.step=step+1;
q.push(tmp);
}
}
return -1;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
scanf("%d",a+i);
for(int i=1,x,y,z;i<=m;i++){
scanf("%d%d",&x,&y);
z=getdis(a[x],a[y]);
if(z>0) addedge(x,y,z);//顺时针
else addedge(y,x,-z);//逆时针
}
printf("%d\n",bfs());
return 0;
}
/*
3 3
0
120
240
1 2
2 3
1 3
*/

完结撒花

POJ 2432的更多相关文章

  1. poj 2432 Around the world bfs+哈希

    由于每个点的状态包含走过来的距离,所以要存二维的状态,但是状态总量太多,所以可以用哈希来搞. 那么就是bfs最短路,哈希记录状态了. #include <iostream> #includ ...

  2. poj和hdu部分基础算法分类及难度排序

    最近想从头开始刷点基础些的题,正好有个网站有关于各大oj的题目分类(http://www.pythontip.com/acm/problemCategory),所以写了点脚本把hdu和poj的一些题目 ...

  3. POJ 3370. Halloween treats 抽屉原理 / 鸽巢原理

    Halloween treats Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 7644   Accepted: 2798 ...

  4. POJ 2356. Find a multiple 抽屉原理 / 鸽巢原理

    Find a multiple Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7192   Accepted: 3138   ...

  5. POJ 2965. The Pilots Brothers' refrigerator 枚举or爆搜or分治

    The Pilots Brothers' refrigerator Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 22286 ...

  6. POJ 1753. Flip Game 枚举or爆搜+位压缩,或者高斯消元法

    Flip Game Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 37427   Accepted: 16288 Descr ...

  7. POJ 3254. Corn Fields 状态压缩DP (入门级)

    Corn Fields Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 9806   Accepted: 5185 Descr ...

  8. POJ 2739. Sum of Consecutive Prime Numbers

    Sum of Consecutive Prime Numbers Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 20050 ...

  9. POJ 2255. Tree Recovery

    Tree Recovery Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 11939   Accepted: 7493 De ...

随机推荐

  1. makefile实验三 理解make工作的基本原则

    代码简单,但测试花样多,若能回答对本博客的每个步骤的预期结果,可以说对makefile的基础掌握是扎实的. 一,当前的makefile代码 root@ubuntu:~/Makefile_Test# r ...

  2. Win32控制台、Win32项目、MFC项目、CLR控制台、CLR空项目、空项目区别

    转载:https://blog.csdn.net/zfmss/article/details/79244696 1.Win32控制台 初始代码模版以main为程序入口,默认情况下,只链接C++运行时库 ...

  3. Arduino - 串口操作函数与示例代码大全

    来源:https://blog.csdn.net/iracer/article/details/50334041 Arduino - 串口操作函数与示例代码大全 本文总结了Arduino常用串口操作函 ...

  4. Fedora version history --- kernel version

    Fedora version history https://en.wikipedia.org/wiki/Fedora_version_history     Version (Code name)[ ...

  5. dubbo使用问题

    新入职此公司, 发现公司使用的框架原来是传说中的分布式的(原谅我以前在传统公司工作,并远离浪潮久矣), 使用过程中发现各服务之间使用 dubbo 进行通信. 特地总结下遇见的坑,为以后总结经验.   ...

  6. 多测师讲解python _练习题002_高级讲师肖sir

    # 1.求出1/1+1/3+1/5--+1/99的和 # 2.用循环语句,计算2-10之间整数的循环相乘的值. # 3.用for循环打印九九乘法表 # 4.求每个字符串中字符出现的个数如:hellow ...

  7. 2020年java全套教程,此套java涵盖了pdf,java源码,项目案例,完整视频约3000G的资源

    疫情期间,百无聊赖,是不是需要充电一下,让自己更有竞争力呢?学习java一定要快呦! 废话不多说了,网盘已经爆炸了,把2006年-2020年的全部资料都发给爱学习的你吧, 希望可以改变你的命运,或者是 ...

  8. centos7下面 es7.5 搭建

    centos6 搭建 参考 https://www.cnblogs.com/php-linux/p/8758788.html 搭建linux虚拟机 https://www.cnblogs.com/ph ...

  9. 说明资源路径位置类型无法解析The type javax.servlet.http.HttpServletResponse cannot be resolved.

    导入dispatch项目后报错: 解决办法:在项目上单击鼠标右键>  Add Libraries  选择 Server Runtime,下一步  选中Apache Tomcat7 完成  切换标 ...

  10. C++ Primer第5版 第一章课后练习

    练习1.9 #include <iostream> int main() { int sum = 0, val = 50; while (val <= 100) { sum += v ...