2 solutions

  • 1
    @ 2022-4-28 0:04:06

    可能有点长

    #include<bits/stdc++.h>
    using namespace std;
    const int N=1e3+7;
    struct st{
    	int x,y;
    };
    int n,m,sum;
    int ma[N][N],dp[N][N];
    int dx[]={0,0,1,-1},dy[]={1,-1,0,0};
    void bfs(int x,int y){
    	queue<st>q,qq;
    	q.push({x,y});
    	while(!q.empty()){
    		st t=q.front();
    		q.pop();
    		if(dp[t.x][t.y]!=0)continue;
    		sum++;
    		qq.push({t.x,t.y});
    		dp[t.x][t.y]=-1;
    		for(int i=0;i<4;i++){
    			int xx=t.x+dx[i];
    			int yy=t.y+dy[i];
    			if(xx>=1&&xx<=n&&yy>=1&&yy<=n&&dp[xx][yy]==0&&ma[t.x][t.y]!=ma[xx][yy])
    				q.push({xx,yy});
    		}
    	}
    	st t;
    	while(!qq.empty()){
    		t=qq.front();
    		qq.pop();
    		dp[t.x][t.y]=sum;
    	}
    }
    char chs[N];
    int main() {
    	scanf("%d %d",&n,&m);
    	getchar();
    	for(int i=1;i<=n;i++){
    		scanf("%s",chs);
    		for(int j=0;j<n;j++)
    			ma[i][j+1]=chs[j]-'0';
    		getchar();
    	}
    	for(int i=1;i<=n;i++)
    		for(int j=1;j<=n;j++)
    			if(dp[i][j]==0){
    				sum=0;
    				bfs(i,j);
    			}
    	int x,y;
    	while(m--){
    		scanf("%d %d",&x,&y);
    		printf("%d\n",dp[x][y]);
    	}
    	return 0;
    }
    
    • 0
      @ 2022-4-20 20:44:34

      这个题本质上是flood fill 变形+记忆化搜索 有m个读入数据,每次bfs或者dfs必然会超时 就把之前已经搜过的点标记一下,然后这个算是01相邻的一个连通块,这个实质上就是求哪一块01连通块里面元素的个数 bfs

      #include<cstring>
      #include<queue>
      using namespace std;
      typedef pair<int,int> PAII;
      int n,m;
      const int N=1010,M=1e6+10;
      char ch[N][N];
      int f[N][N],cnt[M];
      int dx[4]={0,0,1,-1};
      int dy[4]={1,-1,0,0};
      int bfs(int x,int y,int u)
      {
      queue<PAII> q;
      q.push({x,y});
      cnt[u]=1;
      f[x][y]=u;//看看是第几层
      while(q.size())
      {
      auto t=q.front();
      q.pop();
      int a=t.first,b=t.second;
      for(int i=0;i<4;i++)
      {
      int xx=a+dx[i],yy=b+dy[i];
      if(xx<0||yy<0||xx>=n||yy>=n) continue;
      if(ch[a][b]-'0'==!(ch[xx][yy]-'0')&&f[xx][yy]==-1)
      {
      q.push({xx,yy});
      cnt[u]++;
      f[xx][yy]=u;
      }
      }
      }
      return cnt[u];
      }
      int main(){
      cin>>n>>m;
      for(int i=0;i<n;i++)
      for(int j=0;j<n;j++)
      cin>>ch[i][j];
      memset(f,-1,sizeof(f));
      for(int i=0;i<m;i++)
      {
      int a,b;
      cin>>a>>b;
      a--;
      b--;
      if(f[a][b]==-1)
      {
      int t=bfs(a,b,i);
      //cout<<t<<endl;
      cnt[i]=t;
      }
      else cnt[i]=cnt[f[a][b]];
      }
      for(int i=0;i<m;i++) cout<<cnt[i]<<endl;
      return 0;
      }
      

      dfs

      #include<cstring>
      using namespace std;
      const int N=5050;
      int res[100000],f[N][N];//答案和标记 
      char ch[N][N];
      int n,m;
      void dfs(int r,int c,int z,int l)
      {
      	if(r<0||c<0||r>=n||c>=n||f[r][c]!=-1||ch[r][c]-'0'!=z) return ;
      	f[r][c]=l;//
      	res[l]++;
      	dfs(r-1,c,!z,l);
      	dfs(r+1,c,!z,l);
      	dfs(r,c-1,!z,l);
      	dfs(r,c+1,!z,l);
      }
      int main(){
      	cin>>n>>m;
      	for(int i=0;i<n;i++)
      		for(int j=0;j<n;j++)
      			cin>>ch[i][j];
      	memset(f,-1,sizeof(f));
      	for(int i=0;i<m;i++)
      	{
      		int a,b;
      		cin>>a>>b;
      		a--;
      		b--;
      		if(f[a][b]==-1) dfs(a,b,ch[a][b]-'0',i);
      		else res[i]=res[f[a][b]];
      	}
      	for(int i=0;i<m;i++) cout<<res[i]<<endl;
      	return 0;
      }
      /*
      确实可以用dfs,但如果每一次输入都来一次dfs,那必定超时
      所以考虑把每一个点的答案都记录下来,如果没被记录就来一次dfs,可以就直接用 
      */
      
      • 1

      Information

      ID
      1139
      Time
      1000ms
      Memory
      128MiB
      Difficulty
      8
      Tags
      # Submissions
      41
      Accepted
      8
      Uploaded By