上一个博文我们提到象棋运动的函数dealWithChess,但是只是说该函数完成了棋子的选择和移动功能
其实在这个函数里,在移动棋子之前,是要对棋子的移动是否合法进行判断的,我们一起来看看如何对棋子进行判断的
在dealWithChess中,关注如下代码,这个judgeAction就是判断棋子移动是否合法的入口了
这个代码是判断棋子的移动是否合规则,如果不合规则,则清除状态最后返回
bool CCMainMenu::judgeAction(int tx, int ty) { if(!m_pCurChess) { return false; } return judge(m_enCurChessType, ox, oy, tx, ty); }judgeAction本身很简单,只是判断当前有没有棋子被选中,如果有,则去判断是否可以移动
真正判断规则的函数时judge函数
bool judge(CHESS_TYPE enChessType, int x, int y, int tx, int ty) { switch(enChessType) { case CHESS_BK: return judgeBK(x, y, tx, ty); break; case CHESS_BA: return judgeBA(x, y, tx, ty); break; case CHESS_BB: return judgeBB(x, y, tx, ty); break; case CHESS_BP: return judgeBP(x, y, tx, ty); break; case CHESS_RK: return judgeRK(x, y, tx, ty); break; case CHESS_RA: return judgeRA(x, y, tx, ty); break; case CHESS_RB: return judgeRB(x, y, tx, ty); break; case CHESS_RC: case CHESS_BC: return judgeC(x, y, tx, ty); break; case CHESS_BN: case CHESS_RN: return judgeN(x, y, tx, ty); break; case CHESS_BR: case CHESS_RR: return judgeR(x, y, tx, ty); break; case CHESS_RP: return judgeRP(x, y, tx, ty); break; default: return false; } }judge函数也没有对具体规则进行判断,在judge里,分棋子类型,对棋子的移动规则进行判断。在这么多棋子的规则里,我们挑一些来解说
这个是黑色将的移动规则判断函数,直接先上代码
bool judgeBK(int x, int y, int tx, int ty) { if (tx < 0 || tx > 2 || ty < 3 || ty > 5) { return false; } int x0 = tx - x; int y0 = ty - y; for(int i = 0; i < 4; ++i) { if(k[i][0] == x0 && k[i][1] == y0) { return true; } } return false; }在这个函数中,前面两个参数是原始位置,后面两个参数指示目标位置
第一个判断
表示说如果老将的目标不在九宫内,则肯定不能移动
接下来判断两个坐标的关系,将的移动,除了目标要求在九宫内之外,两个坐标之间的关系是如此:
横坐标相减,纵坐标相减得到的值为(0,1)(0,-1)(1,0)(-1,0)这四种情况。所以有了如下代码:
其中k这个数组是在ChessDefine.cpp中定义的全局变量
const int k[4][2] = {{ 0, 1}, { 0, -1}, {-1, 0}, { 1, 0}};
通过了解将的规则算法,我们多少了解一下规则算法是怎么来的:
最主要是判断原点和目标点坐标关系,辅助一些特殊棋子的特殊要求,或者例外情况组成:
1)将
将的坐标关系:横坐标相等,纵坐标相减绝对值等于1,或者纵坐标相等,横坐标相减绝对值等于1
将的特殊要求:目标坐标坐落于九宫内
将的例外情况:如果两个老将面对面而中间没有棋子阻拦,老将可以直接飞到对方九宫吃对方老将
2)士
士的坐标关系:纵坐标和横坐标相减的绝对值都是1,
士的特殊要求:目标坐标坐落于九宫内
3)象
象的坐标关系:纵坐标和横坐标相减的绝对值都是2
象的特殊要求:象眼不能有棋子,不能过河
4)车
车的坐标关系:横坐标或者纵坐标相等
车的特殊要求:两个坐标之间不能有棋子存在
5)马
马的坐标关系:横坐标相减等于1且纵坐标相减等于2,或者反过来
马的特殊要求:马腿不能憋着
6)炮
炮的坐标关系:与车相同
炮的特殊要求:如果目标坐标没有棋子,则与车一样,否则要求中间有一个棋子
7)兵
过河前:
兵的坐标关系:纵坐标相差1,而且只能前进
兵的特殊要求:没有
过河后:
兵的坐标关系:纵坐标相差1或者横坐标相差1,不能后退
兵的特殊要求:没有
了解了一般的算法之后,我们再来看马的规则算法
judgeN是判断马的规则,这个函数通过表来对马行走点进行判断,比较高效
bool judgeN(int x, int y, int tx, int ty) { int x0 = tx - x; int y0 = ty - y; for (int i = 0; i < 8; ++i) { if (n[i][0] == x0 && n[i][1] == y0) { if(abs(x0) == 2 && g_cur_map[x + tx >> 1][y] == 0) { return true; } else if (abs(x0) == 1 && g_cur_map[x][y + ty >> 1] == 0) { return true; } return false; } } return false; }这个是马的规则,循环8次表示马的八个可能的方向,然后再通过g_cur_map判断是否是有碍脚的马腿
其他的规则,大家可以找相应的函数学习,我这里不多赘述