2016年蓝桥杯省赛A组c++第3题(图论)

/*
有一个含有10个格子的图形,现用0~9填充,连续的数不能填充在相邻的格子中(包括对角线相邻)。
现每个数只能填写一次,问有多少种填充方法?
0111
1111
1110 
(1表示有格子,0表示没格子) 

解题思想:深度优先遍历即可
*/ 

#include<cstdio>  
#include<cstring>  
#include<algorithm>  
#include<iostream>  
#include<string>  
#include<vector>  
#include<stack>  
#include<bitset>  
#include<cstdlib>  
#include<cmath>  
#include<set>  
#include<list>  
#include<deque>  
#include<map>  
#include<queue>
using namespace std;

int t;  //t用于存放已经填写了的格子数目 
int flag[10];  //记录数字是否选  
int map[4][5];  //格子表 
int sum=0;  //sum为可行解的数目 

int judge()//判断已经填好的表是否符合要求  
{
    int i,j;  
    for(i=0;i<3;i++)  
    {  
        for(j=0;j<4;j++)  
            if(abs(map[i][j]-map[i][j+1])==1/*左右*/||abs(map[i][j]-map[i+1][j])==1/*上下*/||abs(map[i][j]-map[i+1][j-1])==1/*左上角*/||abs(map[i][j]-map[i+1][j+1])/*右上角*/==1)  
              return 0;  
    }  
    return 1;  
}  

//dfs函数用于填表  
void dfs(int t)  
{
    int i;  
    if(t==11)//t=11,即此时表中的10个格子全部填了数  
    {  
       if(judge())  sum++;
       return;  
    }  
    for(i=0;i<=9;i++)  
    {  
        if(!flag[i])  
        {  
            flag[i]=1;  
            map[t/4][t%4]=i;  
            dfs(t+1);  
            flag[i]=0;  //若递归返回,则说明该解不可行。回溯思想。 
        }  
    }  
}  

int main()  
{  
    int i;  
    for(i=0;i<4;i++)  
    {  
        map[i][4]=1000;  
    }  
    memset(flag,0,sizeof(flag));  //记号表归零 
    for(i=0;i<5;i++)  
        map[3][i]=1000;  //画表格图 
    map[0][0]=map[2][3]=1000;  
    dfs(1);  
    printf("%d\n",sum);  
    return 0;  
}

 

 

 

tz@COI HZAU

2018/3/15

posted on 2018-03-15 20:23  tuzhuo  阅读(398)  评论(0编辑  收藏  举报