【数组和函数实战: 斗地主游戏】

目录

1. 玩法说明
2. 分析和设计
3. 代码实现
4. 游戏演示

1. 玩法说明

在这里插入图片描述
一副54张牌,3最小,两个王最大,其实是2,和上面一样从大到小排列

2. 分析和设计

2.1 分析和设计

常量和变量设计

一副牌有54张,有牌的数值和花色,可以分别用两个数组来存储,card为卡牌表示的数值,color为它的花色

在这里插入图片描述
卡牌创建好要进行游戏,需要将牌分给玩家,那么玩家也需要存储这些卡牌,玩家一般由三人,但也不确定,每个人有一副卡牌,所以适合用二维数组存储

在这里插入图片描述在这里插入图片描述

PLAYER: 玩家数量,默认为3
PLAYCARDS: 底牌为3,所以每个玩家申请的数组大小是54减去三张底牌除以玩家数量,然后考虑加入底牌的情况,数字大小再加3

在这里插入图片描述
lord记录地主的下标

函数设计

菜单显示函数

void menu();

在这里插入图片描述
卡牌初始化赋值函数

void InitCard(int num[],int numcolor[],int len);

在这里插入图片描述

考虑到需要比较卡牌的大小,从1开始赋值,1为最小的牌,通过下面的转换函数,将1转换为最小牌3的ascii码输出,花色每4个一轮,红心从3开始赋值,大小王单独赋值为14和15

转换字符函数

char ChangeCard(int num);
在这里插入图片描述
将牌的大小转换为对应大小的ascii码输出

显示卡牌函数

在这里插入图片描述由于10是两个字符,一个字符放不下,所以单独判断输出,大小王没有花色,也单独输出,17个一换行

洗牌函数

void RandomCard(int num[], int numcolor[], int len);
在这里插入图片描述
传入牌的数组和花色,思路是先从1-53下标找一个和0下标交换,之后从2-53找一个和1下标交换,从3-53找一个和2下标交换。数值和花色都需要交换,这里需要随机数函数,在主程序先置一个随机数种子

发牌函数

void SendCard(int num[], int numcolor[],
int playnum[][PLAYCARDS],int playcolor[][PLAYCARDS],
int len);

在这里插入图片描述
由于玩家是二维数组,而卡牌是一维数组,所以需要一些计算。当第一个玩家,也就是i=0的时候,循环17次,一人17张手牌,所以赋值的卡牌下标就是0-16,当第二个玩家是,下标从17-33,当i=1时,卡牌数组num中括号里的值需要是17,所以用i 乘玩家卡牌常量PLAYCARDS,再减去多余的3张地主牌,就是 1*(20-3)+0,然后把底牌手牌也输出

选地主函数

void LandLord(int num[], int numcolor[],
int playnum[][PLAYCARDS], int playcolor[][PLAYCARDS],
int len,int *lord);
在这里插入图片描述
遍历玩家数组,找到红心4的拥有者,选为地主,传入地主变量的地址,接收地主编号,将地主玩家的手牌中加入底牌

开始出牌显示

void PlayShow(int playnum[][PLAYCARDS], int playcolor[][PLAYCARDS],int len,int lord);
在这里插入图片描述

这时分农民和地主的显示,大致和ShowCard函数的逻辑一致,只是加入了地主和农民的分辨

3. 代码实现

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <windows.h>

#define PLAYER 3   //玩家数量
#define PLAYCARDS (54-3)/PLAYER+3 //玩家申请数量

void InitCard(int num[],int numcolor[],int len) {

	printf("初始化卡牌...\r\n");
	Sleep(1000);
	//牌的大小赋值,从1开始,依次累加
	int index = 0;    //牌大小
	int color = 3;    //牌花色
	for (int i = 0; i < len-2; i++) {

		if (i % 4 == 0) {
			index++;
			color = 3;
		}
		num[i] = index;
		numcolor[i] = color;
		color++;
	}
	num[52] = 14;
	num[53] = 15;	

}

void RandomCard(int num[], int numcolor[], int len) {

	int index = 0;
	for (int i = 1; i < len; i++) {
		//1 1-53   2 2-53  3 3-53
		//从后面随机一个,先和0下标交换,不断往后移
		index = rand() % (len-i) + i;

		int temp = num[i-1];
		int tempcolor = numcolor[i - 1];

		num[i - 1] = num[index];
		numcolor[i - 1] = numcolor[index];

		num[index] = temp;
		numcolor[index] = tempcolor;
	}
}

