1.26学习总结

连通性判断

DFS连通性判断步骤:

1.从图上任意一点u开始遍历,标记u已经走过

2.递归u的所有符合连通条件的邻居点

3.递归结束,找到了的所有与u的连通点,就是一个连通块

4.然后重复这个步骤找到所有的连通块

BFS连通性判断步骤:

1.从图上任意一点u开始遍历,入队

2.弹出队首u,并且u已经被标记过,开始搜索u的邻居点放到队列中

3.弹出队首,重复步骤寻找连通点

题:全球变暖

你有一张某海域 ���NxN 像素的照片,"."表示海洋、"#"表示陆地,如下所示:

.......

.##....

.##....

....##.

..####.

...###.

.......

其中"上下左右"四个方向上连在一起的一片陆地组成一座岛屿。例如上图就有 2 座岛屿。

由于全球变暖导致了海面上升,科学家预测未来几十年,岛屿边缘一个像素的范围会被海水淹没。具体来说如果一块陆地像素与海洋相邻(上下左右四个相邻像素中有海洋),它就会被淹没。

例如上图中的海域未来会变成如下样子:

.......

.......

.......

.......

....#..

.......

.......

请你计算:依照科学家的预测,照片中有多少岛屿会被完全淹没。

输入

第一行包含一个整数 N (1≤N≤1000)。

以下 �N 行 �N 列代表一张海域照片。

照片保证第 1 行、第 1 列、第 �N 行、第 �N 列的像素都是海洋。、

输出一个整数表示答案。

这道题,主要是通过找到第一个陆地,然后遍历整个岛屿,判断其中是否存在高低,如果有则这个岛屿就不会被淹没

1.DFS

#include <bits/stdc++.h>
using namespace std;
const int N=1010;
char mp[N][N];
int vis[N][N];
int flag;
void dfs(int x,int y)
{
	vis[x][y]=1;
	int dir[4][2]={
  {0,-1},{0,1},{1,0},{-1,0}};
	if(mp[x][y]=='.')return;
	if (mp[x+1][y]=='#' && mp[x][y+1]=='#' && mp[x-1][y]=='#' && mp[x][y-1]=='#')flag=1;
	for (int i=0;i<4;++i)
	{
		int  tx=x+dir[i][0],ty=y+dir[i][1];
		if (vis[tx][ty]==0 && mp[tx][ty]=='#')
			dfs(tx,ty);
	}
}
int main()
{
	int n;
	cin>>n;
	for (int i=0;i<n;++i)
	{
		cin>>mp[i];
	}
	int ans=0;
	for (int i=0;i<n;++i)
	{
		for (int j=0;j<n;++j)
		{
      if (mp[i][j]=='#' && vis[i][j]==0)
			{flag=0;
      dfs(i,j);
      if (flag==0)ans++;}
		}
	}
	cout<<ans;
}
2.BFS
#include <bits/stdc++.h>
using namespace std;
const int N=1010;
char mp[N][N],flag;
int vis[N][N];
void bfs(int x ,int y)
{
  queue<pair<int, int> > q;
  q.push({x,y});
  vis[x][y]=1;
  int dir[4][2]={
  {0,-1},{0,1},{1,0},{-1,0}};
  while (q.size())
  {
    pair<int,int>p;
    p=q.front();
    q.pop();
    int tx=p.first,ty=p.second;
    if (mp[tx+1][ty]=='#' && mp[tx][ty+1]=='#' && mp[tx-1][ty]=='#' && mp[tx][ty-1]=='#')flag=1;
    for (int i=0;i<4;++i)
    {
      int nx=tx+dir[i][0],ny=ty+dir[i][1];
      if (mp[nx][ny]=='#' && vis[nx][ny]==0)
      {
        vis[nx][ny]=1;
        q.push({nx,ny});
      }
    }
  }
}
int main()
{
  int n;
  cin>>n;
  int ans=0;
  for (int i=0;i<n;++i)cin>>mp[i];
  for (int i=0;i<n;++i)
  {
    for (int j=0;j<n;++j)
    {
      if (vis[i][j]==0 && mp[i][j]=='#')
      {
        
        bfs(i,j);
        if (flag==0)ans++;
        flag=0;
      }
    }
  }
  cout<<ans;
}

判重

由于DFS和BFS都是暴力的搜索方法,所以很容易超时,所以DFS需要剪枝,BFS需要判重

题:跳蚱蜢https://www.lanqiao.cn/problems/642/learning/?page=1&first_category_id=1&problem_id=642

题目描述

本题为填空题,只需要算出结果后,在代码中使用输出语句将所填结果输出即可。

如下图所示: 有 99 只盘子,排成 11 个圆圈。 其中 88 只盘子内装着 88 只蚱蜢,有一个是空盘。 我们把这些蚱蜢顺时针编号为 11 ~ 88。

