• HNUSTOJ-1674 水果消除(搜索或并查集)


    1674: 水果消除

    时间限制: 2 Sec  内存限制: 128 MB
    提交: 335  解决: 164
    [提交][状态][讨论版]

    题目描述

    “水果消除”是一款手机游戏,相信大家都玩过或玩过类似的游戏。

    下面是“水果消除”游戏的一种初始状态。

    消除的基本规则:如果有2个或2个以上的相同水果连在一起,则可以点选并消除。

    请问在某一种状态下,有几种可以点选并消除的选择方案。

    例如,对于上图所示的初始状态,将有6种点选并消除的选择方案。这6种方案依次如下图所示。

                       

       

                        

    输入

    先输入一个整数n,表示放水果的格子总数为n*n。n取3到1000之间的整数(含3和1000)。

    然后依次输入n*n个表示水果的数据,不同的水果用不同的数字表示,同一种水果用相同的数字表示。

    表示水果的数字编号从1开始,不超过100。

    输出

    在输入数据对应的初始状态下,有几种点选并消除的选择方案。

    输出方案数。

    样例输入

    6
    1 1 2 2 2 2
    1 3 2 1 1 2
    2 2 2 2 2 3
    3 2 3 3 1 1
    2 2 2 2 3 1
    2 3 2 3 2 2
    

    样例输出

    6
    

    提示

     

    来源


    这道题可以直接DFS求连通块,也可以标号以后并查集;求连通块的时候要注意包含的格子至少为两个计数才加一;
    DFS求连通块:

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    
    using namespace std;
    
    const int N = 1000 + 50;
    int mat[N][N];
    bool visit[N][N];
    int cur;
    void DFS(int i,int j,int n,int color){
        if(i<0 || j<0 || i>=n || j>=n) return;
        if(visit[i][j]) return;
        if(mat[i][j]!=color) return;
        visit[i][j] = true;
        cur++;
        DFS(i+1,j,n,color);
        DFS(i-1,j,n,color);
        DFS(i,j+1,n,color);
        DFS(i,j-1,n,color);
    }
    
    int DFS_AL(int n){
        int Count=0;
        memset(visit ,0,sizeof(visit));
        for(int i=0;i<n;i++)
        for(int j=0;j<n;j++)
        if(!visit[i][j]){
            cur=0;
            DFS(i,j,n,mat[i][j]);
            if(cur>1)
             Count++;
        }
        return Count;
    }
    
    void Input_data(int n){
        for(int i=0;i<n;i++)
            for(int j=0;j<n;j++)
            scanf("%d",&mat[i][j]);
    }
    
    int main(){
        int n;
        scanf("%d",&n);
        Input_data(n);
        printf("%d
    ",DFS_AL(n));
    }

    并查集:

    #include<cstdio>
    #include<queue>
    
    using namespace std;
    
    const int MaxSize = 1000+5;
    struct node1
    {
        int p;
        int num;
    }pre[MaxSize*MaxSize];
    
    const int dir[4][2]= {{0,1},{0,-1},{1,0},{-1,0}};
    short mat[MaxSize][MaxSize];
    bool visit[MaxSize][MaxSize];
    int Find(int x)
    {
        return pre[x].p==x?x:(pre[x].p=Find(pre[x].p));
    }
    
    void Merge(int x,int y)
    {
        x= Find(x),y=Find(y);
        if(x!=y)
        {
            pre[y].p=x;
            pre[x].num += pre[y].num;
        }
    }
    int main()
    {
        int n,i,j,k=0,ans,h;
        scanf("%d",&n);
        for(i=0; i<n; i++)
            for(j=0; j<n; j++)
            {
                scanf("%hd",&mat[i][j]);
                pre[i*n+j].p=k++;
                pre[i*n+j].num=1;
                visit[i][j]=false;
            }
        for(i=0; i<n; i++)
            for(j=0; j<n; j++)
            {
                if(!visit[i][j])
                {
                    visit[i][j] = true;
                    for(h=0; h<4; h++)
                        if(i+dir[h][0]>=0 && i+dir[h][0]<n && j+dir[h][1]>=0 && j+dir[h][1]<n )
                            if(mat[i][j] == mat[i+dir[h][0]][j+dir[h][1]])
                                Merge((i*n+j),((i+dir[h][0])*n+j+dir[h][1]));
                }
            }
        for(ans=0,i=0; i<k; i++)
            if(pre[i].p==i && pre[i].num > 1 )
                ans++;
        printf("%d
    ",ans);
    }
    
    //如有错误,还请留言指出
  • 相关阅读:
    Ionic移动应用框架
    JavaScript为iphone添加到主屏幕
    移动前端:在手机上隐藏地址工具栏
    Swipe.js支持网页点击和数字导航
    String类的概述和常用方法
    今天的学习
    集合的区别
    面向对象总结
    面向对象的封装
    方法的多态中的重写
  • 原文地址:https://www.cnblogs.com/Pretty9/p/7347723.html
Copyright © 2020-2023  润新知