char ChangeCard(int num) {
	
		switch (num % 16)
		{
		case 1:
			return '3';
		case 2:
			return '4';
		case 3:
			return '5';
		case 4:
			return '6';
		case 5:
			return '7';
		case 6:
			return '8';
		case 7:
			return '9';
		case 8:
			return '0';
		case 9:
			return 'J';
		case 10:
			return 'Q';
		case 11:
			return 'K'; 
		case 12:
			return 'A';
		case 13:
			return '2';
		case 14:
			return '\x1';
		case 15:
			return '\x2';
		default:
			break;
		}
}
void ShowCard(int num[],int numcolor[], int len) {

	for (int i = 0; i < len; i++) {

		//大小王和10单独显示
		if (num[i] == 14 || num[i] == 15) {
			printf("%c   ", ChangeCard(num[i]));
		}
		else if (num[i] == 8) {
			printf("%c%d ", numcolor[i],10);
		}else {
			printf("%c%c  ", numcolor[i], ChangeCard(num[i]));
		}
		
		if ((i+1) % 17 == 0) {
			printf("\r\n");
		}
	}
	printf("\r\n");
}

//参数,卡牌数组和花色数组,玩家卡牌数组和花色数组,卡牌长度,玩家数量
void SendCard(int num[], int numcolor[],
			int playnum[][PLAYCARDS],int playcolor[][PLAYCARDS],
			int len) {

	system("cls");
	printf("发牌中...");
	Sleep(1000);
	system("cls");
	for (int i = 0; i < PLAYER; i++) {
		for (int j = 0; j < PLAYCARDS-3; j++) {
			playnum[i][j] = num[i * (PLAYCARDS-3) + j];
			playcolor[i][j] = numcolor[i * (PLAYCARDS - 3)+j];
		}
	}

	printf("\r\n");
	//显示手牌
	for (int i = 0; i < PLAYER; i++) {
		printf("play%d:\r\n", i + 1);
		ShowCard(playnum[i], playcolor[i], PLAYCARDS - 3);
	}
	//显示底牌
	printf("底牌:\r\n");
	for (int i = len- PLAYER; i < len; i++) {
		printf("%c%c ", numcolor[i], ChangeCard(num[i]));
	}
	printf("\r\n");
}

void LandLord(int num[], int numcolor[], 
			int playnum[][PLAYCARDS], int playcolor[][PLAYCARDS],
			int len,int *lord) {

	for (int i = 0; i < PLAYER; i++) {
		for (int j = 0; j < PLAYCARDS - 3; j++) {
			
			//红心4为地主
			if (playnum[i][j] == 2 && playcolor[i][j] == 3) {
				*lord = i;
				printf("地主是play%d\r\n", *lord +1);
				goto NEXT;
			}
			
		}
	}
	
NEXT:
	Sleep(3000);
	//地主加入底牌
	playnum[*lord][17] = num[51];
	playcolor[*lord][17] = numcolor[51];
	playnum[*lord][18] = num[52];
	playcolor[*lord][18] = numcolor[52];
	playnum[*lord][19] = num[53];
	playcolor[*lord][19] = numcolor[53];

}

void PlayShow(int playnum[][PLAYCARDS], int playcolor[][PLAYCARDS],int len,int lord) {

	system("cls");
	printf("开始出牌...\r\n");
	Sleep(1000);
	//显示手牌
	for (int i = 0; i < PLAYER; i++) {
		if (i == lord) {
			printf("(地主)play%d:\r\n", i + 1);
		}
		else {
			printf("(农民)play%d:\r\n", i + 1);
		}
		
		for (int j = 0; j < len; j++) {
			//大小王和10单独显示
			if (playnum[i][j] == 14 || playnum[i][j] == 15) {
				printf("%c   ", ChangeCard(playnum[i][j]));
			}
			else if(playnum[i][j] == 8){
				printf("%c%d ", playcolor[i][j], 10);
			}
			else {
				printf("%c%c  ", playcolor[i][j], ChangeCard(playnum[i][j]));
			}
		}
		printf("\r\n");
		
	}
	printf("\r\n");
}
void menu() {
	system("cls");
	printf("********************\r\n");
	printf("*****1.开始游戏*****\r\n");
	printf("*****0.退出游戏*****\r\n");
	printf("********************\r\n");
}
int main()
{
	srand((unsigned int)time(NULL));
	//创建卡牌数组
	int card[54] = { 0 };
	int color[54] = { 0 };
	//创建玩家数组
	int playcard[PLAYER][PLAYCARDS] = {0};
	int playcolor[PLAYER][PLAYCARDS] = {0};
	//卡牌长度
	int len = sizeof(card) / sizeof(card[0]);
	//地主
	int lord = 0;

	int sel;   //获取用户选择
	menu();
	do
	{
		scanf("%d", &sel);
		switch (sel) {
		case 1:
			system("cls");
			//初始化卡牌
			InitCard(card, color, len);
			//洗牌
			RandomCard(card, color, len);
			//ShowCard(card, color, len);
			//发牌
			SendCard(card, color, playcard, playcolor, len);
			//选地主
			LandLord(card, color, playcard, playcolor, len,&lord);
			//开始出牌
			PlayShow(playcard, playcolor, PLAYCARDS,lord);
			break;
		case 0:
			printf("退出游戏\r\n");
			return;
			break;
		default:
			menu();
			printf("输入错误\r\n");			
			break;
		}

	} while (1);
	return 0;
}

