编程之美 ---> 1.2中国象棋将帅问题

上图,将帅不能碰面,列出将帅不碰面的所有可能情况,要求:程序只能用一个只有8位的变量(#define这样的就不算了)
为了更加符合程序员的口味,给将帅位置编号如下:
0--1--2
| | |
3--4--5
| | |
6--7--8
输出将帅所有可能情况,(0,1)等等
--------------------------------------------------------------------------------------------------------------------------------------------------
思路一:程序框架应该如下,关键是怎么样用一个变量表示将帅位置.
遍历将位置
遍历帅位置
如果将帅位置不矛盾,输出
只用8位的变量,要表示2个数,看来可以试试位操作每个表示位置的变量4个位,4位可以表示0-15,足以!
方法一:
用位字段(为此还复习了位操作...http://www.cnblogs.com/jiayith/p/3500367.html)
如下:
#include <iostream> using namespace std; /*定义一个结构体,两个标签,各自4位,总共一个变量的话为8位*/
struct myBits
{
unsigned char a:; //4位可以表示0-15的值了
unsigned char b:;
}; int main(void)
{
myBits my; for (my.a=;my.a<=;my.a++)
{
for (my.b=;my.b<=;my.b++)
{
if (my.a%!=my.b%)
{
cout<<"("<<(int)my.a<<","<<(int)my.b<<")"<<endl;
}
}
} cin.get();
return ;
}
方法二:
既然都用了位字段,那位操作也应该没什么问题
用一个char,我的机器八位,左4位表示将的位置,右位表示帅的位置,关键是怎么给一个8位的char的左右半边赋值再获得左右半边的值.位操作!
#include <iostream> /*下面几个宏用于掩码,在获得左右半边的值时用*/
#define FULLMASK 255 //
#define RMASK (FULLMASK>>4) //
#define LMASK (FULLMASK<<4) // /*下面是几个获得左右半边值的宏*/
#define GETR(t) (t&RMASK) //获得右半边4位的值,用掩码,掩盖住左半边的值,注意这个位操作不改变原有值
#define GETL(t) ((t&LMASK)>>4) //获得左半边4位的值,先掩盖右4位,再把值右移4位,注意这里要价格括号...我猜临时值放在某寄存器里 /*下面是设置左右半边值的宏*/
#define SETR(t,val) (t=(t&LMASK)|val)//((t=t&LMASK),(t=t|val)) //用十进制val设置右4位,但要保证val可以用4位表示,即val表示值的位在低4位.(这里先清空右四位,再与左4位全是0而右4位是值的val或)
#define SETL(t,val) (t=((t&RMASK)|(val<<4)))//((t=t&RMASK),(t=t|(val<<4))) //用十进制val设置左4位 (先清空左4位,再把val右边4位的值移动到左边,再与) int main(void)
{
using namespace std; unsigned char my; //注意这里一定要用无符号的,要不放最高位为1就麻烦了.... for (SETL(my,);(int)GETL(my)<=;SETL(my,(int)GETL(my)+))
{
for (SETR(my,);(int)GETR(my)<=;SETR(my,(int)GETR(my)+))
{
if (((int)GETL(my)%)!=((int)GETR(my)%))
{
cout<<"("<<(int)GETL(my)<<","<<(int)GETR(my)<<")\n";
}
}
} cin.get();
return ;
}
--------------------------------------------------------------------------------------------------------------------------------------------------
思路二:不用上面的程序模式了,换个思路.
将有9种情况,帅也有9种情况,组合起来共81种情况.
能不能用一个变量表示这81种情况,对于每个情况的值,获得将和帅的值???
可以!
将=0,帅=(0,1,2,3,4,5,6,7,8) 值可以从0到8
将=1,帅=(0,1,2,3,4,5,6,7,8) 值可以从9到17
将=2,帅=(0,1,2,3,4,5,6,7,8) ..
将=3,帅=(0,1,2,3,4,5,6,7,8) ..
将=4,帅=(0,1,2,3,4,5,6,7,8)
将=5,帅=(0,1,2,3,4,5,6,7,8)
将=6,帅=(0,1,2,3,4,5,6,7,8)
将=7,帅=(0,1,2,3,4,5,6,7,8)
将=8,帅=(0,1,2,3,4,5,6,7,8) 值从72到80
即用一个八位的值val表示所以上述从0到80的81种情况,
val/9即将的值,val%9即帅的值,搞定!
#include <iostream> int main(void)
{
using namespace std; char val=;
while (val<=)
{
if ((val/%)!=(val%%))
{
cout<<"("<<val/<<","<<val%<<")"<<endl;
}
val++;
} cin.get();
return ;
}
--------------------------------------------------------------------------------------------------------------------------------------------------
总结:
1.多角度看问题
2.C/C++位操作的强大,及几种位操作运算符的使用
编程之美 ---> 1.2中国象棋将帅问题的更多相关文章
- 1.2 中国象棋将帅问题进一步讨论与扩展:如何用1个变量实现N重循环?[chinese chess]
[题目] 假设在中国象棋中只剩下将帅两个棋子,国人都知道基本规则:将帅不能出九宫格,只能上下左右移动,不能斜向移动,同时将帅不能照面.问在这样条件下,所有可能将帅位置.要求在代码中只能使用一个字节存储 ...
- 《编程之美》practice
1.2.中国象棋将帅问题 要求:只用一个字节存储变量,输出将帅不照面的所有可能位置. 思路简单,就是穷举让将和帅不在同一列即可,用char高四字节和低四字节分别存储将和帅的位置,位置编号从1到9.代码 ...
- C/C++编程笔记:C语言打造中国象棋游戏,项目源代码分享!
中国象棋是起源于中国的一种棋,属于二人对抗性游戏的一种,在中国有着悠久的历史.由于用具简单,趣味性强,成为流行极为广泛的棋艺活动. 它是中国棋文化,也是中华民族的文化瑰宝,它源远流长,趣味浓厚,基本规 ...
- JavaScript中国象棋程序(0) - 前言
“JavaScript中国象棋程序” 这一系列教程将带你从头使用JavaScript编写一个中国象棋程序.希望通过这个系列,我们对博弈程序的算法有一定的了解.同时,我们也将构建出一个不错的中国象棋程序 ...
- JavaScript中国象棋程序(4) - 极大极小搜索算法
"JavaScript中国象棋程序" 这一系列教程将带你从头使用JavaScript编写一个中国象棋程序.这是教程的第4节. 这一系列共有9个部分: 0.JavaScript中国象 ...
- C/C++游戏项目:中国程序员一定要会的中国象棋教程
中国象棋是中国一种流传十分广泛的游戏. 下棋双方根据自己对棋局形式的理解和对棋艺规律的掌握,调动车马,组织兵力,协调作战在棋盘这块特定的战场上进行着象征性的军事战斗. 象棋,亦作"象碁&qu ...
- <<编程之美>>1.2读后有感
问题提出 中国象棋的"将","帅"问题,他俩不能在一条直线上.求出他们的合法位置,并且只能用一个变量. 分析 一头雾水,不明所以.往下看了下,感觉像是程序员为难 ...
- BZOJ 1801中国象棋 DP
1801: [Ahoi2009]chess 中国象棋 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 1426 Solved: 826[Submit][ ...
- 【编程之美】2.5 寻找最大的k个数
有若干个互不相等的无序的数,怎么选出其中最大的k个数. 我自己的方案:因为学过找第k大数的O(N)算法,所以第一反应就是找第K大的数.然后把所有大于等于第k大的数取出来. 写这个知道算法的代码都花了2 ...
随机推荐
- [Mapreduce]eclipse下写wordcount
上传两个文件到hdfs上的input目录下 代码例如以下: import java.io.IOException; import java.util.StringTokenizer; import o ...
- 几种server模型
TCP測试用客户程序 每次执行客户程序,在命令行參数指定server的ip地址,port,发起连接的子进程数,和一个待发送的字符串数据,客户程序将模拟多个客户依据指定的子进程数创建子进程来并发的连接到 ...
- 安卓高手之路之 ClassLoader
我不喜欢那些泛泛而谈的去讲那些形而上学的道理,更不喜欢记那些既定的东西.靠记忆去弥补思考的人,容易陷入人云亦云的境地,最后必定被记忆所围困,而最终消亡的是创造力.希望这个高手之路系列能够记录我学习安卓 ...
- Proteus仿真_01、 8086 IO译码仿真
最近在学习一些微机原理与接口技术方面的知识. 参考书籍<微机原理与接口技术---基于8086Proteus仿真> 顾晖 梁惺彦 编著 实验一.利用8086 芯片来实现对I/O设备的读取和控 ...
- ${var}变量替换
${age},${name} 给定一个上下文 Map<String,String> context 使用上下文替换变量private static final Pattern VAR_PA ...
- Python 学习之二:Python超短教程
前言 本教程综合Stanford CS231N和UC Berkerley CS188的Python教程. 教程非常短,但适合有一定编程基础.学过其它语言的童鞋. Python 启动Python 解释器 ...
- javascript操作
1. >> 按位右移运算符 result = expression1 >> expression2 右移表达式的位,保持符号不变. >> 运算符将 expressi ...
- Android的属性系统
http://blog.csdn.net/jerryutscn/article/details/5519423 Android的属性系统 每条属性包含了名字和其对应的值,两者都用字符串来描述.Andr ...
- 沈逸老师ubuntu速学笔记(1)--安装flashplayer,配置中文输入法以及常用命令
开篇首先感谢程序员在囧途(www.jtthink.com)以及沈逸老师,此主题笔记主要来源于沈老师课程.同时也感谢少年郎,秦少.花旦等同学分享大家的学习笔记. 1.安装flash player ctr ...
- 笔记——Function类型 及其 call、apply方法
每个函数都是Function类型的实例.函数有三种定义方式和两个内部属性arguments和this. 同时函数也是对象,也有属性和方法.本篇主要其call()和apply()方法 属性 length ...