当前位置: 首页 > article >正文

星际争霸小程序:用Java实现策略模式的星际大战

  在游戏开发的世界里,策略模式是一种非常实用的设计模式,它允许我们在运行时动态地选择算法或行为。今天,我将带你走进一场星际争霸的奇幻之旅,用Java实现一个简单的星际争霸小程序,通过策略模式来模拟不同种族单位的战斗行为。

实验背景

  星际争霸是一款经典的即时战略游戏,玩家需要控制不同种族的单位进行战斗。每个种族都有其独特的单位和战斗策略。在本次实验中,我选择了Java语言,利用策略模式来实现不同种族单位的攻击行为。通过策略模式,我们可以轻松地为单位切换不同的攻击策略,从而使代码更加灵活和可扩展。

实验设计

1. 整体思路

  为了实现星际争霸小程序,我们需要构建一个包含不同种族单位的战场。每个单位都有其生命值、攻击力和特定的攻击策略。通过策略模式,我们可以为单位动态切换攻击策略,从而模拟不同的战斗场景。

2.UML图

3. 主要模块设计

(1)攻击策略接口
  定义一个攻击策略接口 'Strategy',所有具体的攻击策略类都将实现这个接口。

package attack;import creature.Creature;public interface Strategy {void fight(Creature c);int getattack();
}

 (2)具体攻击策略类
实现具体的攻击策略类,如 'Air' 和 'Ground',分别对应空中攻击和地面攻击。

package attack;import mutalisk.Mutalisk;
import creature.Creature;public class Air implements Strategy {private int attack = 120;private int up = 200;private Mutalisk mutalisk;public Air(Mutalisk mutalisk) {this.mutalisk = mutalisk;}public void recover(int amount) {mutalisk.life += amount;}@Overridepublic void fight(Creature c) {System.out.println("空中攻击模式");System.out.println(mutalisk.name + "攻击" + c.name + " " + attack + "滴血");c.updateLife(attack);recover(up);}@Overridepublic int getattack() {return this.attack;}
}
package attack;import mutalisk.Mutalisk;
import creature.Creature;public class Ground implements Strategy {private int attack = 400;private int up_rc = 100;private Mutalisk mutalisk;public Ground(Mutalisk mutalisk) {this.mutalisk = mutalisk;}public void recover(int amount) {mutalisk.life += amount;}@Overridepublic void fight(Creature c) {System.out.println("地面攻击模式");System.out.println(mutalisk.name + "攻击" + c.name + " " + attack + "滴血");c.updateLife(attack);recover(up_rc);}@Overridepublic int getattack() {return this.attack;}
}

(3)单位抽象类
定义一个抽象类 'Creature',所有具体的单位类都将继承这个类。

package creature;import attack.Strategy;
import common.ICommon;public abstract class Creature implements ICommon, Strategy {public String name;public int life;protected int attack;public Creature(String name, int life, int attack) {this.name = name;this.life = life;this.attack = attack;}public Creature(String name, int life) {this.name = name;this.life = life;}@Overridepublic void state() {if (this.isDead()) {System.out.println("[" + this.name + "]已阵亡");} else {System.out.println("[" + this.name + "]的当前生命值为" + life + ";当前攻击值为" + attack);}}@Overridepublic abstract void fight(Creature c);public boolean isDead() {return life <= 0;}public void updateLife(int damage) {life -= damage;if (life <= 0) {life = 0;System.out.println(name + "已阵亡");}}@Overridepublic int getattack() {return 0;}
}

(4)具体单位类
实现具体的单位类,如 'Mutalisk' 和 'Terran'。

package mutalisk;import attack.Air;
import attack.Strategy;
import creature.Creature;
import zerg.Zerg;public class Mutalisk extends Zerg {private Strategy strategy;protected int attack;public Mutalisk(String name, int life) {super(name, life);}public void setStrategy(Strategy strategy) {this.strategy = strategy;this.attack = strategy.getattack();}public void fightStrategy(Creature c) {strategy.fight(c);}public void healZerg(Zerg z) {if (z.isDead()) {z.setlife(this.life);this.life = 0;System.out.println("忠诚的献身");}}
}

