| 网站首页 | 业界新闻 | 小组 | 威客 | 人才 | 下载频道 | 博客 | 代码贴 | 在线编程 | 编程论坛
欢迎加入我们,一同切磋技术
用户名:   
 
密 码:  
共有 783 人关注过本帖, 1 人收藏
标题:爬山算法解决 64 皇后问题,算法补全。求助~
只看楼主 加入收藏
编编程就好
Rank: 1
等 级:新手上路
帖 子:3
专家分:0
注 册:2014-7-10
收藏(1)
 问题点数:0 回复次数:2 
爬山算法解决 64 皇后问题,算法补全。求助~
#include<stdio.h>
#include<stdlib.h>
#include<time.h>

#define NUM_QUEENS 64  //皇后的个数
#define MAX_NODES 1000  //队列大小,存放最好的邻居
#define MAX_SIDE_WAYS 100 //最多连续水平移动的次数
#define DEBUG_MODE 0   //为0,不输出调试信息,否则输出调试信息
#define MAX_TRY 100  //最大random restart 次数

//队列操作
#define put_node(st) _queue[_tl++] = st  //放入新的节点
#define get_node() _queue[_hd++];   //获取队列头部节点
#define reset_queue() _hd=_tl=0  //初始化队列
#define queue_not_empty (_hd < _tl)  //判断队列是否为空
#define reset_queue_head() _hd=0  //将队列头重置为0

//状态
typedef struct state{
        int value; //该状态的价值,value越大,价值越高
        int pos[NUM_QUEENS]; //每一列皇后的位置
}state;

void generateInit(state *);
        //随机产生一个初始状态, 将状态存入输入参数指针所指向的空间
state findBestNeighbor(state);
        //找到输入状态的最好邻居
int evaluate(state);
        //评估一个状态的价值
void printSolution(state );
        //打印输出一个状态的信息
int isConflict(state );
        //判断一个状态是否是要找的结果
int try_once();
        //产生一个初始状态,并找到局部最优解;若找到了一个目标结果
        //(皇后相互不冲突)返回1,否则返回0   
int isEqual(state s1, state s2);
    //判断两个状态是否相同

//注意,全局变量的命名,均以_开始
state _queue[MAX_NODES];//队列,存放一个状态的邻居
int _hd=0, _tl = 0;//初始化队列头尾
 
void main(){
        int count = 0;//已经尝试的次数
 
        srand((unsigned int)(time(NULL)));//设置随机序列器的种子(注意程序其他地方不要再调用srand)
        
        while (count++<MAX_TRY){
                if (DEBUG_MODE){
                        printf("*********************************\n");
                        printf("begin the %d try.\n", count);
                }
                if (try_once()){
                        break;//找到了目标
                }               
        }

        if (DEBUG_MODE){
                printf("*********************");
        }      
}


/*
基于一个初始状态,进行尝试
*/
int try_once(){
        state  current, neighbor;
    generateInit(&current);//新生成一个初始状态
        
        int side_away_counter = 0;//水平移动次数
        
        if (DEBUG_MODE)
                printSolution(current);//输出初始状态

        if (!isConflict(current)){
                //初始状态即是目标
                printf("You are very lucky!\n");
                printSolution(current);
                return 1;
        }

        while(1){
                neighbor = findBestNeighbor(current);//找到一个最好的邻居
               
                if (current.value == neighbor.value){
                        //current 与 neighbor 价值相等,说明是水平移动
                        
                        side_away_counter ++ ;
                                //连续水平移动次数加1

                        if (side_away_counter>=MAX_SIDE_WAYS )  
                                break; //连续水平移动次数超出限制

                        if (isEqual(current,neighbor)){
                                //若找到的邻居和当前状态相同,跳过这个邻居
                                continue;
                        }
                }
                else if (current.value > neighbor.value){
                        break;//找不到更好的邻居,得到局部最优解
                }
                else{
                        side_away_counter = 0;
                                //找到更好的邻居,将连续水平移动次数置0
                }               

                current = neighbor;
                        //用最好的邻居更换当前状态
        }

        if (isConflict(current)){
                //局部最优非目标
                if (DEBUG_MODE){
                        printf("failed in finding a valid solution!\n");
                        printf("the current best one is:\n");
                        printSolution(current);
                }
               
                return 0;
        }
        else{
                //局部最优是目标
                printf("find a valid solution:\n");
                printSolution(current);
                return 1;
        }
         
}

//产生一个随机状态
void generateInit(state * p_st){
        int i,p;
   
        for (i=0;i<NUM_QUEENS;i++){
                p = rand() % NUM_QUEENS;
                p_st->pos[i] = p;
        }
         
        p_st->value = evaluate(*p_st);
}

 

/*
找到当前状态的最好的邻居
邻居的定义可以是固定7个列,更换一个列的皇后的位置,这样共有8*7=56个邻居
注意,若有多个邻居的价值均等于最大值,则随机从中选择一个
*/
state findBestNeighbor(state current){
    state nbr;
    return nbr;
}
 

//判断两个状态是否相同,相同返回1,不同返回0
int isEqual(state s1, state s2){
        return 0;
}

//评估一个状态的价值
//可以将价值定义为 -1 * 可以相互攻击的皇后对数
//提示,可以用三个数组,分别结算每行的皇后个数,
//每条主对角线和副对角线上的皇后个数,然后计算可以相互攻击的皇后对数
//比如,若第一行有k个皇后,则该行上可相互攻击的皇后对数为 k*(k-1)/2
int evaluate(state   st){
        return 0;
}
        
//打印一个状态的信息
void printSolution(state st){
        int i;
        for (i=0;i<NUM_QUEENS;i++){
                printf("%d ", st.pos[i]);
        }
        printf("\n");
        printf("value is %d\n", st.value);
}

//判断一个状态中是否存在某对皇后,它们可以相互攻击
//注意,若isConflict返回0,则说明该状态为目标状态
int isConflict(state st){
        int i, j;

        st.value = evaluate(st);

        return value<0;
}
搜索更多相关主题的帖子: 爬山 include 最好 信息 
2014-07-10 13:14
编编程就好
Rank: 1
等 级:新手上路
帖 子:3
专家分:0
注 册:2014-7-10
收藏
得分:0 
需要补全的地方有注释
2014-07-10 13:14
编编程就好
Rank: 1
等 级:新手上路
帖 子:3
专家分:0
注 册:2014-7-10
收藏
得分:0 
对于爬山算法我是一个头两个大,希望有大神能帮帮忙吧
对编程没天赋的孩子伤不起啊
2014-07-10 13:15
快速回复:爬山算法解决 64 皇后问题,算法补全。求助~
数据加载中...
 
   



关于我们 | 广告合作 | 编程中国 | 清除Cookies | TOP | 手机版

编程中国 版权所有,并保留所有权利。
Powered by Discuz, Processed in 0.026011 second(s), 7 queries.
Copyright©2004-2024, BCCN.NET, All Rights Reserved