首页 > 其他分享 >旅游景点 Tourist Attractions

旅游景点 Tourist Attractions

时间:2023-07-03 10:33:19浏览次数:23  
标签:旅游景点 Tourist temp int Attractions 停留 le heap FGD

目录

题目链接

题目描述

题目描述
FGD想从成都去上海旅游。在旅途中他希望经过一些城市并在那里欣赏风景,品尝风味小吃或者做其他的有趣的事情。经过这些城市的顺序不是完全随意的,比如说FGD不希望在刚吃过一顿大餐之后立刻去下一个城市登山,而是希望去另外什么地方喝下午茶。幸运的是,FGD的旅程不是既定的,他可以在某些旅行方案之间进行选择。由于FGD非常讨厌乘车的颠簸,他希望在满足他的要求的情况下,旅行的距离尽量短,这样他就有足够的精力来欣赏风景或者是泡MM了_. 整个城市交通网络包含N个城市以及城市与城市之间的双向道路M条。城市自1至N依次编号,道路亦然。没有从某个城市直接到它自己的道路,两个城市之间最多只有一条道路直接相连,但可以有多条连接两个城市的路径。任意两条道路如果相遇,则相遇点也必然是这N个城市之一,在中途,由于修建了立交桥和下穿隧道,道路是不会相交的。每条道路都有一个固定长度。在中途,FGD想要经过K(K<=N-2)个城市。成都编号为1,上海编号为N,而FGD想要经过的N个城市编号依次为2,3,…,K+1.

举例来说,假设交通网络如下图。
image

FGD想要经过城市2,3,4,5,并且在2停留的时候在3之前,而在4,5停留的时候在3之后。那么最短的旅行方案是1-2-4-3-4-5-8,总长度为19。注意FGD为了从城市2到城市4

可以路过城市3,但不在城市3停留。这样就不违反FGD的要求了。并且由于FGD想要走最短的路径,因此这个方案正是FGD需要的。

输入格式
第一行包含3个整数N(2<=N<=20000),M(1<=M<=200000),K(0<=K<=20),意义如上所述。以下M行,每行包含3个整数X,y,z,(1<=x,y<=n,0<z<=1000);

接下来一行,包含一个整数q,表示有q个限制条件(0<=q<n)。以下q行,每行两个整数f,l(1<=l,f<=n),表示在f停留的时候要在l之前。

输出格式
只包含一行,包含一个整数,表示最短的旅行距离。

样例
样例输入

8 15 4
1 2 3
1 3 4
1 4 4
1 6 2
1 7 3
2 3 6
2 4 2
2 5 2
3 4 3
3 6 3
3 8 6
4 5 2
4 8 6
5 7 4
5 8 6
3
2 3
3 4
3 5

样例输出

19

提示

对于 \(100\%\) 的数据, 满足:

  • \(2\le n\le2\times10^4\)
  • \(1\le m\le2\times10^5\)
  • \(0\le k\le\min(20, n-2)\)
  • \(1\le p_i<q_i\le n\)
  • \(1\le l_i\le 10^3\)
  • \(2\le r_i, s_i\le k+1, r_i\not=s_i\)
  • 保证不存在重边且一定有解。

题意概括

