24点游戏程序设计.doc_第1页
24点游戏程序设计.doc_第2页
24点游戏程序设计.doc_第3页
24点游戏程序设计.doc_第4页
24点游戏程序设计.doc_第5页
已阅读5页,还剩9页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

杭州电子科技大学2012年6月短学期姓名: 谷蒙蒙 学号: 09056002 班级: 09056011班 专业: 计算机科学与技术(2+2) 学院: 计算机学院 课题: 24点游戏实现 指导老师: 张彦斌 完成时间: 2012.06.19 一、程序功能及规则设定本题目将设计一个24点游戏。程序主要负责实现随机发牌、设置牌面、表达式验算、计算机给出解答、结果统计等功能。系统可随机给出4个数,也可由用户自行输入四张牌的值。用户输入表达式,先判断表达式是否正确,如果正确,计算是否24,不正确的话,给出可能的正确表达式。用户可放弃输入,由系统直接给出可能的答案24点游戏是一种常见的纸牌游戏,就是利用加减乘除以及括号将给出的四张牌组成一个值为24的表达式,玩法十分简单,是一个消遣的好方法。本程序对传统24点作了一定的改变,用户不必自己手动输入表达式,程序提供相应的按钮,通过点击按钮键入表达式,方便用户操作。二、程序实现思路通过一定的功能分析,将程序基本化为:随机发牌,设置牌面,退出游戏,表达式键入,表达式验证,系统给出答案,判断是否无解这7个功能模块。122.1 随机发牌: 产生4个随机数rand(),将4个数保存下来,对相应的按钮及图片做出对应的处理,即显示相对应的扑克牌、修改数字按钮上显示的数字、设置按钮的可用性等。这些处理写在主窗口的私有成员函数中,名为startupCard(), 其中调用较多的API函数,如载入图片loadImage()、设置按钮内容SetDlgItemText()等。2.2 设置牌面: 用户自己设置扑克牌牌面(此时跳出一个设置窗口),获取用户输入的数字,其余操作与随机发牌操作相同,调用startupCard()完成。2.3 退出游戏: 弹出关闭提示窗口,询问用户事故否确认退出。点击确定,则退出;点击取消,则返回,继续游戏。2.4 表达式键入: 设置4个数组、加、减、乘、除、左括号、右括号、清除、完成等按钮,并且一定时期某些按钮可用,某些按钮不可用。其规则如下:1)程序开始时,只有数字及左括号可用;2)当按下数字按钮时,加减乘除按钮可用,若前面输入过左括号,则右括号可用;3)当按下左括号时,只有未使用的数字可用;4)当按下加减乘除时,未使用的数字及左括号可用;5)当按下右括号时,未使用的数字、加减乘除、右括号可用;6)当按下数字或右括号按钮,且数字都已经使用,左右括号匹配,此时完成按钮可用;7)清除按钮用于清除已键入的表达式。关于按钮可用性设置利用EnableWindow()函数来实现。2.5 表达式验证: 即计算用户输入的表达式,首先获取表达式GetDlgItemText(),根据加减乘除运算法则,使用C+ STL库所提供的栈,进行表达式计算,验证用户输入的是否等于24。算法思路:(1) 运算符的优先级处理。为根据运算符的优先级对表达式进行运算,引入优先级最低的运算符“;”,该运算符不参加实际的运算,只出现在表达式的开始和最后,仅起到保证表达式的其他运算符运算完为止的作用。由此,规定运算符从低到高的优先级依次为:0级:;,1级:),2级:+和-, 3级:*和/, 4级:(。(2) 运算过程中的运算符和运算数的操作处理。在从左到右依次读入表达式字符串中字符的过程中,若遇到连续若干个数码字符就转换为数值,并将其入运算数栈;若遇到运算符,当该运算符的优先级高于运算符栈顶的运算符时,将该运算符入运算符栈,否则弹出运算数栈顶的两个元素依次作为右边和左边的运算数,弹出运算符栈顶的元素作为运算符进行运算,将结果入栈到运算数栈。但要注意以下情况: 读入的运算符)不入运算符栈; 当运算符栈顶是运算符(时,读入的运算符除)外均入栈; 当读入的运算符是)而运算符栈顶是(时,运算符栈退栈; 当读入的是运算数或是运算符而入栈了或是运算符)而运算符栈顶是(时,继续读入下一字符,否则不能读入下一字符。(3) 比较运算符的优先级算法。根据存于一维数组中的运算符极其优先级,对运算符栈顶的运算符与读入的运算符比较其优先级,返回响应的信息。(4) 核心算法:完成表达式的求值并输出响应的操作及运算符栈和运算数栈的变化状态。对依次从表达式字符串读入的字符,若是数码字符,则将连续的若干个数码字符转换成数值后,入栈到运算数栈;否则,若是运算符且不是):则当该运算符的优先级比运算符栈顶的运算符优先级高或运算符栈顶的运算符是(时,该运算符入运算符栈,继续读入后续字符;否则,若是运算符且是),而运算符栈顶是(时,运算符栈退栈,继续读入后续字符;否则,弹出运算数栈顶的两个元素依次作为右边和左边的运算数,弹出运算符栈顶的元素作为运算符进行运算,将结果入栈到运算数栈,此时,如果当前运算符的优先级高于运算符栈顶元素的优先级,则当前运算符入运算符栈,继续读入后续字符,如果表达式已读完而且运算符栈已空,则继续读入后续字符。以上各种情况处理完后,均输出响应的操作及运算符栈和运算数栈的变化状态。最后返回表达式的运算结果运算数栈中唯一的一个数据。2.6 系统给出答案: 使用递归的方式,达到穷举的效果求24点的解 算法思路:首先,从宏观上说,这种问题都是遍历穷举。再看看运算符,其中+,* 都是没有顺序的。即(a*b=b*a), 但是 -、/ 是有顺序的。那么假设都有顺序的。那么就可以统一处理了(最多效率低点,先解决问题。再考虑优化)。那么遍历所有a,b,c,d 以及 三次运算 。即A(4,4) *4*4*4 就是该算法的复杂度。微观上,由于中间有除法,那么不能用int类型来储存数据了。需要用double或者float.每次运算都只有两个数运算。2.7 判断是否无解: 同样系统计算答案,判断有没有解三、程序流程图结束否显示结果放弃重新开始否提示错误是提示正确提示正确判断是否无解是玩家认为无解等待玩家输入表达式判断表达式是否正确等待用户输入表达式玩家认为有解初始化界面,包括按钮、图片等玩家输入1-13范围内的4个数随机产生1-13范围内的4个数结束游戏设置牌面开始游戏开始四、程序调试12344.1 图片显示:图片导入难以实现,不断修改、查阅信息、修改完成。4.2 表达式求值算法:计算结构有误。原因为运算符优先级设置不正确,如加号和减号的优先级相同,乘号和除号的优先级相同。4.3 按钮可用性设定:不可连续输入括号,如:(6+6)+6)+6,是正确的解,但是无法输入,修改了按钮可用性设定的规则:当按下左括号时,左括号仍可用,并且,四个数字都使用了以后,系统自动完成括号匹配的工作。4.4 有待改进的地方:程序经常出现闪屏现象,可以增加定时、难度级别设置的功能。五、部分截图5.1主界面:5.2设置牌面: 5.3退出游戏: 六、部分关键代码(完整请见poin24工程)/*计算表达式*/int calculate(CString &expression)string str= expression.GetBuffer(0);/将CString转换成string类型str+=;/添加分号,作为表达式结束stack Sign;/运算符栈stack Num;/数字栈Sign.push(;);/分号压入栈底int pos = 0;/从0号为开始读取表达式char current = str0, topSign;/当前字符while(Sign.top() != ; | current != ;)int num = apartNum(str,pos);/拆分数字与字符if(num != -1)/读取出数字,添加到数字栈中Num.push(num);else/读取出运算符current = strpos;/当前字符pos+;/继续下次循环topSign = Sign.top();/得到当前栈顶的运算符if(current = ) & topSign=()/当前读到右括号,栈顶为左括号Sign.pop();/直接弹出左括号else if(current != ) & cmpPriority(current, topSign) | topSign = () Sign.push(current);/将当前运算符压栈elseint a,b,c;a = Num.top();/得到数字栈栈顶元素Num.pop();b = Num.top();/得到第二个数字Num.pop();topSign = Sign.top();/弹出运算符Sign.pop();if(topSign = +) c = a + b;/加法else if(topSign = -) c = b - a;/减法else if(topSign = *) c = a * b;/乘法else if(topSign = /) c = b / a;/除法else return -1;Num.push(c);/将新的数存入数字栈中topSign = Sign.top();/得到符号栈栈顶运算符pos-;/向前退一个,继续操作当前的运算符current = strpos;/下一轮的当前字符return Num.top();/*系统计算24点的解*/bool search(int n, int* number, string* expression) if (n = 1) if ( fabs(number0 - 24) PRECISION ) /精度范围内,表达式正确/cout expression0 endl; /输出表达式 return true; else return false; /穷举法,递归调用求表达式 for (int i = 0; i n; i+) for (int j = i + 1; j GetWindowText(expression);GetDlgItem(IDC_NUMONE)-GetWindowText(num);/获取当前输入的数字expression += num;/将新输入的数字添加到表达式中SetDlgItemText(ID_EXPRESSION, expression);/将新表达式显示到文本框中if(!allVisited()GetDlgItem(IDC_ADD)-EnableWindow(TRUE); GetDlgItem(IDC_SUB)-EnableWindow(TRUE); GetDlgItem(IDC_MUL)-EnableWindow(TRUE); GetDlgItem(IDC_DIV)-EnableWindow(TRUE); elsefor(int i = 0; i EnableWindow(TRUE); /完成按钮可用GetDlgItem(IDC_NUMONE)-EnableWindow(FALSE);/数字按钮不可用GetDlgItem(IDC_NUMTWO)-EnableWindow(FALSE); GetDlgItem(IDC_NUMTHREE)-EnableWindow(FALSE); GetDlgItem(IDC_NUMFOUR)-EnableWindow(FALSE); GetDlgItem(IDC_LEFTBRACKET)-EnableWindow(FALSE); /左括号不可用if(m_leftCnt 0) /前面出现过左括号,则右括号可用GetDlgItem(IDC_RIGHTBRACKET)-EnableWindow(TRUE); /*按下加减乘除键时,表达式相关的按钮的可用性的设置*/void CPoint24Dlg:OnAdd() /加号CString expression, sign; /获取已输入的表达式GetDlgItem(ID_EXPRESSION)-GetWindowText(expression);GetDlgItem(IDC_ADD)-GetWindowText(sign);/获取新输入的运算符expression += sign;/将新输入的运算符添加到表达式中SetDlgItemText(ID_EXPRESSION, expression);/将新表达式显示到文本框中if(!m_visitNum0) / /未使用的数字按钮可用GetDlgItem(IDC_NUMONE)-EnableWindow(TRUE); if(!m_visitNum1)GetDlgItem(IDC_NUMTWO)-EnableWindow(TRUE); if(!m_visitNum2)GetDlgItem(IDC_NUMTHREE)-EnableWindow(TRUE); if(!m_visitNum3)GetDlgItem(IDC_NUMFOUR)-EnableWindow(TRUE); GetDlgItem(IDC_LEFTBRACKET)-EnableWindow(TRUE); /左括号可用GetDlgItem(IDC_RIGHTBRACKET)-EnableWindow(FALSE); /右括号不可用GetDlgItem(IDC_ADD)-EnableWindow(FALSE); /运算符不可用GetDlgItem(IDC_SUB)-EnableWindow(FALSE); GetDlgItem(IDC_MUL)-EnableWindow(FALSE); GetDlgItem(IDC_DIV)-EnableWindow(FALSE); /*按下左括号键时,表达式相关的按钮的可用性的设置*/void CPoint24Dlg:OnLeftbracket() /左括号CString expression, sign;GetDlgItem(ID_EXPRESSION)-GetWindowText(expression);/获取已输入的表达式GetDlgItem(IDC_LEFTBRACKET)-GetWindowText(sign);/获取新输入的运算符expression += sign;/将新输入的运算符添加到表达式中SetDlgItemText(ID_EXPRESSION, expression);/将新表达式显示到文本框中m_leftCnt +;/左括号计数器加1if(!m_visitNum0)GetDlgItem(IDC_NUMONE)-EnableWindow(TRUE);/未使用的数字按钮可用if(!m_visitNum1)GetDlgItem(IDC_NUMTWO)-EnableWindow(TRUE); if(!m_visitNum2)GetDlgItem(IDC_NUMTHREE)-EnableWindow(TRUE); if(!m_visitNum3)GetDlgItem(IDC_NUMFOUR)-EnableWindow(TRUE); GetDlgItem(IDC_LEFTBRACKET)-EnableWindow(TRUE); /左括号可用GetDlgItem(IDC_RIGHTBRACKET)-EnableWindow(FALSE); /右括号不可用GetDlgItem(IDC_ADD)-EnableWindow(FALSE); /运算符不可用GetDlgItem(IDC_SUB)-EnableWindow(FALSE); GetDlgItem(IDC_MUL)-EnableWindow(FALSE); GetDlgItem(IDC_DIV)-EnableWindow(FALSE); /*按下右括号键时,表达式相关的按钮的可用性的设置*/void CPoint24Dlg:OnRightbracket() /右括号CString expression, sign;GetDlgItem(ID_EXPRESSION)-GetWindowText(expression);/获取已输入的表达式GetDlgItem(IDC_RIGHTBRACKET)-GetWindowText(sign);/获取新输入的运算符expression += sign;/将新输入的运算符添加到表达式中SetDlgItemText(ID_EXPRESSION, expression);/将新表达式显示到文本框中m_leftCnt -;GetDlgItem(IDC_NUMONE)-EnableWindow(FALSE);/数字按钮不可用GetDlgItem(IDC_NUMTWO)-EnableWindow(FALSE); GetDlgItem(IDC_NUMTHREE)-EnableWindow(FALSE); GetDlgItem(IDC_NUMFOUR)-EnableWindow(FALSE); GetDlgItem(IDC_LEFTBRACKET)-EnableWindow(FALSE); /左括号不可用if(m_leftCnt 0)GetDlgItem(IDC_RIGHTBRACKET)-EnableWindow(FALSE); /前面出现过左括号,则有括号可用GetDlgItem(IDC_ADD)-EnableWindow(TRUE); GetDlgItem(IDC_SUB)-EnableWindow(TRUE); GetDlgItem(IDC_MUL)-EnableWindow(TRUE); GetDlgItem(IDC_DIV)-EnableWindow(TRUE); if(allVisited() & m_leftCnt = 0)GetDlgItem(IDC_COMPLETE)-EnableWindow(TRUE); /完成按钮可用/*开始设置*/void CPoint24Dlg:startupCard()/修改数字按钮上的显示信息m_leftCnt = 0;/左括号计数器为0SetDlgItemText(ID_EXPRESSION, );/清空表达式栏GetDlgItem(IDC_NUMONE);SetDlgItemText(IDC_NUMONE, m_buff0); /将数字显示到按钮中SetDlgItemText(IDC_NUMTWO, m_buff1); SetDlgItemText(IDC_NUMTHREE, m_buff2); SetDlgItemText(IDC_NUMFOUR, m_buff3);fill(m_visitNum, m_visitNum+4, false);/访问标志初始化为:未访问GetDlgItem(IDC_NUMONE)-EnableWindow(TRUE);/数字按钮可用GetDlgItem(IDC_NUMTWO)-EnableWindow(TRUE); GetDlgItem(IDC_NUMTHREE)-EnableWindow(TRUE); GetDlgItem(IDC_NUMFOUR)-EnableWindow(TRUE); GetDlgItem(IDC_GIVEUP)-EnableWindow(TRUE);/放弃按钮可用GetDlgItem(IDC_NOANSWER)-EnableWindow(TRUE);/无解按钮可用GetDlgItem(IDC_COMPLETE)-EnableWindow(FALSE);GetDlgItem(IDC_LEFTBRACKET)-EnableWindow(TRUE);CPaintDC pDC(this);CRect rect;CBitmap card4,*OldBitMap;CDC *dc,dcMem;dc=GetDC();for(int i = 0; iBitBlt(20+100*i,20,500,1000,&dcMem,0,0,SRCAND);BITMAP bm; (&cardi)- GetBitmap(&bm);dc-StretchBlt( 25+108*i, 20, 105, 175, &dcMem,/显示图片0, 0, bm.bmWidth, bm.bmHeight, SRCCOPY);dcMem.SelectObject(OldBitMap);DeleteObject(&cardi);dcMem.DeleteDC();/*无解按钮的响应事件*/void CPoint24Dlg:OnNoanswer() / TODO: Add your control notification handler code herem_totalCnt +;string expression4;for(int i = 0; i 4; i+)expressioni = m_buffi; if(search(4, m_op, expression)/24点求解CString str;str.Format(%s, expression0.c_str();/格式化str += = 24 ;AfxMessageBox(回答错误!此局有解,如: + str); /提示窗口elsem_rightCnt +;/答对次数加1AfxMessageBox(回答正确!此局无解!重新开牌.);displayCnt();/显示统计结果startupCard();/重新开局/*完成按钮的响应事件*/void CPoint24Dlg:OnComplete() / TODO: Add your control notification handler code herem_totalCnt +;/回答次数加1CString expression, totalResult;GetDlgItemText(ID_EXPRESSION, expression);/得到表达式int result = calculate(expression);/计算表达式char strResult10;itoa(result, strResult, 10);/类型转化expression += = ;expression += strResult;if(result = 24)m_rightCnt +;/答对次数加1AfxMessa

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

最新文档

评论

0/150

提交评论