PAT 1091 Acute Stroke


这一题的大意是说给出一个三维的矩阵,让我们找上下左右前后相互连通,面积大于等于T的数量,统计最终的面积。

看懂题目后实际上还是比较好写的。

我一开始无脑DFS搜索,找连通块

但写好代码后发现最后两个测试点段错误

DFS代码如下:

cpp 复制代码
#include<bits/stdc++.h>
#include<iostream> 
using namespace std;
int M;
int N;
int L;
//L是一个大脑中片的数量 
int T;
//M,N是每一个片的大小面积
//最大是1286*128
//T是临界值,如果一个量连接核心是少于T那么这个核心一定不被数
 //每一个玻片是由01构成 1代表stroke
 // 0代表正常
 // 因为玻片的厚度是不断的,我们仅仅不得不数1的数量去得到量
 // 然后也许几个分开的核心区域在大脑中,并且仅仅这些量是不少于T
 // 两个轴呗连接因此属于相同的区域,如果它们分享相同的方面 
int t[80][1350][130]; 
int ans;
bool flag[80][1305][135];
void dfs(int x,int y,int z,int& cnt)
{
	if(t[x][y][z]==1)
	{
		cnt++;
	}
	flag[x][y][z]=1;
	if(x+1<L&&y<M&&z<N)
	{
	
	if(t[x+1][y][z]==1&&flag[x+1][y][z]==0)
	{
		dfs(x+1,y,z,cnt);
	}
		
	}
	if(x<L&&y+1<M&&z<N)
	{
		if(t[x][y+1][z]==1&&flag[x][y+1][z]==0)
    	{
		dfs(x,y+1,z,cnt);
	    }
		
	}
	if(x<L&&y<M&&z+1<N)
	{
		if(t[x][y][z+1]==1&&flag[x][y][z+1]==0)
    	{
		dfs(x,y,z+1,cnt);
	    }
	}
	if(x-1>=0&&y<M&&z<N)
	{
		if(t[x-1][y][z]==1&&flag[x-1][y][z]==0)
    	{
		dfs(x-1,y,z,cnt);
	    }
	   
	}
	if(x<L&&y-1>=0&&z<N)
	{
		if(t[x][y-1][z]==1&&flag[x][y-1][z]==0)
    	{
		dfs(x,y-1,z,cnt);
	    }
		
	}
	if(x<L&&y<M&&z-1>=0)
	{
		if(t[x][y][z-1]==1&&flag[x][y][z-1]==0)
    	{
		dfs(x,y,z-1,cnt);
	    }
		
	}
}
int main()
{
    cin>>M>>N>>L>>T;
    for(int i=0;i<L;i++)
    {
    	for(int j=0;j<M;j++)
    	{
    		for(int k=0;k<N;k++)
    		{
    			int x;
    			cin>>x;
    			t[i][j][k]=x;
    		
			}
		
		}
	}
	
	 for(int i=0;i<L;i++)
    {
    	for(int j=0;j<M;j++)
    	{
    		for(int k=0;k<N;k++)
    		{
    		
    			if(t[i][j][k]==1&&flag[i][j][k]==0) 
    			{
    				int cnt=0;
    				dfs(i,j,k,cnt);
    			    if(cnt>=T)
    			    {
    				ans+=cnt;
				    }
				}
    			
		
    	
			}
		}
	}
	cout<<ans<<endl;
	
	
    return 0;
}

原因是数据范围过大,dfs递归栈无法开那么大,于是就段错误

因此这一题应该采用BFS,也就是用一个队列来保存

这样就不会递归爆栈了,思路很简单:

完整代码如下:

