vijos:P1155集合位置(次短路)

描述

每次有大的活动,大家都要在一起“聚一聚”,不管是去好乐迪,还是避风塘,或者汤姆熊,大家都要玩的痛快。还记得心语和花儿在跳舞机上的激情与释放,还记得草草的投篮技艺是如此的高超,还记得狗狗的枪法永远是‘S‘……还有不能忘了,胖子的歌声永远是让我们惊叫的!!
今天是野猫的生日,所以想到这些也正常,只是因为是上学日,没法一起去玩了。但回忆一下那时的甜蜜总是一种幸福嘛。。。

但是每次集合的时候都会出现问题!野猫是公认的“路盲”,野猫自己心里也很清楚,每次都提前出门,但还是经常迟到,这点让大家很是无奈。后来,野猫在每次出门前,都会向花儿咨询一下路径,根据已知的路径中,总算能按时到了。

现在提出这样的一个问题:给出n个点的坐标,其中第一个为野猫的出发位置,最后一个为大家的集合位置,并给出哪些位置点是相连的。野猫从出发点到达集合点,总会挑一条最近的路走,如果野猫没找到最近的路,他就会走第二近的路。请帮野猫求一下这条第二最短路径长度。

格式

输入格式

第一行是两个整数n(1<=n<=200)和m,表示一共有n个点和m条路,以下n行每行两个数xi,yi,(-500<=xi,yi<=500),代表第i个点的坐标,再往下的m行每行两个整数pj,qj,(1<=pj,qj<=n),表示两个点相通。

输出格式

只有一行包含一个数,为第二最短路线的距离(保留两位小数),如果存在多条第一短路径,则答案就是第一最短路径的长度;如果不存在第二最短路径,输出-1。

输入:

3 3
0 0
1 1
0 2
1 2
1 3
2 3

输出:

2.83

思路:利用dijkstra求最短路记录路径,再将最短路中的边枚举删除,利用spfa求最短路。(删边枚举)

#include<cstdio>
#include<cstring>
#include<queue>
#include<vector>
#include<cmath>
#include<algorithm>
using namespace std;
const int MAXN=205;
const double INF=100000000.0;
typedef pair<int,double> P;
struct Node{
    int x,y;
}pos[MAXN];
struct Edge{
    int to;
    double w;
    Edge(){}
    Edge(int to,double w)
    {
        this->to=to;
        this->w=w;
    }
};
int pre[MAXN];
vector<Edge> mp[MAXN];
double dist(int x,int y,int x1,int y1)
{
    return sqrt((x-x1)*(x-x1)+(y-y1)*(y-y1));
}
int n,m;
double d[MAXN];
void dijkstra(int s)
{
    for(int i=1;i<=n;i++)
    {
        d[i]=INF;
    }
    d[s]=0;
    priority_queue<P,vector<P>,greater<P> > que;
    que.push(P(0,s));

    while(!que.empty())
    {
        P now = que.top();que.pop();
        int u=now.second;
        if(d[u]<now.first)    continue;
        for(int i=0;i<mp[u].size();i++)
        {
            Edge e=mp[u][i];
            if(d[e.to]>d[u]+e.w)
            {
                d[e.to]=d[u]+e.w;
                pre[e.to]=u;
                que.push(P(d[e.to],e.to));
            }
        }
    }
}
double d1[MAXN];
int vis[MAXN];
void spfa(int s,int u,int v)
{
    for(int i=1;i<=n;i++)
    {
        d1[i]=INF;
        vis[i]=0;
    }

    queue<int> que;
    que.push(s);
    d1[s]=0;
    vis[s]=1;
    while(!que.empty())
    {
        int now=que.front();que.pop();
        vis[now]=0;
        for(int i=0;i<mp[now].size();i++)
        {
            Edge e=mp[now][i];
            if((now==u&&e.to==v)||(now==v&&e.to==u))
                continue;
            if(d1[e.to]>d1[now]+e.w)
            {
                d1[e.to]=d1[now]+e.w;
                if(!vis[e.to])
                {
                    vis[e.to]=1;
                    que.push(e.to);
                }
            }
        }
    }
}

int main()
{

    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
    {
        scanf("%d%d",&pos[i].x,&pos[i].y);
    }
    for(int i=1;i<=m;i++)
    {
        int u,v;
        scanf("%d%d",&u,&v);
        double w=dist(pos[u].x,pos[u].y,pos[v].x,pos[v].y);
        mp[u].push_back(Edge(v,w));
        mp[v].push_back(Edge(u,w));

    }
    dijkstra(1);

    double mn=INF;
    int pr=n;
    while(pre[pr]!=0)
    {
        spfa(1,pr,pre[pr]);
        mn=min(mn,d1[n]);
        pr=pre[pr];
    }

    if(fabs(mn-INF)<0.000001)
        printf("-1\n");
    else
        printf("%.2f\n",mn);

    return 0;
}    
时间: 2024-10-07 06:39:33

vijos:P1155集合位置(次短路)的相关文章

P2176路障与P1186玛丽卡与P1491集合位置全面胜利