4. 游戏演示

需要将控制台右键属性切换为点阵字体,显示ascii符号

  1. 开始游戏

在这里插入图片描述

  1. 初始化卡牌

在这里插入图片描述
3. 选完地主显示

在这里插入图片描述
4. 出牌显示

在这里插入图片描述

后续功能等待开发…

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/213184.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Git 标签管理

前言 标签 tag&#xff0c;就相当于对 某一次的 commit 做一个标识&#xff0c;起了一个别名&#xff0c;例如&#xff1a;在某个项目发布版本的时候&#xff0c;可针对最后一次 commit 起一个别名 v1.0 来标识这一次的commit。tag 的作用&#xff1a;commit id 相对于 tag 是很…

openwrt上开启syslog打印方法

最近在openwrt上调试蓝牙时&#xff0c;出现问题&#xff0c;设备上的蓝牙适配器已经正常工作了&#xff0c;执行pair命令后&#xff0c;openwrt和待连接的设备上都出现了配对码&#xff0c;两边都同意&#xff0c;但连接失败 尝试分析log&#xff0c;发现在如下代码处打印了错…

代码随想录算法训练营 ---第五十二天

第一题&#xff1a; 简介&#xff1a; 动态规划五部曲&#xff1a; 1.确定 dp数组下标的定义 dp[i] 到达 i 时 最长递增子序列的长度 2.确定递推公式 我们确定当前的最大长度需要遍历前面所有的最大长度&#xff0c;然后如果序列最后一个值小于nums[i]那就dp[j] 1&#xf…

Redis--13--缓存一致性问题

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 缓存一致性问题1、先更新缓存&#xff0c;再更新DB方案二&#xff1a;先更新DB&#xff0c;再更新缓存方案三&#xff1a;先删缓存&#xff0c;再写数据库推荐1&…

Elk-filebeat

前言 Elk&#xff1a;filebeat搜集日志工具和logstash相同 Filebeat是一个轻量级的日志收集工具&#xff0c;所使用的资源比logstash部署和启动时使用的资源更小 Filebeat可以运行在非Java环境&#xff0c;他可以代理logstash在非Java环境上收集日志 Filebeat无法实现数据的…

【选择题】校招笔试选择题第一辑

题目 以下程序的运行结果是&#xff08; &#xff09; #include <stdio.h> int main(void) {printf("%s , %5.3s\n", "computer", "computer");return 0; }A. computer , puter B. computer , com C. computer , computer D. computer…

zookeeper+kafka+ELK+filebeat集群

目录 一、zookeeper概述&#xff1a; 1、zookeeper工作机制&#xff1a; 2、zookeeper主要作用&#xff1a; 3、zookeeper特性&#xff1a; 4、zookeeper的应用场景&#xff1a; 5、领导者和追随者&#xff1a;zookeeper的选举机制 二、zookeeper安装部署&#xff1a; 三…

STM32-SPI 中断

SPI协议 1.1 SPI总线介绍 SPI接口是Motorola &#xff08;motorola | Smartphones, Accessories & Smart Home Devices&#xff09;首先提出的全双工三线/四线同步串行外围接口采用主从模式&#xff08;Master Slave&#xff09;架构。 时钟由Master控制&#xff0c;在时钟…

【Leetcode题单】(01 数组篇)刷题关键点总结02【统计数组中的元素】