package terran;import creature.Creature;public class Terran extends Creature {public int up = 500;public Terran(String name, int life, int attack) {super(name, life, attack);}@Overridepublic void fight(Creature c) {System.out.println(name + "攻击" + c.name + " " + attack + "滴血");c.updateLife(attack);repair(up);}public void repair(int amount) {attack += amount;}
}

 4. 测试程序

编写测试程序,模拟星际争霸的战斗场景。

package test;import attack.Air;
import attack.Ground;
import attack.Strategy;
import creature.Creature;
import mutalisk.Mutalisk;
import terran.Terran;
import xrace.Xrace;
import zerg.Zerg;public class Test {public static void main(String[] args) {// 创建战场人员int n = 1;Terran[] terrans = new Terran[n];for (int i = 0; i < terrans.length; i++) {terrans[i] = new Terran("Terran" + (i + 1), 800, 100);}Xrace xrace = new Xrace("Xrace", 100, 8000);Zerg zerg = new Zerg("Zerg", 50000, 400);Mutalisk mutalisk = new Mutalisk("Mutalisk", 20000);// 创建两种攻击策略Air air = new Air(mutalisk);Ground ground = new Ground(mutalisk);// 战争开始前展示人员状态for (Terran terran : terrans) {terran.state();}xrace.state();zerg.state();mutalisk.state();// 游戏开始boolean gameEnd = false;while (!gameEnd) {// 检测Terran是否死亡boolean zergAttacked = false;for (Terran terran : terrans) {if (!terran.isDead()) {zerg.fight(terran);zergAttacked = true;break;}}if (!zergAttacked && !xrace.isDead()) {zerg.fight(xrace);}for (Creature terran : terrans) {if (!terran.isDead() && !zerg.isDead()) {terran.fight(zerg);}}if (!xrace.isDead() && !zerg.isDead()) {xrace.fight(zerg);}if (zerg.isDead() && !mutalisk.isDead()) {mutalisk.healZerg(zerg);}if (!xrace.isDead() && !mutalisk.isDead()) {mutalisk.setStrategy(air);mutalisk.fightStrategy(xrace);} else if (xrace.isDead()) {for (Creature terran : terrans) {if (!terran.isDead()) {mutalisk.setStrategy(ground);mutalisk.fightStrategy(terran);break;}}}// 更新状态for (Creature terran : terrans) {terran.state();}xrace.state();zerg.state();mutalisk.state();if ((xrace.isDead() && terrans[n - 1].isDead()) || (zerg.isDead() && mutalisk.isDead())) {gameEnd = true;if (xrace.isDead()) {System.out.println(xrace.name + "已阵亡,游戏失败");}if (zerg.isDead()) {System.out.println(zerg.name + "已阵亡,游戏胜利");}System.out.println("游戏结束!");}}}
}

实现效果

  通过调试和验证,星际争霸小程序成功实现!程序能够模拟不同种族单位的战斗行为,包括空中攻击和地面攻击。每个单位都可以动态切换攻击策略,从而适应不同的战斗场景。整个程序运行流畅,战斗逻辑清晰,状态更新及时。

战斗场景模拟

策略模式验证

总结

1. 策略模式的魅力:通过策略模式,我们可以轻松地为单位切换不同的攻击策略,从而使代码更加灵活和可扩展。
2. 面向对象编程的优势:利用Java的面向对象特性,我们可以将不同种族单位的行为封装在各自的类中,从而使代码更加模块化和易于维护。
3. 编程技巧的提升:通过实现这个小程序,我进一步掌握了Java的接口、抽象类和多态等概念,提升了编程技巧。
4. 游戏开发的乐趣:通过模拟星际争霸的战斗场景,我体验到了游戏开发的乐趣,激发了我对游戏开发的进一步学习兴趣。

结语

  通过这次实验,我不仅在技术上有了新的突破,更对游戏开发充满了兴趣。策略模式虽然简单,但它在游戏开发中有着广泛的应用。如果你对Java编程或游戏开发感兴趣,欢迎一起交流探讨!让我们在技术的道路上共同进步,继续探索更多神奇的魔法!✨

相关文章:

星际争霸小程序:用Java实现策略模式的星际大战

在游戏开发的世界里&#xff0c;策略模式是一种非常实用的设计模式&#xff0c;它允许我们在运行时动态地选择算法或行为。今天&#xff0c;我将带你走进一场星际争霸的奇幻之旅&#xff0c;用Java实现一个简单的星际争霸小程序&#xff0c;通过策略模式来模拟不同种族单位的战…...

请问交换机和路由器的区别?vlan 和 VPN 是什么?

交换机和路由器的区别 特性交换机&#xff08;Switch&#xff09;路由器&#xff08;Router&#xff09;工作层级数据链路层&#xff08;L2&#xff0c;基于MAC地址&#xff09;网络层&#xff08;L3&#xff0c;基于IP地址&#xff09;主要功能在局域网&#xff08;LAN&#…...

BERT 作为Transformer的Encoder 为什么采用可学习的位置编码

摘要 BERT 在位置编码上与原始 Transformer 论文中的 sin/cos 公式不同&#xff0c;选择了可学习&#xff08;learned&#xff09;的位置嵌入方案。本文将从 Transformer 原始位置编码选项入手&#xff0c;分析 BERT 选择 learned positional embeddings 的四大核心原因&#x…...

Python数据可视化高级实战之一——绘制GE矩阵图

目录 一、课程概述 二、GE矩阵? 三、GE 矩阵图的适用范围 五、GE 矩阵的评估方法 (一)市场吸引力的评估要素 二、企业竞争实力的评估要素 三、评估方法与实践应用 1. 定量与定性结合法 2. 数据来源 六、GE矩阵的图形化实现 七、总结:GE 矩阵与 BCG 矩阵的对比分析 (一)GE…...

StreamSaver实现大文件下载解决方案

StreamSaver实现大文件下载解决方案 web端 安装 StreamSaver.js npm install streamsaver # 或 yarn add streamsaver在 Vue 组件中导入 import streamSaver from "streamsaver"; // 确保导入名称正确完整代码修正 <!--* projectName: * desc: * author: dua…...

【Vue 3全栈实战】从响应式原理到企业级架构设计

目录 &#x1f31f; 前言&#x1f3d7;️ 技术背景与价值&#x1fa79; 当前技术痛点&#x1f6e0;️ 解决方案概述&#x1f465; 目标读者说明 &#x1f9e0; 一、技术原理剖析&#x1f4ca; 核心概念图解&#x1f4a1; 核心作用讲解&#x1f527; 关键技术模块说明⚖️ 技术选…...

Java线程池调优与实践经验

在Java面试中&#xff0c;线程池调优是一个常见且重要的考察点&#xff0c;尤其是当涉及Spring生态时&#xff0c;ThreadPoolTaskExecutor的使用经验通常会被深入追问。以下是针对该问题的结构化回答&#xff0c;结合原理、实践和调优经验&#xff1a; 1. 线程池调优的核心参数…...

【科研项目】大三保研人科研经历提升

大三保研人&#xff0c;五月科研项目经历提升 现在已经是五月下旬&#xff0c;各大高校的夏令营通知陆续发布&#xff0c;九月的预推免也近在眼前。我知道很多大三的同学正在焦虑——绩点已经定型&#xff0c;竞赛经历又不够丰富&#xff0c;简历上能写的东西太少&#xff0c;面…...

期刊采编系统安装升级错误

我们以ojs系统为例&#xff1a; PHP Fatal error: Uncaught Error: Call to a member function getId() on null in /esci/data/html/classes/install/Upgrade.inc.php:1019 Stacktrace: #0 /esci/data/html/lib/pkp/classes/install/Installer.inc.php(415): Upgrade->con…...

CSS【详解】弹性布局 flex

适用场景 一维&#xff08;行或列&#xff09;布局 基本概念 包裹所有被布局元素的父元素为容器 所有被布局的元素为项目 项目的排列方向&#xff08;垂直/水平&#xff09;为主轴 与主轴垂直的方向交交叉轴 容器上启用 flex 布局 将容器的 display 样式设置为 flex 或 i…...

自回归图像编辑 EditAR: Unified Conditional Generation with Autoregressive Models

Paperhttps://arxiv.org/pdf/2501.04699 Code (coming soon) 目录 方法 实验 EditAR是一个统一的自回归框架&#xff0c;用于各种条件图像生成任务——图像编辑、深度到图像、边缘到图像、分割到图像。 next-token预测的功效尚未被证明用于图像编辑。 EditAR主要构建在Ll…...

React Flow 中 Minimap 与 Controls 组件使用指南:交互式小地图与视口控制定制(含代码示例)

本文为《React Agent&#xff1a;从零开始构建 AI 智能体》专栏系列文章。 专栏地址&#xff1a;https://blog.csdn.net/suiyingy/category_12933485.html。项目地址&#xff1a;https://gitee.com/fgai/react-agent&#xff08;含完整代码示​例与实战源&#xff09;。完整介绍…...

基于YOLOv8 的分类道路目标系统-PyTorch实现

本文源码: https://download.csdn.net/download/shangjg03/90873939 1. 引言 在智能交通和自动驾驶领域,道路目标分类是一项关键技术。通过对摄像头捕获的图像或视频中的目标进行分类识别,可以帮助车辆或系统理解周围环境,做出更安全的决策。本教程将介绍如何使用 PyTorch …...

STM32之串口通信WIFI上云

一、W模块的原理与应用 基本概念 如果打算让硬件设备可以通过云服务器进行通信&#xff08;数据上报/指令下发&#xff09;&#xff0c;像主流的云服务器有阿里云、腾讯云、华为云&#xff0c;以及其他物联网云平台&#xff1a;巴法云.......&#xff0c;硬件设备需要通过TCP…...

PCB智能报价系统——————仙盟创梦IDE

软件署名 代码贡献&#xff1a; 紫金电子科技有限公司 文案正路&#xff1a;cybersnow 正文 对企业的竞争力有着深远影响。传统的 PCB 报价方式往往依赖人工核算&#xff0c;不仅耗时较长&#xff0c;还容易出现误差。随着科技的发展&#xff0c;PCB 自动报价系统应运而生&a…...

EXO分布式部署deepseek r1

EXO 是一个支持分布式 AI 计算的框架&#xff0c;可以用于在多个设备&#xff08;包括 Mac Studio&#xff09;上运行大语言模型&#xff08;LLM&#xff09;。以下是联调 Mac Studio 512GB 的步骤&#xff1a; 安装 EXO • 从 EXO GitHub 仓库 下载源码或使用 git clone 获取…...

每日算法 -【Swift 算法】寻找两个有序数组的中位数(O(log(m+n)))详细讲解版

&#x1f9e0; 用 Swift 寻找两个有序数组的中位数&#xff08;O(log(mn))&#xff09;详细讲解版 寻找两个有序数组的中位数&#xff0c;是 LeetCode 上非常经典的一道题&#xff0c;难度为 困难&#xff08;Hard&#xff09;&#xff0c;但它的本质是一个 二分查找 的变形应…...

Linux问题排查-找到偷偷写文件的进程

在 Linux 系统中&#xff0c;若要通过已修改的文件找到修改该文件的进程 PID&#xff0c;可以结合以下方法分析&#xff0c;具体取决于文件是否仍被进程打开或已被删除但句柄仍存在&#xff1a; 一、文件仍被进程打开&#xff08;未删除&#xff09; 如果文件当前正在被某个进…...

SOPHGO算能科技BM1688内存使用与编解码开发指南

1. BM1688内存分配接口详解 1.1 设备内存分配接口区别 BM1688提供了三个主要的设备内存分配接口,它们的主要区别如下: // 基本设备内存分配接口 void* bm_malloc_device_byte(bm_handle_t handle, unsigned int size);// 指定heap区域的设备内存分配 void*</...

kotlin flow的两种SharingStarted策略的区别

一 两种 SharingStarted 策略的区别&#xff1a; SharingStarted.Eagerly: 立即开始收集上游流&#xff0c;即使没有下游订阅者持续保持活跃状态&#xff0c;直到 ViewModel 被清除优点&#xff1a;响应更快&#xff0c;数据始终保持最新缺点&#xff1a;消耗更多资源&#x…...

LeetCode-链表-合并两个有序链表

LeetCode-链表-合并两个有序链表 ✏️ 关于专栏&#xff1a;专栏用于记录 prepare for the coding test。 文章目录 LeetCode-链表-合并两个有序链表&#x1f4dd; 合并两个有序链表&#x1f3af;题目描述&#x1f50d; 输入输出示例&#x1f9e9;题目提示&#x1f9ea;AC递归&…...

sqli-labs靶场29-31关(http参数污染)

目录 前言 less29&#xff08;单引号http参数污染&#xff09; less30&#xff08;双引号http参数污染&#xff09; less31(双引号括号http参数污染) 前言 在JSP中&#xff0c;使用request.getParameter("id")获取请求参数时&#xff0c;如果存在多个同名参数&a…...

独占内存访问指令LDXR/STXR

一、原子操作的介绍 在计算机领域里&#xff0c;如果要在多线程的情况下要保持数据的同步&#xff0c;需要引入称作Load-Link&#xff08;LL&#xff09;和Store-Conditional&#xff08;SC&#xff09;的操作&#xff0c;通常简称为LL/SC。 LL操作返回一个内存地址上当前存储…...

JVM 垃圾回收机制深度解析(含图解)

JVM 垃圾回收机制深度解析&#xff08;含图解&#xff09; 一、垃圾回收整体流程 垃圾回收图解 #mermaid-svg-KPtxlwWntQx8TOj3 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-KPtxlwWntQx8TOj3 .error-icon{fill…...

如何利用 Conda 安装 Pytorch 教程 ?

如何利用 Conda 安装 Pytorch 教程 &#xff1f; 总共分为六步走&#xff1a; &#xff08;1&#xff09;第一步&#xff1a;验证conda 环境是否安装好&#xff1f; 1) conda -V2) conda --version&#xff08;2&#xff09;第二步&#xff1a;查看现有环境 conda env list…...

【ffmpeg】SPS与PPS的概念

PPS&#xff08;Picture Parameter Set&#xff09;详解 PPS&#xff08;图像参数集&#xff09;是H.264/H.265视频编码标准中的关键数据结构&#xff0c;与SPS&#xff08;序列参数集&#xff09;共同组成视频的解码配置信息&#xff0c;直接影响视频的正确解码和播放。以下是…...

uniapp vue 开发微信小程序 分包梳理经验总结

嗨&#xff0c;我是小路。今天主要和大家分享的主题是“uniapp vue 开发微信小程序 分包梳理经验总结”。 在使用 UniAppvue框架开发微信小程序时&#xff0c;当项目比较大的时候&#xff0c;经常需要分包加载。它有助于控制主包的大小&#xff0c;从而提升小程序的启…...

什么是VR展示?VR展示的用途

随着科技的迅猛发展&#xff0c;我们步入一个全新的数字时代。在这个时代&#xff0c;虚拟现实&#xff08;VR&#xff09;技术崭露头角&#xff0c;逐步改变我们对世界的认知。全景展示厅作为VR技术与传统展览艺术的完美结合&#xff0c;以独特的全景视角&#xff0c;引领我们…...

.NET外挂系列:4. harmony 中补丁参数的有趣玩法(上)

一&#xff1a;背景 1. 讲故事 前面几篇我们说完了 harmony 的几个注入点&#xff0c;这篇我们聚焦注入点可接收的几类参数的解读&#xff0c;非常有意思&#xff0c;在.NET高级调试 视角下也是非常重要的&#xff0c;到底是哪些参数&#xff0c;用一张表格整理如下&#xff…...

Go语言中new与make的深度解析

在 Go 语言中&#xff0c;new 和 make 是两个用于内存分配的内置函数&#xff0c;但它们的作用和使用场景有显著区别。 理解它们的核心在于&#xff1a; new(T): 为类型 T 分配内存&#xff0c;并将其初始化为零值&#xff0c;然后返回一个指向该内存的指针 (*T)。make(T, ar…...