图片描述

每只蚱蜢都可以跳到相邻的空盘中, 也可以再用点力,越过一个相邻的蚱蜢跳到空盘中。

请你计算一下,如果要使得蚱蜢们的队形改为按照逆时针排列, 并且保持空盘的位置不变(也就是 1−81−8 换位,2−72−7换位,...),至少要经过多少次跳跃?

这道题,就 引入了map来记录经过的字符串,下次在变成这个串的时候就可以直接跳过

#include <bits/stdc++.h>
using namespace std;
struct node{
	node(){}
	node(string ss,int tt){s=ss,step=tt;}
	string s;
	int step;
};
int cnt=0;
queue<node>q;
map<string,bool>mp;
void solve()
{
	while (!q.empty())
	{
		node now=q.front();
		q.pop();
		string s=now.s;
		int step=now.step;
		if (s=="087654321")
		{
			cout<<step<<endl;
			cout<<cnt<<endl;
			break;
		}
		int i;
		for (i=0;i<10;++i)
		{
			if (s[i]=='0')break;
		}
		for (int j=i-2;j<=i+2;++j)
		{
			int k=(j+9)%9;
			if (k==i)continue;
			string news=s;
			char temp=news[i];
			news[i]=news[k];
			news[k]=temp;
			cnt++;
			if (!mp[news])
			{
				mp[news]=true;
				q.push(node(news,step+1));
			}
		}
	}
}
int main()
{
	string s="012345678";
	q.push(node(s,0));
	mp[s]=true;
	solve();
}

剪枝

剪格子https://www.lanqiao.cn/problems/211/learning/?page=1&first_category_id=1&problem_id=211

题目描述

如下图所示,3 x 3 的格子中填写了一些整数。

我们沿着图中的红色线剪开,得到两个部分,每个部分的数字和都是 60。

本题的要求就是请你编程判定:对给定的 �×�m×n 的格子中的整数,是否可以分割为两个部分,使得这两个区域的数字和相等。

如果存在多种解答,请输出包含左上角格子的那个区域包含的格子的最小数目。

如果无法分割,则输出 0。

输入描述

输入描述

程序先读入两个整数 �,�m,n 用空格分割 (�,�<10)(m,n<10),表示表格的宽度和高度。

接下来是 �n 行,每行 �m 个正整数,用空格分开。每个整数不大于 104104。

输出描述

在所有解中,包含左上角的分割区可能包含的最小的格子数目。

这道题的思路很简单,就是找到所有格子总和的一半,所以当此时相加的总和超过一半的时候,就可以直接退出,达到剪枝的效果

#include <bits/stdc++.h>
using namespace std;
int a[11][11];
int m,n,sum,minx=100005;
int vis[11][11];
void dfs(int x,int y,int s,int l)
{
	if (s==sum/2 )
	{
		if (minx>l && vis[0][0])minx=l;
		return;	
	}
	if (s>sum/2)return;
	int dir[4][2]={
  {0,1},{1,0},{0,-1},{-1,0}};
	for (int i=0;i<4;++i)
	{
		int tx=x+dir[i][0],ty=y+dir[i][1];
		if (vis[tx][ty]==1 || tx<0 || ty<0 || tx>=m || ty>=n)continue;
		vis[tx][ty]=1;
		dfs(tx,ty,s+a[tx][ty],l+1);
		vis[tx][ty]=0;
	}
	return ;
}
int main()
{
	cin>>m>>n;
	for (int i=0;i<n;++i)
	{
		for(int j=0;j<m;++j)
		{
			cin>>a[i][j];
			sum+=a[i][j];
		}
	}
	vis[0][0]=1;
	dfs(0,0,a[0][0],1);
	cout<<(minx==100005?0:minx);
	return 0;
}

相关推荐

  1. 2.12学习总结

    2024-01-31 20:56:02       40 阅读
  2. 学习总结16

    2024-01-31 20:56:02       27 阅读
  3. 学习总结12

    2024-01-31 20:56:02       23 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-01-31 20:56:02       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-01-31 20:56:02       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-01-31 20:56:02       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-01-31 20:56:02       20 阅读

热门阅读

  1. DAY35:贪心算法part4、860\406\452

    2024-01-31 20:56:02       50 阅读
  2. PHP雪花算法

    2024-01-31 20:56:02       33 阅读
  3. MBA十日读--营销章节 笔记

    2024-01-31 20:56:02       25 阅读
  4. Aspose-words详细功能介绍

    2024-01-31 20:56:02       35 阅读
  5. 同花顺C# MACD的计算方式

    2024-01-31 20:56:02       27 阅读
  6. 电销机器人的未来发展趋势

    2024-01-31 20:56:02       36 阅读