在之前的那篇文章中,我们已经认识了顺序表—>http://t.csdnimg.cn/2I3fE
基于此,便好理解ArrayList和后面的洗牌游戏了。
ArrayList底层是一段连续的空间,并且可以动态扩容,是一个动态类型的顺序表,与普通数组的区别就是它是没有固定大小的限制。
首先我们要导入一个ArrayList包
import java.util.ArrayList;
在正式调用方法之前,别忘了创建一个ArrayList对象哦~
ArrayList<String> list = new ArrayList<>();
在这里,我就创建了一个只能存储String类型数据的ArrayList(顺序表)
下面便是ArrayList一些常见的使用方法,仔细观察就不难发现这些方法在模拟实现顺序表的时候就已经实现过了,现在直接用就行了,若有不记得了,可查看帮助手册-->Overview (Java SE 17 & JDK 17)
import java.util.ArrayList;
public class commonTest {
public static void main(String[] args) {
// 创建一个ArrayList并添加一些元素
ArrayList<String> list = new ArrayList<>();
list.add("apple");
list.add("banana");
list.add("orange");
// 使用get方法获取指定位置的元素
String fruit = list.get(1);
System.out.println("下标为1的水果是: " + fruit);
// 使用set方法替换指定位置的元素
list.set(0, "hello");
System.out.println("替换之后 " + list);
// 使用remove方法移除指定位置的元素
list.remove(2);
System.out.println("移除之后 " + list);
// 使用size方法获取列表大小
int size = list.size();
System.out.println("列表大小: " + size);
// 使用contains方法判断列表是否包含指定元素
boolean containsBanana = list.contains("banana");
System.out.println("是否包含banana " + containsBanana);
}
}
运行结果如图
纸牌相信大家都接触过,来看看如何利用ArrayList去实现洗牌?
这才是本文的重点哦~理论最终还是要服务与实践的,接下来就由我来带着各位看官来实现吧~
首先思考:
要洗牌,首先要有牌,想要表示出来,那么便要想好牌的属性--->花色和数字,好,那么就可以定义一个Card类专门存储这个,看下图:
public class Card {
public int rank; // 表示牌面的数字
public String suit; // 表示花色
// 构造方法,用于初始化牌的数字和花色
public Card(int rank, String suit) {
this.rank = rank;
this.suit = suit;
}
// 重写toString()方法,以便打印Card对象时能够直观地展示牌的信息
@Override
public String toString() {
return "Card{" + suit + " " + rank + "}";
}
}
这样一来,这个Card类很好地封装了扑克牌的属性。
一张牌肯定不够,一副牌有52张(去掉大小王)---4个花色,1到13的数字,在洗牌之前,先有一副完整的牌才行显示出来才行
于是就新定义一个Cards类,规定好牌的花色
public static final String[] suits = {"", "", "", ""};//定义四种花色
还记得final 的作用吗?这里用final 修饰就是为了不被修改
接着通过两个循环,一个遍历花色,一个遍历数字 ,这样一张新牌就创建好了,别忘了要创建存放牌的List,这里就导入了ArrayList包以便用add方法了。
import java.util.ArrayList;
import java.util.List;
// 买一副牌
public List<Card> buyCards() {
List<Card> cardList = new ArrayList<>(); // 基于ArrayList创建一个用于存放牌的List
for (int i = 0; i < 4; i++) { // 循环四种花色
for (int j = 0; j < 13; j++) { // 循环13个数字(2-10,J,Q,K,A)
int rank = j; // 当前数字
String suit = suits[i]; // 当前花色
Card card = new Card(rank, suit); // 创建一张新的牌对象
cardList.add(card); // 将新牌加入到牌组中
}
}
return cardList; // 返回完整的一副牌
}
此时运行便是这个样子
这下牌有了那就开始洗牌了,洗牌肯定讲究随机,讲到随机少不了Random方法,要调用它,也到导入对应的Random包。将生成的随机数对应的牌(看索引)在遍历的过程中与每个牌的位置进行交换,这样便完成了洗牌,配合代码食用效果更佳~
import java.util.Random;
// 洗牌
public void shuffle(List<Card> cardList) {
Random random = new Random(); // 创建一个随机数生成器
for (int i = cardList.size(); i > 0; i--) { // 从最后一张牌开始向前遍历
int index = random.nextInt(i); // 生成一个随机的索引
swap(cardList, i - 1, index); // 将当前牌与随机选出的牌进行交换
}
}
// 洗牌的原理-交换
public void swap(List<Card> cardList, int i, int j) {
Card tmp = cardList.get(i); // 临时保存第i张牌
cardList.set(i, cardList.get(j)); // 将第i张牌替换为第j张牌
cardList.set(j, tmp); // 将第j张牌替换为临时保存的牌
}
还记得set方法是干嘛的吗?--->替换指定位置的元素,常见的记住可以提高效率的
此时效果是这样的
那么到此牌洗好了,那就开始发牌,假设我们有三个人打牌,为了方便展示,就每人抽4张牌,思考一下,拿到牌后是不是也要存储才是?于是便要创建3个List去接受摸到的牌,
// 发牌,三个人,每个人轮流摸4
public void drawCard(List<Card> cardList) {
List<Card> hand1 = new ArrayList<>(); // 第一个人的手牌
List<Card> hand2 = new ArrayList<>(); // 第二个人的手牌
List<Card> hand3 = new ArrayList<>(); // 第三个人的手牌
List<List<Card>> hand = new ArrayList<>(); // 存放三个人的手牌列表
hand.add(hand1);
hand.add(hand2);
hand.add(hand3);
for (int i = 0; i < 4; i++) { // 每个人轮流摸4张牌
for (int j = 0; j < 3; j++) { // 三个人依次摸牌
Card card = cardList.remove(0); // 从牌组中抽取一张牌,并从原来的牌组中移除
hand.get(j).add(card); // 将抽取的牌加入到对应人的手牌列表中
}
}
// 打印每个人摸到的牌
System.out.println("第1个人摸到的牌:" + hand1);
System.out.println("第2个人摸到的牌:" + hand2);
System.out.println("第3个人摸到的牌:" + hand3);
}
开始摸牌,用嵌套循环去模拟分发,这里要明白为啥每次只remove(0),这是因为存储牌的也是是基于ArrayList的List,每次移除第一个元素,后续的元素会自动补上,这样一来,每次遍历,下标为0的元素就是原来的下一张牌,就很好地模拟了实际的摸牌体验,
当然,循环停止后,也可以打印剩下的牌
System.out.println("剩余的牌");//显示剩余的牌
System.out.println(cardList);
最后在创建一个测试类来看看效果是否符合预期
import java.util.List;
public class demo {
public static void main(String[] args) {
Cards cards=new Cards();
List<Card> cardList=cards.buyCards();
System.out.println("刚开始的牌"+cardList);
cards.shuffle(cardList);
System.out.println("洗牌之后"+cardList);
cards.drawCard(cardList);//摸牌
}
}
最后的效果便是这般
说到这里,简单的洗牌游戏就被成功实现啦,如果感觉文章对你有所帮助的话,还请点点赞~
期待我们下次相会
文章浏览阅读645次。这个肯定是末尾的IDAT了,因为IDAT必须要满了才会开始一下个IDAT,这个明显就是末尾的IDAT了。,对应下面的create_head()代码。,对应下面的create_tail()代码。不要考虑爆破,我已经试了一下,太多情况了。题目来源:UNCTF。_攻防世界困难模式攻略图文
文章浏览阅读2.9k次,点赞3次,收藏10次。偶尔会用到,记录、分享。1. 数据库导出1.1 切换到dmdba用户su - dmdba1.2 进入达梦数据库安装路径的bin目录,执行导库操作 导出语句:./dexp cwy_init/[email protected]:5236 file=cwy_init.dmp log=cwy_init_exp.log 注释: cwy_init/init_123..._达梦数据库导入导出
文章浏览阅读1.9k次。1. 在官网上下载KindEditor文件,可以删掉不需要要到的jsp,asp,asp.net和php文件夹。接着把文件夹放到项目文件目录下。2. 修改html文件,在页面引入js文件:<script type="text/javascript" src="./kindeditor/kindeditor-all.js"></script><script type="text/javascript" src="./kindeditor/lang/zh-CN.js"_kindeditor.js
文章浏览阅读2.3k次,点赞6次,收藏14次。SPI的详情简介不必赘述。假设我们通过SPI发送0xAA,我们的数据线就会变为10101010,通过修改不同的内容,即可修改SPI中0和1的持续时间。比如0xF0即为前半周期为高电平,后半周期为低电平的状态。在SPI的通信模式中,CPHA配置会影响该实验,下图展示了不同采样位置的SPI时序图[1]。CPOL = 0,CPHA = 1:CLK空闲状态 = 低电平,数据在下降沿采样,并在上升沿移出CPOL = 0,CPHA = 0:CLK空闲状态 = 低电平,数据在上升沿采样,并在下降沿移出。_stm32g431cbu6
文章浏览阅读1.2k次,点赞2次,收藏8次。数据链路层习题自测问题1.数据链路(即逻辑链路)与链路(即物理链路)有何区别?“电路接通了”与”数据链路接通了”的区别何在?2.数据链路层中的链路控制包括哪些功能?试讨论数据链路层做成可靠的链路层有哪些优点和缺点。3.网络适配器的作用是什么?网络适配器工作在哪一层?4.数据链路层的三个基本问题(帧定界、透明传输和差错检测)为什么都必须加以解决?5.如果在数据链路层不进行帧定界,会发生什么问题?6.PPP协议的主要特点是什么?为什么PPP不使用帧的编号?PPP适用于什么情况?为什么PPP协议不_接收方收到链路层数据后,使用crc检验后,余数为0,说明链路层的传输时可靠传输
文章浏览阅读587次。软件测试工程师移民加拿大 无证移民,未受过软件工程师的教育(第1部分) (Undocumented Immigrant With No Education to Software Engineer(Part 1))Before I start, I want you to please bear with me on the way I write, I have very little gen...
文章浏览阅读304次。Thinkpad X250笔记本电脑,装的是FreeBSD,进入BIOS修改虚拟化配置(其后可能是误设置了安全开机),保存退出后系统无法启动,显示:secure boot failed ,把自己惊出一身冷汗,因为这台笔记本刚好还没开始做备份.....根据错误提示,到bios里面去找相关配置,在Security里面找到了Secure Boot选项,发现果然被设置为Enabled,将其修改为Disabled ,再开机,终于正常启动了。_安装完系统提示secureboot failure
文章浏览阅读10w+次,点赞93次,收藏352次。1、用strtok函数进行字符串分割原型: char *strtok(char *str, const char *delim);功能:分解字符串为一组字符串。参数说明:str为要分解的字符串,delim为分隔符字符串。返回值:从str开头开始的一个个被分割的串。当没有被分割的串时则返回NULL。其它:strtok函数线程不安全,可以使用strtok_r替代。示例://借助strtok实现split#include <string.h>#include <stdio.h&_c++ 字符串分割
文章浏览阅读2.3k次。1 .高斯日记 大数学家高斯有个好习惯:无论如何都要记日记。他的日记有个与众不同的地方,他从不注明年月日,而是用一个整数代替,比如:4210后来人们知道,那个整数就是日期,它表示那一天是高斯出生后的第几天。这或许也是个好习惯,它时时刻刻提醒着主人:日子又过去一天,还有多少时光可以用于浪费呢?高斯出生于:1777年4月30日。在高斯发现的一个重要定理的日记_2013年第四届c a组蓝桥杯省赛真题解答
文章浏览阅读851次,点赞17次,收藏22次。摘要:本文利用供需算法对核极限学习机(KELM)进行优化,并用于分类。
文章浏览阅读1.1k次。一、系统弱密码登录1、在kali上执行命令行telnet 192.168.26.1292、Login和password都输入msfadmin3、登录成功,进入系统4、测试如下:二、MySQL弱密码登录:1、在kali上执行mysql –h 192.168.26.129 –u root2、登录成功,进入MySQL系统3、测试效果:三、PostgreSQL弱密码登录1、在Kali上执行psql -h 192.168.26.129 –U post..._metasploitable2怎么进入
文章浏览阅读257次。本文将为初学者提供Python学习的详细指南,从Python的历史、基础语法和数据类型到面向对象编程、模块和库的使用。通过本文,您将能够掌握Python编程的核心概念,为今后的编程学习和实践打下坚实基础。_python人工智能开发从入门到精通pdf