题解 P5318 【【深基18.例3】查找文献】

题目传送门

根据本蒟蒻细致粗略的分析

这明显是一道水题模(du)板(liu)题

可我竟然没有一遍AC;

为更好地食用本题解需要了解以下内容

1.dfs(大法师深搜)

2.bfs(冰法师广搜)/dij最短路

3.STL vector的使用

4.STL 队列的使用

其次就是一些优化(本人在此处卡了while(1)n++次)

下面附上代码

#include<bits/stdc++.h>//万能文件头 
using namespace std;
inline int read()//二进制优化的快读 
{
    int x=0,f=1;
    char ch=getchar();
    while(ch<'0'||ch>'9')
	{
        if(ch=='-')
            f=-1;
        ch=getchar();
    }
    while(ch>='0'&&ch<='9')
	{
        x=(x<<1)+(x<<3)+(ch^48);
        ch=getchar();
    }
    return x*f;
}
int n,m;
bool b[100005];//定义b数组防止重复输出 
vector<int > a[100005];//STL大法好 
void dfs(int x,int r)//x表示所在点,r表示剩余未遍历的点 
{
	b[x]=true;//记录某点已经输出过 
	if(!r)//如果每个点都遍历过终止递归 
	{
		cout<<x<<' ';
		return ;
	}
	cout<<x<<' ';//输出 
	for(int i=0;i<a[x].size();i++)
	if(!b[a[x][i]]) dfs(a[x][i],r-1);//查找从x可以到的点,并遍历 
}
void bfs()
{
	queue<int> q;//还是STL 
	q.push(1);b[1]=true;//把1点放入队列中,并标记1点已经遍历过 
	while(!q.empty())
	{
		int s=q.front(); q.pop();//拿出队列首的那个点 
		cout<<s<<' ';//输出 
		for(int i=0;i<a[s].size();i++) if(b[a[s][i]]==false) q.push(a[s][i]),b[a[s][i]]=true;//把点s所能到达的点遍历,为防止TLE和重复输出,记录已遍历过的点 
	}
}
int main()
{
	n=read();//读入 
	m=read();
	for(int i=1;i<=m;i++) 
	{
		int x,y;
		x=read();//读入
		y=read();
		a[x].push_back(y);//建图 表示x可以到y 
	}
	for(int i=1;i<=n;i++)//把每条路所通向的点从小到大排列(题目中有要求) 
	sort(a[i].begin(),a[i].end());//快排  
	dfs(1,n);//进行深搜 从1点开始,进行n次 
	cout<<endl;//换行 
	for(int i=1;i<=n;i++) b[i]=false;//初始化 
	bfs();//进行广搜 
	return 0;
}
posted @ 2021-05-05 12:00  Varuxn  阅读(81)  评论(0编辑  收藏  举报