cpp 复制代码
#include<bits/stdc++.h>
#include<iostream> 
using namespace std;
int M;
int N;
int L;
//L是一个大脑中片的数量 
int T;
//M,N是每一个片的大小面积
//最大是1286*128
//T是临界值,如果一个量连接核心是少于T那么这个核心一定不被数
 //每一个玻片是由01构成 1代表stroke
 // 0代表正常
 // 因为玻片的厚度是不断的,我们仅仅不得不数1的数量去得到量
 // 然后也许几个分开的核心区域在大脑中,并且仅仅这些量是不少于T
 // 两个轴呗连接因此属于相同的区域,如果它们分享相同的方面 
int t[80][1350][130]; 
int ans;
bool flag[80][1305][135];
void dfs(int x,int y,int z,int& cnt)
{
	if(t[x][y][z]==1)
	{
		cnt++;
	}
	flag[x][y][z]=1;
	if(x+1<L&&y<M&&z<N)
	{
	
	if(t[x+1][y][z]==1&&flag[x+1][y][z]==0)
	{
		dfs(x+1,y,z,cnt);
	}
		
	}
	if(x<L&&y+1<M&&z<N)
	{
		if(t[x][y+1][z]==1&&flag[x][y+1][z]==0)
    	{
		dfs(x,y+1,z,cnt);
	    }
		
	}
	if(x<L&&y<M&&z+1<N)
	{
		if(t[x][y][z+1]==1&&flag[x][y][z+1]==0)
    	{
		dfs(x,y,z+1,cnt);
	    }
	}
	if(x-1>=0&&y<M&&z<N)
	{
		if(t[x-1][y][z]==1&&flag[x-1][y][z]==0)
    	{
		dfs(x-1,y,z,cnt);
	    }
	   
	}
	if(x<L&&y-1>=0&&z<N)
	{
		if(t[x][y-1][z]==1&&flag[x][y-1][z]==0)
    	{
		dfs(x,y-1,z,cnt);
	    }
		
	}
	if(x<L&&y<M&&z-1>=0)
	{
		if(t[x][y][z-1]==1&&flag[x][y][z-1]==0)
    	{
		dfs(x,y,z-1,cnt);
	    }
		
	}
}
int main()
{
    cin>>M>>N>>L>>T;
    for(int i=0;i<L;i++)
    {
    	for(int j=0;j<M;j++)
    	{
    		for(int k=0;k<N;k++)
    		{
    			int x;
    			cin>>x;
    			t[i][j][k]=x;
    		
			}
		
		}
	}
	
	 for(int i=0;i<L;i++)
    {
    	for(int j=0;j<M;j++)
    	{
    		for(int k=0;k<N;k++)
    		{
    		
    			if(t[i][j][k]==1&&flag[i][j][k]==0) 
    			{
    				int cnt=0;
    				dfs(i,j,k,cnt);
    			    if(cnt>=T)
    			    {
    				ans+=cnt;
				    }
				}
    			
		
    	
			}
		}
	}
	cout<<ans<<endl;
	
	
    return 0;
}
相关推荐
zxsz_com_cn4 分钟前
设备预测性维护算法核心功能有哪些?六大模块拆解智能运维的“技术骨架”
运维·算法
期末考复习中,蓝桥杯都没时间学了5 分钟前
力扣刷题13
数据结构·算法·leetcode
2201_7569890916 分钟前
C++中的事件驱动编程
开发语言·c++·算法
多米Domi01127 分钟前
0x3f 第48天 面向实习的八股背诵第五天 + 堆一题 背了JUC的题,java.util.Concurrency
开发语言·数据结构·python·算法·leetcode·面试
2301_8223776528 分钟前
模板元编程调试方法
开发语言·c++·算法
故以往之不谏43 分钟前
函数--值传递
开发语言·数据结构·c++·算法·学习方法
渐暖°1 小时前
【leetcode算法从入门到精通】5. 最长回文子串
vscode·算法·leetcode
今天_也很困1 小时前
LeetCode热题100-560. 和为 K 的子数组
java·算法·leetcode
v_for_van1 小时前
力扣刷题记录2(无算法背景,纯C语言)
c语言·算法·leetcode
2301_811232981 小时前
低延迟系统C++优化
开发语言·c++·算法