[3D游戏编程]Priests And Devils

2023-10-25 04:50
文章标签 编程 3d 游戏 priests devils

本文主要是介绍[3D游戏编程]Priests And Devils,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

游戏简介

游戏规则

将左岸上的priest和devil都移到右岸即为胜利。

如果出现某一边(如果船开往该边,还要算上船上的)的devil数量大于priest数量,且此时的priest数量不为0,则游戏失败。

游戏效果

课堂任务

1)用unity实现该游戏

2)通过LoadResources动态生成对象

3)严格按照MVC结构编写程序

4)注意细节,船未靠岸,牧师与魔鬼上下船运动中,均不能接受用户事件

游戏实现

游戏框架设计(UML图)

动作表

对象动作对应函数
priest上下船

priestsGetOn()

priestsGetOff()

devil上下船

devilsGetOn()

devilsGetOff()

boat向两岸移动boatMove()

代码实现

游戏严格按照课程所提出的MVC总体框架设计(保证看得懂),所以正如课程所说,分为五个类:FirstController、ISceneController、IUserAction、SSDirector、UserInterface

(各位学长大佬的代码真的看不懂_(:зゝ∠)_)

项目代码(https://github.com/Ivan53471/Priests-And-Devils.git)

SSDirector

这部分没啥好说的,按老师教的写就行,不影响游戏逻辑实现

public class SSDirector : System.Object
{// singlton instanceprivate static SSDirector _instance;public ISceneController currentSceneController { get; set; }// get instance anytime anywhare!public static SSDirector getInstance(){if (_instance == null){_instance = new SSDirector();}return _instance;}public int getFPS(){return Application.targetFrameRate;}public void setFPS(int fps){Application.targetFrameRate = fps;}public void NextScene(){Debug.Log("Waiting next Scene now...");
#if UNITY_EDITORUnityEditor.EditorApplication.isPlaying = false;//UnityEditor.EditorApplication.Exit(0);
#elseApplication.Quit();  
#endif}
}

UserInterface

这部分主要实现的是MVC架构中的View,主要用于规定按键出现的位置,以及按下按键对应执行什么函数

void OnGUI(){//游戏正常进行if(mySceneController.GameOver() == 0){if (GUI.Button(new Rect(btnWidth / 2, 250, btnWidth, btnHeight), "Priests GetOn")){myActions.priestsGetOn();}if (GUI.Button(new Rect(btnWidth / 2 + btnWidth, 250, btnWidth, btnHeight), "Priests GetOff")){myActions.priestsGetOff();}if (GUI.Button(new Rect(btnWidth / 2 + 2 * btnWidth, 250, btnWidth, btnHeight), "Go!")){myActions.boatMove();}if (GUI.Button(new Rect(btnWidth / 2 + 3 * btnWidth, 250, btnWidth, btnHeight), "Devils GetOn")){myActions.devilsGetOn();}if (GUI.Button(new Rect(btnWidth / 2 + 4 * btnWidth, 250, btnWidth, btnHeight), "Devils GetOff")){myActions.devilsGetOff();}}//loseelse if(mySceneController.GameOver() == 1){GUI.Box(new Rect(2 * btnWidth, btnHeight, 2 * btnWidth, btnHeight), "\nYOU LOSE");if (GUI.Button(new Rect(2.5f * btnWidth, 2 * btnHeight, btnWidth, btnHeight), "Restart")){mySceneController.restart();}}//winelse{GUI.Box(new Rect(2 * btnWidth, btnHeight, 2 * btnWidth, btnHeight), "\nYOU WIN");if (GUI.Button(new Rect(2.5f * btnWidth, 2 * btnHeight, btnWidth, btnHeight), "Restart")){mySceneController.restart();}}}

IUserAction与ISceneController

这是为FirstController定义的两个接口,要弄明白这两个接口有什么用,其实可以从名字上看出来。

IUserAction,顾名思义,也就是和对象的动作有关。从之前的动作表可以得出,一共应该有

priestsGetOn()、priestsGetOff()、devilsGetOn()、devilsGetOff()、boatMove()五个函数

public interface IUserAction
{void boatMove();void priestsGetOn();void priestsGetOff();void devilsGetOn();void devilsGetOff();
}

ISceneController又是做什么的呢?本人在这里也疑惑了很久,既然都有FirstController这个东西了,为什么还要再加一个名字感觉差不多的东西呢?

先看一下这个接口里声明了什么函数:

public interface ISceneController
{void LoadResources();int GameOver();void restart();
}

可以看到,这些函数应该是每个场景都需要实现的功能,现在只有一个场景,确实可以省去ISceneController,但是如果有多个场景呢?这个时候总不能在SecondController、ThirdController又声明一次这些函数,那不就冗余了嘛。

同样的,上面的IUserAction也可以这么理解,同一个游戏不同场景之间应该是有关联的,那么对象所实现的动作应该也有所关联,IUserAction的存在同样也是为了减少代码冗余。

FirstController

这部分比较繁杂,需要分开解释。

MVC架构中的Model部分

为了记录每种对象的状态,创建用于记录状态的类(devil与priest相似,不展示)

    // 对每种物体新建类记录当前状态public class PriestsStatus{public bool onBoatLeft, onBoatRight;public bool onBankLeft, onBankRight;public PriestsStatus(){this.onBoatLeft = false;this.onBoatRight = false;this.onBankLeft = true;this.onBankRight = false;}}public class BoatBehaviour{public bool isMoving;public bool atLeftSide;public bool leftPosEmpty, rightPosEmpty;public BoatBehaviour() {this.isMoving = false;this.atLeftSide = true;this.leftPosEmpty = true;this.rightPosEmpty = true;}}

然后声明了该场景需要用到的变量

    // 记录物体样式、位置等等public List<GameObject> Priests, Devils;public GameObject boat, bankLeft, bankRight;// 物体状态public List<PriestsStatus> p_status;public List<DevilsStatus> d_status;public BoatBehaviour myBoatBehaviour;// 记录某个位置有多少个priest(devil)public int leftBankPriests, rightBankPriests;public int leftBankDevils, rightBankDevils;public int boatPriest, boatDevil;

MVC架构中的Controller部分

ISceneController.LoadResource

实现了为所有变量初始化,设置物体的样式、初始位置等等(只展示priest)

    public void LoadResources(){// priestsPriests = new List<GameObject>();p_status = new List<PriestsStatus>();for (int i = 0; i < 3; i++){GameObject priests = Instantiate<GameObject>(Resources.Load<GameObject>("Prefabs/Priests"),Vector3.zero, Quaternion.identity);priests.name = "Priest " + (i + 1);//priests.tag = "Priest";Priests.Add(priests);}// 初始化restart();}

其中调用了ISceneController.restart函数

该函数用于初始化所有变量的值,其中leftBankDevils = leftBankPriests = 3是因为priest和devil初始设定都在左岸。

    public void restart(){// priestsp_status = new List<PriestsStatus>();for (int i = 0; i < 3; i++){p_status.Add(new PriestsStatus());}Priests[0].transform.position = new Vector3(-8.6f, 3, 0);Priests[1].transform.position = new Vector3(-7.3f, 3, 0);Priests[2].transform.position = new Vector3(-6, 3, 0);//devilsd_status = new List<DevilsStatus>();for (int i = 0; i < 3; i++){d_status.Add(new DevilsStatus());}Devils[0].transform.position = new Vector3(-12.9f, 3, 0);Devils[1].transform.position = new Vector3(-11.6f, 3, 0);Devils[2].transform.position = new Vector3(-10.3f, 3, 0);//boatmyBoatBehaviour = new BoatBehaviour();boat.transform.position = new Vector3(-2.4f, 0.5f, 0);//bankbankLeft.transform.position = new Vector3(-8.5f, 1.5f, 0);bankRight.transform.position = new Vector3(8.5f, 1.5f, 0);// 初始化leftBankDevils = 3;leftBankPriests = 3;rightBankDevils = 0;rightBankPriests = 0;boatPriest = 0;boatDevil = 0;}

ISceneController.GameOver

返回值有0、1、2,分别代表游戏正常进行、游戏失败、游戏成功三种情况。

public int GameOver(){// 船靠岸分情况讨论if(myBoatBehaviour.atLeftSide){if ((leftBankPriests + boatPriest > 0&& leftBankDevils + boatDevil > leftBankPriests + boatPriest)|| (rightBankDevils > rightBankPriests && rightBankPriests > 0)){return 1;}}else{if ((rightBankPriests + boatPriest > 0&& rightBankDevils + boatDevil > rightBankPriests + boatPriest)|| (leftBankDevils > leftBankPriests && leftBankPriests > 0)){return 1;}if (rightBankDevils + boatDevil == 3 && rightBankPriests + boatPriest == 3){return 2;}}return 0;}

IUserAction.priestsGetOn、IUserAction.devilsGetOn

IUserAction.devilsGetOn逻辑相似,只展示IUserAction.priestsGetOn

priest上船可以分为:从左岸上船到船的左边、从左岸上船到船的右边、从右岸上船到船的左边、从右岸上船到船的右边,这里只展示第一种情况

    public void priestsGetOn(){// 如果船正在动,那么不能响应用户事件if (myBoatBehaviour.isMoving)return;// 船在左边if (myBoatBehaviour.atLeftSide){for (int i = 0; i < Priests.Count; i++){// 左侧岸上有牧师if (p_status[i].onBankLeft){// 上船位置if (myBoatBehaviour.leftPosEmpty){// 更改岸上、船上状态p_status[i].onBankLeft = false;p_status[i].onBoatLeft = true;p_status[i].onBoatRight = false;// 把船上左边位置设为有人myBoatBehaviour.leftPosEmpty = false;// 将priest移动到船左边的位置Priests[i].transform.position = new Vector3(-3.3f, 1.5f, 0);// 修改船上、岸上人数boatPriest++;leftBankPriests--;break;}if (myBoatBehaviour.rightPosEmpty){}}}}else{}}

IUserAction.priestsGetOff、IUserAction.devilsGetOff

IUserAction.devilsGetOff逻辑相似,只展示IUserAction.priestsGetOff

priest下船可以分为:从船的左边到左岸、从船的右边到左岸、从船的左边到左岸、从船的右边到右岸,这里只展示第一种情况

    public void priestsGetOff(){if (myBoatBehaviour.isMoving)return;// 船在左边if (myBoatBehaviour.atLeftSide){for (int i = 0; i < Priests.Count; i++){//船左侧有priestif (p_status[i].onBoatLeft){p_status[i].onBoatLeft = false;p_status[i].onBankLeft = true;p_status[i].onBankRight = false;myBoatBehaviour.leftPosEmpty = true;Priests[i].transform.position = new Vector3((-8.6f + i * 1.3f), 3, 0);boatPriest--;leftBankPriests++;break;}//船右侧有priestif (p_status[i].onBoatRight){}}}else{}}

IUserAction.boatMove

    public void boatMove(){// 船在动或者船上没人时不能开船if(!myBoatBehaviour.isMoving && (boatPriest + boatDevil > 0)){myBoatBehaviour.isMoving = true;}}

亮点设计(自认为)

本项目亮点设计来了,上面也提到了船是要动起来的,而不是瞬移过去的,这个操作该怎么实现呢?

在网上如果直接搜如何实现运动,大多数都是推荐使用协程,每帧通过协程调用状态改变的函数,达到运动的效果。

而本项目巧妙地运用了MonoBehavior中的Update函数,该函数每帧都会被调用,实现了和使用协程一样的效果

    // Update is called once per framevoid Update(){//give advice first//用于船和人的移动onBoatMoving();}

其中Update中调用的onBoatMoving函数,就是船的状态改变函数(具体实现参见详细代码)

详细代码

Github(https://github.com/Ivan53471/Priests-And-Devils.git)

总结

该项目用Unity实现了经典小游戏Priests And Devils,完成了老师所布置的任务

(预告:下期将优化代码,实现Priests And Devils动作分离版,敬请期待)

这篇关于[3D游戏编程]Priests And Devils的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

无人叉车3d激光slam多房间建图定位异常处理方案-墙体画线地图切分方案

墙体画线地图切分方案 针对问题:墙体两侧特征混淆误匹配,导致建图和定位偏差,表现为过门跳变、外月台走歪等 ·解决思路:预期的根治方案IGICP需要较长时间完成上线,先使用切分地图的工程化方案,即墙体两侧切分为不同地图,在某一侧只使用该侧地图进行定位 方案思路 切分原理:切分地图基于关键帧位置,而非点云。 理论基础:光照是直线的,一帧点云必定只能照射到墙的一侧,无法同时照到两侧实践考虑:关

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

国产游戏崛起:技术革新与文化自信的双重推动

近年来,国产游戏行业发展迅猛,技术水平和作品质量均得到了显著提升。特别是以《黑神话:悟空》为代表的一系列优秀作品,成功打破了过去中国游戏市场以手游和网游为主的局限,向全球玩家展示了中国在单机游戏领域的实力与潜力。随着中国开发者在画面渲染、物理引擎、AI 技术和服务器架构等方面取得了显著进展,国产游戏正逐步赢得国际市场的认可。然而,面对全球游戏行业的激烈竞争,国产游戏技术依然面临诸多挑战,未来的

【编程底层思考】垃圾收集机制,GC算法,垃圾收集器类型概述

Java的垃圾收集(Garbage Collection,GC)机制是Java语言的一大特色,它负责自动管理内存的回收,释放不再使用的对象所占用的内存。以下是对Java垃圾收集机制的详细介绍: 一、垃圾收集机制概述: 对象存活判断:垃圾收集器定期检查堆内存中的对象,判断哪些对象是“垃圾”,即不再被任何引用链直接或间接引用的对象。内存回收:将判断为垃圾的对象占用的内存进行回收,以便重新使用。

Go Playground 在线编程环境

For all examples in this and the next chapter, we will use Go Playground. Go Playground represents a web service that can run programs written in Go. It can be opened in a web browser using the follow

深入理解RxJava:响应式编程的现代方式

在当今的软件开发世界中,异步编程和事件驱动的架构变得越来越重要。RxJava,作为响应式编程(Reactive Programming)的一个流行库,为Java和Android开发者提供了一种强大的方式来处理异步任务和事件流。本文将深入探讨RxJava的核心概念、优势以及如何在实际项目中应用它。 文章目录 💯 什么是RxJava?💯 响应式编程的优势💯 RxJava的核心概念

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

MiniGPT-3D, 首个高效的3D点云大语言模型,仅需一张RTX3090显卡,训练一天时间,已开源

项目主页:https://tangyuan96.github.io/minigpt_3d_project_page/ 代码:https://github.com/TangYuan96/MiniGPT-3D 论文:https://arxiv.org/pdf/2405.01413 MiniGPT-3D在多个任务上取得了SoTA,被ACM MM2024接收,只拥有47.8M的可训练参数,在一张RTX

Java并发编程之——BlockingQueue(队列)

一、什么是BlockingQueue BlockingQueue即阻塞队列,从阻塞这个词可以看出,在某些情况下对阻塞队列的访问可能会造成阻塞。被阻塞的情况主要有如下两种: 1. 当队列满了的时候进行入队列操作2. 当队列空了的时候进行出队列操作123 因此,当一个线程试图对一个已经满了的队列进行入队列操作时,它将会被阻塞,除非有另一个线程做了出队列操作;同样,当一个线程试图对一个空