简单C++程序——掷骰子

2024-03-03 09:38
文章标签 简单 c++ 程序 掷骰子

本文主要是介绍简单C++程序——掷骰子,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

课程首页在:http://blog.csdn.net/sxhelijian/article/details/11890759,内有完整教学方案及资源链接


  看到一位同学发布的博文,模拟了掷骰子。很好的题目,初学编程,就可以这样,找到一个很小,很好玩的需求去完成,这是未来做大项目的“引子”。

  他用代码实现了需求,这是最重要的。

  我也按捺不住,给一个新的版本。目前,同学们习惯了面向过程的思维,只管着让程序按照流程做完事即可,对面向对象程序的结构还没有感觉。类的封装,该如何体现?函数的接口,怎样才算简洁?这些问题,用讲的方式起个头可以,更重要的是,在用的过程中得到领悟。

  学生的程序是:

#include <iostream>
#include <cstdlib>
#include <ctime>using namespace std;
class shaizi
{
public:void yaodian();
private:int a;
};
int main()
{cout << "摇出的点数" << endl;shaizi s1;s1.yaodian();return 0;
}
void shaizi::yaodian()
{int x;srand(time(0));x=rand()%5+1;a=x;cout<<a;
}

  我改写的程序是:

#include <iostream>
#include <cstdlib>
#include <ctime>using namespace std;
class shaizi
{
public:int getdian();
private:int dian;
};
int main()
{shaizi s1;cout << "摇出的点数" <<s1.getdian()<<endl;return 0;
}
int shaizi::getdian()
{srand(time(0));dian=rand()%6+1;return dian;
}
  这个程序的结构要好一些。输出的工作交给main,对象s1只管提供数就行了。

  一个很简单的要求,输入输出尽可能交给测试函数,类只做围绕数据的处理工作。除非实在必要,不要在成员函数中用cin和cout。

  学习了后面的构造函数,这个简单程序还可以别的改造。


再写了一个好玩的,希望用常量定下局数后,能够多轮决输赢。例如,下面的程序,想9局5胜。

#include <iostream>
#include <cstdlib>
#include <ctime>using namespace std;
class shaizi
{
public:void setdian();int getdian();
private:int dian;
};
const int round=9;   //round必须置一个奇数
int main()
{shaizi s1,s2;int i=1,n1=0,n2=0;while(i<=round){s1.setdian();s2.setdian();cout<<"第"<<i<<"轮: 甲 "<<s1.getdian()<<",乙: "<<s2.getdian();if(s1.getdian()>s2.getdian())n1++;else if (s2.getdian()>s1.getdian())n2++;else{cout<<",平局重掷 ";continue;}i++;cout<<endl;if(n1==(round/2+1)||n2==(round/2+1))break;}cout << "甲赢 "<<n1<<" 轮,乙赢 "<<n2<<" 轮";if(n1>n2)cout<<",甲胜"<<endl;else if(n2>n1)cout<<",乙胜"<<endl;return 0;
}
int shaizi::getdian()
{return dian;
}void shaizi::setdian()
{srand(time(0));dian=rand()%6+1;
}
  程序中用了类的“标配”,set和get成员函数都有,这个结构更好。在main函数中多设了些道道,能够9局5胜,且平局不算。

  但是,程序运行是死循环!

  在屏幕上的数字快速滚动中,猜测问题的原因是,每次都平局,都continue了。把28-32行的处理平局的代码去掉运行,果然如此。见图:

  

  为什么会这样?还得说随机数的原理。用rand()得到的随机数,并不是完全随机,是“伪随机”,随机数序列取决于“种子数”,种子数由srand(long)设置。也就是说,当种子数相同时,得到的随机序列就是完全相同的。为此,常取系统时间(time(0)返回的是从1970年1月1日午夜起到现在的秒数)作种子数,这是个可以让每次运行时种子数都不同的办法。

  然而 在这个程序中,调用s1和s2两个对象的setdian成员函数时,间隔的时间太短了,现在再慢的计算机,也不会让再次调用间隔超过1秒,甚至在这1秒末,那一秒初的那一瞬间的可能性都没有。每次都平局,不可避免。

  好不容易想到个好玩的简单游戏,就此罢休?想到的一个解决方案,让两次“投掷”停顿一下(这在专业中叫做“延时”),可以做到能够得出不同的种子数。

  程序如下:

#include <iostream>
#include <cstdlib>
#include <ctime>
#include <windows.h>using namespace std;
class shaizi
{
public:void setdian();int getdian();
private:int dian;
};
const int round=9;   //round必须置一个奇数
int main()
{shaizi s1,s2;int i=1,n1=0,n2=0;while(i<=round){s1.setdian();Sleep(1000);   //延时1000毫秒,即1秒s2.setdian();cout<<"第"<<i<<"轮: 甲 "<<s1.getdian()<<",乙: "<<s2.getdian();if(s1.getdian()>s2.getdian())n1++;else if (s2.getdian()>s1.getdian())n2++;else{cout<<",平局重掷 "<<endl;continue;}i++;cout<<endl;if(n1==(round/2+1)||n2==(round/2+1))break;}cout << "甲赢 "<<n1<<" 轮,乙赢 "<<n2<<" 轮";if(n1>n2)cout<<",甲胜"<<endl;else if(n2>n1)cout<<",乙胜"<<endl;return 0;
}
int shaizi::getdian()
{return dian;
}void shaizi::setdian()
{srand(time(0));     //设置种子数,由于有了间隔,能够保证种子数不同dian=rand()%6+1;    //实际每次取出的是,种子数确定的随机序列中的第一个
}

  加入的Sleep(1000)是延迟1000毫秒,为了调用此函数,需要#include<window.h>。

  由于延时,结果每隔1秒出来一行,倒也好看。见图:

  


  下面再给一种方案,种子数不再每次获得随机数前置,而是在main函数开始时设置,也能得到好的效果。见程序:

#include <iostream>
#include <cstdlib>
#include <ctime>
using namespace std;
class shaizi
{
public:void setdian();int getdian();
private:int dian;
};
const int round=9;   //round必须置一个奇数
int main()
{shaizi s1,s2;int i=1,n1=0,n2=0;srand(time(0));    //设置一次种子数,决定整个程序中用到的随机序列while(i<=round){s1.setdian();s2.setdian();cout<<"第"<<i<<"轮: 甲 "<<s1.getdian()<<",乙: "<<s2.getdian();if(s1.getdian()>s2.getdian())n1++;else if (s2.getdian()>s1.getdian())n2++;else{cout<<",平局重掷 "<<endl;continue;}i++;cout<<endl;if(n1==(round/2+1)||n2==(round/2+1))break;}cout << "甲赢 "<<n1<<" 轮,乙赢 "<<n2<<" 轮";if(n1>n2)cout<<",甲胜"<<endl;else if(n2>n1)cout<<",乙胜"<<endl;return 0;
}
int shaizi::getdian()
{return dian;
}void shaizi::setdian()
{dian=rand()%6+1;
}

  该还有别的方案,评论中接龙。



  

china-pub
卓越网
图灵社区  官方试读本下载
豆瓣
==================== 迂者 贺利坚 CSDN博客专栏=================
|== IT学子成长指导专栏 专栏文章分类目录(不定期更新)  ==|
|== C++ 课堂在线专栏  贺利坚课程教学链接(分课程年级) ==|
===== 为IT菜鸟起飞铺跑道,和学生一起享受快乐和激情的大学 =====

  
    








这篇关于简单C++程序——掷骰子的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/769210

相关文章

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu2289(简单二分)

虽说是简单二分,但是我还是wa死了  题意:已知圆台的体积,求高度 首先要知道圆台体积怎么求:设上下底的半径分别为r1,r2,高为h,V = PI*(r1*r1+r1*r2+r2*r2)*h/3 然后以h进行二分 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#includ

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

usaco 1.3 Prime Cryptarithm(简单哈希表暴搜剪枝)

思路: 1. 用一个 hash[ ] 数组存放输入的数字,令 hash[ tmp ]=1 。 2. 一个自定义函数 check( ) ,检查各位是否为输入的数字。 3. 暴搜。第一行数从 100到999,第二行数从 10到99。 4. 剪枝。 代码: /*ID: who jayLANG: C++TASK: crypt1*/#include<stdio.h>bool h

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直