这道题题目描述好长(

有\(N\)个点\(M\)条边的无向图,不存在重边与自环。

要求寻找一条从\(1\)到\(n\)的最短路径,而且还必须经过\(2-K+1\)并且按照\(g\)给出的要求停留在这些城市。

而且你可以选择不停留,类似于就是样例图中,我们到从\(2\)到\(3\)的时候经过了城市\(4\),但是我们选择不停留,这样的话就满足了\(2-3-4\)的停留顺序。

然后就是这个题目有一个很重要的隐藏条件,就是如果我们没有要求必须停留的点就没有所谓的停留顺序限制,也就是说\(k==0\)时\(g==0\),直接dijkstra即可

思路历程

私货:初音未来什么时候来中国开演唱会旅游啊(

1.找最短路

找最短路,没有负数和重边、自环,我想到了\(dijkstra\)

所以先粘贴一下我的与众不同的\(dijkstra\)板子,用了\(pair\)

粘贴的香甜的黄油这道题()

已经忘光力(

Miku's dijkstra code
int head[maxm<<1],t;
struct edge{
	int u,v,w;
	int next_;
};edge e[maxm<<1];
void add_edge(int u,int v,int w){
	e[++t].u=u;
	e[t].v=v;
	e[t].w=w;
	head[u]=t;
}
int dis[maxn];
bool judge[maxn];
typedef pair<int,int> strack;

void search_dijkstra(int x){		//x是终点牧场的下标
	memset(judge,false,sizeof judge);	//将judge初始化 
	memset(dis,0x3f,sizeof dis);	//将距离定义为无穷大
	dis[x]=0;						//终点距离为0
	
	priority_queue<strack,vector<strack>,greater<strack> >heap;
	//建立一个小根堆
	while(!heap.empty()){			//小根堆初始化 
		heap.pop();
	}
	heap.push({0,x});				//终点牧场入堆
	while(!heap.empty()){
		strack t=heap.top();
		heap.pop();
		int temp=t.second,distance=t.first;
		//temp是节点编号,distance是节点距离
		if(judge[temp]==true)	continue;//如果节点被访问过则跳过
		judge[temp]=true;
		for(int i=head[temp];i!=0;i=next[i]){
			int j=to[i];			//取出节点编号
			if(distance+w[i]<dis[j]){ 
				dis[j]=distance+w[i];
				heap.push({dis[j],j});
			} 
		} 
	}
}

2.设计状态

所以先考虑设置状态。

刚刚的题意概括,已经说了,显然是有三种状态:没经过、经过但未停留、停留。

没经过与经过的区别在于是否累加我们的\(dis\),而经过与停留的区别在于我们的限制条件判断

然而我们的停留是属于经过状态的,并且经过可以是多次的,但停留我们只有一次,所以我们设计状态应该在停留上下手。

我们设置\(f_{cur,i,j}\)作为dp数组,其中\(cur\)是当前状态,只有\(0\)和\(1\),通过从上一个状态的转移满足\(2-k+1\)这些点都经过,而枚举状态时要求满足所有的限制条件

\(i\)用来找到状态,\(j\)则是上一个停留的点

转移方程看起来就比较简单:

\[\begin{aligned} f_{cur,i,j}=min(f_{cur,i,j},f_{cur异或1,i的二进制去掉某一个停留点q,q}+diss_{q,j}) \end{aligned} \]

代码实现

(ps:把注释删掉,不要使用long long,可以通过洛谷的测试)

Miku's Code
#include<bits/stdc++.h>
using namespace std;

const int maxn=2e4+50,maxm=2e5+50,maxk=25;
typedef long long intx;
int n,m,k;
int g,limits[maxk];
int belong[maxk],cur;
int f[2][184757][maxk];
//f[cur][i][j],cur表示当前状态,i表示状态在容器中的索引,也表示经过这些状态的停留点,j表示上一个停留的点

int sum[(1<<maxk)+50],pbelong[(1<<maxk)+50];
vector <int> p[maxk];
/*
sum[s]数组表示停留状态中停了几个点,也就是有几个1,通过递推获得
pbelong[s]是s状态在容器里的索引
p[sum[s]]表示停留了sum[s]个点的现在的状态
*/

int head[maxm<<1],t;
struct edge{
	int u,v,w;
	int next_;
};edge e[maxm<<1];
void add_edge(int u,int v,int w){
	e[++t].u=u;
	e[t].v=v;
	e[t].w=w;
	e[t].next_=head[u];
	head[u]=t;
}
int dis[maxn],dist[maxk][2],diss[maxk][maxk];
/*
dis[]是dijkstra相关数组
dist[i][0]表示点1到点i的最短路距离,[1]表示点i到点n的最短路距离
diss[i][j]表示二进制的第j个点(点的编号为j+2)到点的编号为i的第i个点的距离
*/
bool judge[maxn];
typedef pair<int,int> strack;

void input(){
	scanf("%d %d %d",&n,&m,&k);
	int p,q,l;
	for(int i=1;i<=m;++i){
		scanf("%d %d %d",&p,&q,&l);
		add_edge(p,q,l);
		add_edge(q,p,l);
	}
	if(k!=0){							//只有k不等于0时,才可能有限制条件
		scanf("%d",&g);
		int r,s;
		for(int i=1;i<=g;++i){
			scanf("%d %d",&r,&s);
			limits[s-2]|=(1<<(r-2));
			/*
			我们将2-k+1这些必须停留的点设置状态,统一左移2位为0~k-1,与二进制数保持一致
			'|'表示只有两个位置都为0时,结果为0,这样我们得到的状态就是s城市停留之前的状态
			*/
		}
	}
}

void dijkstra(int x){
	memset(dis,0x3f,sizeof(dis));
	memset(judge,false,sizeof(judge));
	dis[belong[x]]=0;
	priority_queue<strack,vector<strack>,greater<strack> >heap;
	while(!heap.empty()){
		heap.pop();
	}
	heap.push({0,belong[x]});
	while(!heap.empty()){
		strack s=heap.top();
		heap.pop();
		int temp=s.second,distance=s.first;
		if(judge[temp]==true)	continue;
		judge[temp]=true;
		for(int i=head[temp];i;i=e[i].next_){
			int j=e[i].v;
			if(distance+e[i].w<dis[j]){
				dis[j]=distance+e[i].w;
				heap.push({dis[j],j});
			}
		}
	}
	dist[x][0]=dis[1],dist[x][1]=dis[n];
	for(int i=0;i<k;++i){
		diss[x][i]=dis[belong[i]];
	}
}

inline int lowbit(int x){
	return x&(-x);
}

void pre(){
	for(int i=0;i<k;++i){
			belong[i]=i+2;
		}
		for(int s=1;s<(1<<k);++s){
			sum[s]=sum[s&(~(lowbit(s)))]+1;
			/*
			从1开始,0的话位运算会出现错误
			lowbit(x)得到最后一个1
			'~'表示取反,'&'只有同为1才返回1
			所以这样我们就得到了停留了多少个点
			*/
			p[sum[s]].push_back(s);
			pbelong[s]=p[sum[s]].size()-1;
			/*
			pbelong[s]是s状态的一个索引,因为s是刚刚放进容器里的,所以它在容器里的位置一定是p[sum[s]].size()-1
			*/
		}
		memset(f,0x3f,sizeof(f));
		for(int i=0;i<k;++i){
			dijkstra(i);
			if(!limits[i])	f[cur][pbelong[1<<i]][i]=dist[i][0];
		}
}

void work(){
	cur=0;
	for(int i=2;i<=k;++i){
		int len=p[i].size();
		cur^=1;					//最后一位取反,f数组第一位状态只有0与1
		memset(f[cur],0x3f,sizeof(f[cur]));
		for(int e=0;e<len;++e){
			int s=p[i][e];
			for(int j=0;j<k;++j){
				
				if( (s&(1<<j)) && ( ( limits[j] & (s&(~(1<<j))) )==limits[j]) ){
					/*
					我们枚举停留了i个点,最后一个被停留的点是点j,目的是判断合法状态
					len表示停留了i个点的状态总数
					那么枚举s就表示停留了i个点的各个状态
					一定停留了第j个点所以s&(1<<j)==true
					要求满足点j停留的限制条件所以limits[j] &(s&(~(1<<j))==limits[j]
					其中,s&(~(1<<j))表示j停留之前的状态
					该状态与限制状态取&应该为限制状态
					*/
					for(int q=0;q<k;++q){
						if(j!=q	&&	(s&(1<<q)) ){
							f[cur][e][j]=min(f[cur][e][j],f[cur^1][pbelong[s&(~(1<<j))]][q]+diss[q][j]);
							//cout<<"###"<<i<<' '<<(cur^1)<<' '<<pbelong[s&(~(1<<j))]<<' '<<q<<endl;
							//cout<<"###"<<f[cur^1][pbelong[s&(~(1<<j))]][q]<<' '<<"###"<<diss[q][j]<<endl;
						}
						/*
						转移状态,q是枚举的上一个停留点
						*/
					}
				}
			}
		}
	}
}

int main(){
	input();
	if(k==0){
		belong[1]=1;			//没有前置条件,直接dijkstra
		dijkstra(1);
		printf("%d\n",dis[n]);
		return 0;
	}
	pre();
	work();
	int ans=0x3f3f3f3f;
	for(int i=0;i<k;++i){
		ans=min(ans,f[cur][0][i]+dist[i][1]);
		/*
		为什么第二维的索引是0?
		因为我们的cur表示的是当前状态,而当前状态的所有点已经从上一状态转移
		也就是说,我们现在的状态已经经过了2-k+1所有点并且满足了所有的限制条件
		现在我们这个状态,不需要在任何点停留了,所以索引是0
		*/
	}
	printf("%lld\n",ans);
	return 0;
}

标签:旅游景点,Tourist,temp,int,Attractions,停留,le,heap,FGD
From: https://www.cnblogs.com/sonnety-v0cali0d-kksk/p/17519052.html

相关文章

  • 旅游景点人物进出系统[OC项目]
    要求:展览中心有2条入场通道,在入场处需要登记入场人员的姓名,年龄以及电话。展览中心最多只能容纳100人。当展览中心满员时应当立即通知门卫不再允许人员入场。当有人员出场时才会允许人员入场,但同时在展览中心的人员不会超过100人。当展览中心关闭后,输出所有入场过的人员信息。需要......
  • GYM 101147K Touristic Trip
    首先可以看出这是一个条件概率\(P(A/B)=\frac{P(AB)}{P(B)}\),其中\(A\)事件为“满足在\(Z\)城市时寄出第\(Q\)张明信片”,\(B\)事件为“满足得到的明信片序列与给出的明信片序列相同”那只需要求出\(P(AB)\)和\(P(B)\)就能得到最终答案了首先考虑\(B\)事件发现......
  • 旅游景点信息API接口大全
    1、分享数据:“http://www.shareapi.cn/docs/api/id/127”,免费,次数1000次返回JSON示例{"SceneryID":10224,/*景区ID*/"SceneryName":"金鸡湖",/*景区名称*/"SceneryGrade":4,/*景区等级*/"SceneryAddress":"sr",/*景区地址*/"SceneryP......
  • Google Maps新增功能 为旅游景点增加3D照片
    GoogleMaps近日宣布新增Phototours功能,为全球15000万个旅游景点增加3D图片,让你免费环游世界。首先利用GoogleMaps搜索目的地,定位,如果目的图片地有Phototours链接,搜索结......
  • 【题解】CF487E Tourists / 圆方树
    概念圆方树是一种基于无向图构造的树。我们知道,圆方树最早是WC上提出的处理仙人掌的东西,用于将树上做法拓展到复杂度正确的仙人掌做法。但是一些关于点双有性质的题也......
  • Python爬虫之旅游景点评论
    目录爬取景点评论准备工作获取HTML页面解析处理worldcloudreference:爬取携程景点评论数据本博客记录一个爬取携程景点评论并制作词云的例子,并且可以很轻易地拓展到多个......
  • HTML我的家乡宁夏学生网页设计作品 dreamweaver作业静态HTML网页设计模板 宁夏旅游景
    文章目录......
  • HTML学生个人网站作业设计:旅游景点网站设计——北京故宫(9页) HTML+CSS+JavaScript 简
    ......
  • CF487E Tourists
    题意给定一张无向图,点有点权。每次可以修改一个点的点权,或者询问从\(a\)到\(b\)所有不经过重复点的路径上最小的点权是多少。Solution考虑一个点双,点双中任意两个点......
  • CF487E Tourists(Tarjan,圆方树,树链剖分,线段树)
    CF487ETourists带权无向图\(N,M\),\(Q\)次询问\(s,t\)所有不经过重复点可能路径经过的最小值的最小值。CODE每次修改一个圆点\(u\)周围的方点有点难。可是李神......