P2176 [USACO14FEB]路障Roadblock P1186 玛丽卡 P1491 集合位置 怎么又做到三倍经验,五年计划都超额完成了 这几道题像极了. 想起来不难吧,要让边改变之后与原来的最短路差值最大,就把最短路上的边改了呗. 用一个队列来记录最短路上的边,然后枚举这个队列里的元素,依次改变,刷出最大值. 代码有点不好写,这次我加上注释了. 不要问我为什么变量名怎么这么长,最近沉迷代码补全. P2176的代码: #include<bits/stdc++.h> using names

spfa求次短路

思路:先算出每个点到1的最短路d1[i],记录下路径,然后枚举最短路上的边 删掉之后再求一遍最短路,那么这时的最短路就可能是答案. 既然这样为甚么不用A*求次短路呢?因为A*求次短路处理不了无向图啊,他会来回的走.ε=(′ο`*)))唉 上菜: 集合位置 题目描述 每次有大的活动,大家都要在一起"聚一聚",不管是去好乐迪,还是避风塘,或者汤姆熊,大家都要玩的痛快.还记得心语和花儿在跳舞机上的激情与释放,还记得草草的投篮技艺是如此的高超,还记得狗狗的枪法永远是'S'--还有不能忘了,胖子

洛谷 P1466 [USACO2.2]集合 Subset Sums

题目描述 对于从1到N (1 <= N <= 39) 的连续整数集合,能划分成两个子集合,且保证每个集合的数字和是相等的.举个例子,如果N=3,对于{1,2,3}能划分成两个子集合,每个子集合的所有数字和是相等的: {3} 和 {1,2} 这是唯一一种分法(交换集合位置被认为是同一种划分方案,因此不会增加划分方案总数) 如果N=7,有四种方法能划分集合{1,2,3,4,5,6,7},每一种分法的子集合各数字和是相等的: {1,6,7} 和 {2,3,4,5} {注 1+6+7=2+3+4+5}

20151230训练题解(最短路+拓扑排序)

http://acm.hust.edu.cn/vjudge/contest/view.action?cid=103223#problem/B 这道题用经典的dijsktra算法,大概思路就是用dist[x]存储x到已经确定集合的最短路,n次循环到这个这个最小值,然后更新其他点到新集合的最短路即对应的dist[] 1 #include <iostream> 2 #include <cstring> 3 #include <cstdio> 4 #include <al

集合划分(subset)

对于从1到N(1<=N<=3)的连续整数集合,划分成两个子集合,使得每个集合的数字之和相等.举个例子,如果N=3,对于{1,2,3}能划分成两个子集合,他们每个的所有数字和是相等的:{3} and {1,2} 这是唯一的一种分法(交换集合位置被认为是同一种划分方案,因此不会增加划分方案总数).如果N=7,有四种方法能划分集合{1,2,3,4,5,6,7},每一种分法的子集合各数字和是相等的: {1,6,7} and {2,3,4,5};{2,5,7} and {1,3,4,6}; {3,4,7

集合的定义,操作及运算 (Python)

集合的定义: 集合和列表([  ]) 与  字典 ( {    }) 不同,没有特别的特别的语法格式.可以使用set () 创建. 集合同字典一样是无序的.也是不具有重复性的.因此可以把列表变成集合进行去重. 集合具有特别的关系性能,交集,并集,差集等. # hanbb come on! list1 = [1,4,5,7,3,6,7,9] # 列表 s1 = set(list1) # remove repeat,such as 7 ; 自动去重 (列表转化为集合) s2 = set([1,6,0

Java集合知识总结详解

两大体系:Collection,Map 一.Collection: List 接口 : List:里面对象全部是有序的(通过三种方法来遍历) ArrayList,LinkedList,Vertor ArrayList:本质上 ArrayList 里维护的就是动态可变长度的数组. 常用方法:  增:add(4) 对象 位置+对象  集合 集合+位置.  删:remove(2)下标删对象,直接删对象.  改:set(1) 下标  对象.  查:get() 下标得对象, indexOf()对象得下标.

01-05-01-1【Nhibernate (版本3.3.1.4000) 出入江湖】延迟加载及其class和集合(set、bag等)的Lazy属性配置组合对Get和Load方法的影响

这篇文章 http://ayende.com/blog/3988/nhibernate-the-difference-between-get-load-and-querying-by-id One of the more common mistakes that I see people doing with NHibernate is related to how they are loading entities by the primary key. This is because the

CODEVS 2055 集合划分

[题目描述] 对于从1到N(1<=N<=39)的连续整数集合,划分成两个子集合,使得每个集合的数字之和相等. 举个例子,如果N=3,对于{1,2,3}能划分成两个子集合,他们每个的所有数字和是相等的:{3} and {1,2} 这是唯一的一种分法(交换集合位置被认为是同一种划分方案,因此不会增加划分方案总数). 如果N=7,有四种方法能划分集合{1,2,3,4,5,6,7},每一种分法的子集合各数字和是相等的: {1,6,7} and {2,3,4,5};{2,5,7} and {1,3,4,