【Leetcode题单】&#xff08;01 数组篇&#xff09;刷题关键点总结02【统计数组中的元素】&#xff08;6题&#xff09; 统计数组中的元素645. 错误的集合 Easy697. 数组的度 Easy448. 找到所有数组中消失的数字 Easy442. 数组中重复的数据 Medium41. 缺失的第一个正数 Hard27…

Docker镜像制作与推送

目录 Docker镜像制作 搭建私服 将本地镜像推送到私有库 Docker镜像制作 以创建一个新ubuntu镜像&#xff0c;并安装vim命令示例 运行一个ubuntu镜像&#xff0c;发现在镜像里面无法使用vim命令&#xff0c;因为该ubuntu镜像只包括了其最基本的内核命令 [rootlocalhost ~]…

找不到msvcp110.dll如何修复?分享5个亲测有效的修复方法

在计算机使用过程中&#xff0c;我们经常会遇到一些错误提示&#xff0c;其中之一就是“msvcp110.dll丢失”。这个错误通常发生在运行某些程序时&#xff0c;系统无法找到所需的动态链接库文件。那么&#xff0c;msvcp110.dll到底是什么呢&#xff1f;它又有什么作用&#xff1…

算法通关村第七关—理解二叉树的遍历(白银)

深入理解前中后序遍历 给定一棵二叉树 二叉树前序遍历 public void preorder(TreeNode root,List<Integer>res){if&#xff08;rootnull){return;}res.add(root.val);preorder(root.left,res);preorder(root.right,res); }递归的过程如下图所示 从图中可以看到&#x…

〖大前端 - 基础入门三大核心之JS篇㊺〗- 定时器和延时器

说明&#xff1a;该文属于 大前端全栈架构白宝书专栏&#xff0c;目前阶段免费&#xff0c;如需要项目实战或者是体系化资源&#xff0c;文末名片加V&#xff01;作者&#xff1a;不渴望力量的哈士奇(哈哥)&#xff0c;十余年工作经验, 从事过全栈研发、产品经理等工作&#xf…

12月1号作业

实现运算符重载 #include <iostream>using namespace std; class Person{friend const Person operator-(const Person &L,const Person &R);friend bool operator<(const Person &L,const Person &R);friend Person operator-(Person &L,const …

第二十二章 指定元素和属性的命名空间 - 指定被视为Global元素的对象的命名空间

文章目录 第二十二章 指定元素和属性的命名空间 - 指定被视为Global元素的对象的命名空间指定被视为Global元素的对象的命名空间指定映射为元素的属性的命名空间案例1&#xff1a;属性被视为本地元素案例2:属性被视为Global元素 第二十二章 指定元素和属性的命名空间 - 指定被视…

Unity 简单打包脚本

打包脚本 这个打包脚本适用于做demo&#xff0c;脚本放在Editor目录下 using System; using System.Collections; using System.Collections.Generic; using System.IO; using UnityEditor; using UnityEngine;public class BuildAB {[MenuItem("Tools/递归遍历文件夹下…

【蓝桥杯】带分数

带分数 题目要求用一个ab/c的形式得到一个值&#xff0c;而且只能在1~9里面不重复的组合。 可以对1~9进行全排列&#xff0c;然后不断划分区间。 #include<iostream> #include<vector> using namespace std; int st[15]; int num[15]; int res; int n;int calc(i…

基于Python实现的滑动验证码自动识别工具源码

滑动验证码识别 今天的目标地址是字节的巨量纵横&#xff0c;目前东家是一家广告营销型的公司&#xff0c;专注于在各大平台投放信息流广告。巨量纵横为字节跳动的广告平台&#xff0c;用于管理推广账户。今天破解一下这个平台的登陆入口&#xff0c;为今后的数据爬取开个头。…

Android 源码编译

一&#xff0c;虚拟机安装 ​ 1.1 进入https://cn.ubuntu.com/download中文官网下载iso镜像 1.2 这里我们下载Ubuntu 18.04 LTS 1.3虚拟VM机安装ubuntu系统&#xff0c;注意编译源码需要至少16G运行内存和400G磁盘空间&#xff0c;尽量设大点 二 配置编译环境 2.1 下载andr…

进行主从复制时出现的异常FATAL CONFIG FILE ERROR (Redis 6.2.6)Reading the configuration file

错误如下所示&#xff1a; FATAL CONFIG FILE ERROR (Redis 6.2.6) Reading the configuration file, at line 1 >>> include/myredis/redis.conf Bad directive or wrong number of arguments出现错误的原因是.conf文件中命令之间缺少空格&#xff0c;如下所示&…