百科问答小站 logo
百科问答小站 font logo



在中国象棋中,最少用多少只马才能控制住整个棋盘?(马控棋盘)? 第1页

  

user avatar   zhang-ming-76-89 网友的相关建议: 
      

最少22。

只用22只马的最优解应该是260种,考虑有棋盘上下对称的结果的话就是130种。

用的是DFS+剪枝。跑了两次DFS,一次求出最优解需要用到的马的个数是22,第二次找出22只马所有可能的解。跑了一个多小时。(其实跑一次应该就可以了,懒得改了直接复制了代码里第一次的DFS= =)

主要的剪枝是考虑落子后,如果继续下去此点之前的某个点无论如何也不能控制,则返回。

代码和解法如下(只传前30种解法吧,太多了。。。

):

#include <stdio.h>

#include <memory.h>

#include <stdlib.h>


int fi,fj;

typedef struct Point{

int x;

int y;

}Point;

int check_w(int * path,int i,int j){

if((i>=0 && i<10) && (j>=0 && j<9)){

if(path[i*9+j]){

return 1;

}

}

return 0;

}


void in_reg(int ** use,Point * newp,int * newpc,int i,int j){

if((i>=0 && i<10) && (j>=0 && j<9)){

if(use[i][j]){

return;

}

newp[*newpc].x=i;

newp[*newpc].y=j;

(*newpc)++;

use[i][j]=1;

}

}

void to_use(int ** use,Point * newp,int * newpc,int i,int j){

in_reg(use,newp,newpc,i,j);

in_reg(use,newp,newpc,i-2,j-1);

in_reg(use,newp,newpc,i-1,j-2);

in_reg(use,newp,newpc,i+1,j-2);

in_reg(use,newp,newpc,i+2,j-1);

in_reg(use,newp,newpc,i-2,j+1);

in_reg(use,newp,newpc,i-1,j+2);

in_reg(use,newp,newpc,i+1,j+2);

in_reg(use,newp,newpc,i+2,j+1);

}

void dfs2path(int ** use,int usc,int * path,int minuse,int nowuse,int nowi,int nowj){

int i,j,k,l,m,newpc;

Point newP[9];

if(nowuse> minuse){

return;

}

if(usc==90){

if(nowuse== minuse){

k=0;

for(i=0;i<10;i++){

for(j=0;j<9;j++){

if(path[i*9+j]){

printf("M");

k++;

}else{

printf(".");

}

}

printf(" ");

}

printf(" ");

}

return;

}

for(i=nowi;i<10;i++){

for(j=0;j<9;j++){

if(i==nowi && j<nowj){

continue;

}

if(check_w(path,i-1,j)|| check_w(path,i+1,j)||check_w(path,i,j-1) ||check_w(path,i,j+1)){

continue;

}

if(i>nowi+4){

return;

}


for(l=0;l<i-2;l++){

for(m=0;m<9;m++){

if(!use[l][m]){

return;

}

}

}

l=i-2;

if(l>=0){

for(m=0;m<j-1;m++){

if(!use[l][m]){

return;

}

}

}

newpc=0;

to_use(use,newP,&newpc,i,j);

path[i*9+j]=1;

if(j>=8){

dfs2path(use,usc+newpc,path,minuse,nowuse+1,i+1,0);

}else{

dfs2path(use,usc+newpc,path,minuse,nowuse+1,i,j+1);

}

path[i*9+j]=0;

for(k=0;k<newpc;k++){

use[newP[k].x][newP[k].y]=0;

}

}

}

}

void dfs(int ** use,int usc,int * path,int * minuse,int nowuse,int nowi,int nowj){

int i,j,k,l,m,newpc;

Point newP[9];

if(nowuse>= *minuse){

return;

}

if(usc==90){

if(nowuse< *minuse){

*minuse=nowuse;

}

return;

}

for(i=nowi;i<10;i++){

for(j=0;j<9;j++){

if(i==nowi && j<nowj){

continue;

}

if(i>nowi+4){

return;

}

if(check_w(path,i-1,j)|| check_w(path,i+1,j)||check_w(path,i,j-1) ||check_w(path,i,j+1)){

continue;

}

for(l=0;l<i-2;l++){

for(m=0;m<9;m++){

if(!use[l][m]){

return;

}

}

}

l=i-2;

if(l>=0){

for(m=0;m<j-1;m++){

if(!use[l][m]){

return;

}

}

}

newpc=0;

to_use(use,newP,&newpc,i,j);

path[i*9+j]=1;

if(j>=8){

dfs(use,usc+newpc,path,minuse,nowuse+1,i+1,0);

}else{

dfs(use,usc+newpc,path,minuse,nowuse+1,i,j+1);

}

path[i*9+j]=0;

for(k=0;k<newpc;k++){

use[newP[k].x][newP[k].y]=0;

}

}

}

}


int main(){

int ** use,i,j,minuse,psc,usc;

int * ps;

minuse=90;

use=(int **)malloc(sizeof(int *)*10);

ps=(int *) malloc(sizeof(int)*90);

memset(ps,0,sizeof(int)*90);

for(i=0;i<10;i++){

use[i]=(int * )malloc(sizeof(int)*9);

memset(use[i],0,sizeof(int)*9);

}

dfs(use,0,ps,&minuse,0,0,0);

printf("minuse:%d ",minuse);

memset(ps,0,sizeof(int)*90);

for(i=0;i<10;i++){

memset(use[i],0,sizeof(int)*9);

}

dfs2path(use,0,ps,minuse,0,0,0);

return 0;

}






  

相关话题

  编程到底难在哪里? 
  我们高中数学为什么不重视算法?高中学的数列,三角函数,求导,圆锥曲线相关问题的解法和算法有什么关系? 
  如果你有很多枚鸡蛋,和一个n层高的楼,你想知道鸡蛋的抗摔能力。如何在消耗蛋数与实验速度之间找到最优解? 
  如何评价caffe作者贾扬清加入Facebook? 
  运维是计算机行业里技术含量最低的岗位吗? 
  计算机语言是如何做到靠0和1就表达出这么多东西的? 
  腾讯面试题,如何寻找一个数组里面唯一不重复的元素?要求时间复杂度o(n)和空间复杂度o(1)? 
  为什么有些算法工程师从来不谈业务,不谈解决问题,不谈价值挖掘,开口闭口就是算法模型,炼丹调参工程化? 
  不查表,如何求 sin36 度? 
  大学计算机专业有非编程的吗? 

前一个讨论
如何看待用盗版 Windows 和办公软件,却在 Steam 上花了几千元买游戏的人?
下一个讨论
有哪些“不要相信歌词,他们为了押韵什么都做得出来”的例子?





© 2025-01-19 - tinynew.org. All Rights Reserved.
© 2025-01-19 - tinynew.org. 保留所有权利