3D游戏编程——牧师和恶魔

2024-01-10 18:50
文章标签 编程 3d 游戏 恶魔 牧师

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

简答题

游戏对象运动的本质

游戏对象的运动过程实际就是游戏对象随着时间进行空间位置、旋转角度、大小的变化。
游戏对象的运动过程本质上是使用矩阵变换(平移、旋转、缩放)改变游戏对象的空间属性。

抛物线运动

请用三种方法以上方法,实现物体的抛物线运动。

方法一:使用物体的重力属性

public class usegravity : MonoBehaviour
{// Start is called before the first frame updatevoid Start(){gameObject.AddComponent<Rigidbody>();// 给一个初始速度gameObject.GetComponent<Rigidbody>().velocity = new Vector3(1, 0, 0);}
}

方法二:使用transform.position

主要使用的是公式
y = x 2 y=x^2 y=x2

public class usegravity : MonoBehaviour
{public Vector3 initial;// Start is called before the first frame updatevoid Start(){initial = transform.position;}// Update is called once per framevoid Update(){transform.position -= new Vector3(Time.deltaTime, Time.deltaTime * Time.deltaTime + 2 * initial.y * Time.deltaTime, 0);}
}

方法三:使用transform.translate

与方法二类似。

public class usegravity : MonoBehaviour
{public Vector3 initial;// Start is called before the first frame updatevoid Start(){initial = transform.position;}// Update is called once per framevoid Update(){initial.x -= 0.1f;transform.Translate(Time.deltaTime * new Vector3(initial.x, initial.x*initial.x, 0), Space.World);}
}

太阳系模拟

写一个程序,实现一个完整的太阳系, 其他星球围绕太阳的转速必须不一样,且不在一个法平面上。

public class sunround : MonoBehaviour {public Transform sun;public int x, y, z, speed;// Use this for initializationvoid Start () {speed = Random.Range(150, 200);x = Random.Range(-50, 50);y = Random.Range(-50, 50);z = Random.Range(-50, 50);}// Update is called once per framevoid Update () {var axis = new Vector3(x, y, z);transform.RotateAround(sun.position, axis, speed * Time.deltaTime);}
}

在这里插入图片描述

Priests and Devils

完整工程文件在github(https://github.com/JennySRH/3DGame/tree/master/PriestsAndDevils)。

游戏视频https://www.bilibili.com/video/av68091576

游戏中的事物

牧师,恶魔,船,河流,两岸

其中牧师是红色方块,恶魔是蓝色球体,船是棕色长方体。

在这里插入图片描述
但是这样的游戏界面实在太过于单调,所以经过我的修改,最终游戏界面如下图所示,其中,水面还利用正弦函数增加了波浪效果。
在这里插入图片描述

规则表

当前状态玩家操作结果
牧师或恶魔在岸边,船上有空位玩家点击靠近船一侧岸上的恶魔或牧师恶魔或牧师上船
牧师或恶魔在船上玩家点击船上的恶魔或牧师恶魔或牧师上靠近船的岸
其中一侧的恶魔的数量大于牧师的数量——显示玩家输
恶魔和牧师全部到河边另一侧——显示玩家赢

游戏对象

灰色长方体——堤岸

棕色长方体——船

蓝色长方体——河

红色正方体——牧师

蓝色球体——恶魔

设计过程

游戏框架如下图所示

在这里插入图片描述

Director职责如下所示:

  • 获取当前游戏的场景
  • 控制场景运行、切换、入栈与出栈
  • 暂停、恢复、退出
  • 管理游戏全局状态
  • 设定游戏的配置
  • 设定游戏全局视图
public class Director : System.Object {private static Director _instance;             //导演类的实例public SceneController scene { get; set; }public static Director GetInstance(){if (_instance == null){_instance = new Director();}return _instance;}
}

Director类是单实例的,具有全局属性,可以在任何地方访问它,它继承至 C# 根对象,所以不会受 Unity 引擎管理,也不要加载。它通过一个抽象的场景接口访问不同场的场记(控制器)。例如:它不知道每个场景需要加载哪些资源,它可以与实现 SceneController 的不同场记对话。

controller

虽然在本次项目中,我们只创建了一个场景控制器FirstController,但是为了设计的完整性,我们还是需要定义一个场记控制器接口SceneController

public interface SceneController
{void LoadResources();           //加载场景void print();                   // 打印 测试用int getPosition(string name);   // 获取位置void moveto(string name);		// 移动void Reset();	                // reset
}

我们实现SceneController接口,创建了FirstController,并挂载到一个空物体上,便于初始化加载。该场景控制器首先需要加载游戏中的资源,并进行各种初始化。在此之前,我们先要将游戏对象做成预制,便于我们的直接使用。

在这里插入图片描述

然后加载到场景中。

   public void LoadResources(){Debug.Log("load resource");state = -1;// 创建新的GUImygui = new UserGUI();// 加载数量fromDevilNum = fromPriestNum = 3;toDevilNum = toPriestNum = 0;// 加载地面GroundFrom = Object.Instantiate(Resources.Load("ground", typeof(GameObject)), new Vector3(0, -5, 15), Quaternion.identity, null) as GameObject;GroundFrom.name = "groundfrom";GroundTo = Object.Instantiate(Resources.Load("ground", typeof(GameObject)), new Vector3(0, -5, -15), Quaternion.identity, null) as GameObject;GroundTo.name = "groundto";// 加载水Water = Object.Instantiate(Resources.Load("Water", typeof(GameObject)), new Vector3(0, -9, 0), Quaternion.identity, null) as GameObject;Water.name = "water";// 加载船boat = new Boat();// 加载牧师对象for (int i = 0;i < 3;i ++){Role temp = new Role("Cube");temp.setName("Priest"+ i);Priests[i] = temp;float p_z = (float)(8 + i * 1.5);Priests[i].setOriginalPos(new Vector3(0, 0, p_z));Priests[i].setDestPos(new Vector3(0, 0, -p_z));}// 加载恶魔对象for (int i = 0; i < 3; i++){Role temp = new Role("Sphere");temp.setName("Devil" + i);Devils[i] = temp;float p_z = (float)(13 + i * 1.5);Devils[i].setOriginalPos(new Vector3(0, 0, p_z));Devils[i].setDestPos(new Vector3(0, 0, -p_z));}}

作为场记类,FirstController还需要响应各种事件。在本游戏中需要响应的事件为:鼠标点击、游戏重置、检测游戏输赢。游戏点击的事件触发是通过Move类来通知的。Move类实际上是一个检测类,挂载到所有可以移动的游戏对象上,用来检测是否有鼠标点击事件。如果鼠标点击了某个可移动的游戏对象,该类将告知场记,然后场记再交给相应的处理程序。

public class Move : MonoBehaviour {void OnMouseDown(){//Debug.Log("onmousedown");Director.GetInstance().scene.moveto(transform.name);}// Use this for initializationvoid Start () {}// Update is called once per framevoid Update () {}
}

model

而本游戏中的事物如船、牧师、恶魔等,便是MVC模式中的model了。我们创建一个Role类和Boat类来作为相应的模型类,用来记录每个事物的状态等。

public class Role {public GameObject character;public Move moveto;public int position;    // 0:from 1:to 2:boatpublic int boatPos;     // 0 1public Vector3 origin;public Vector3 dest;public Role(string name){position = 0;character = Object.Instantiate(Resources.Load(name, typeof(GameObject)), Vector3.zero, Quaternion.identity, null) as GameObject;moveto = character.AddComponent(typeof(Move)) as Move;}public void setName(string name){character.name = name;}public void getBoat(int i){position = 2;if (i == 1){boatPos = 0;character.transform.position = new Vector3(0, -3, 4);}else if(i == 2){boatPos = 1;character.transform.position = new Vector3(0, -3, 6);}else if(i == 3){boatPos = 1;character.transform.position = new Vector3(0, -3, -4);}else{boatPos = 0;character.transform.position = new Vector3(0, -3, -6);}}public void setOriginalPos(Vector3 a){character.transform.position = a;origin = a;}public void setDestPos(Vector3 a){dest = a;}public void MoveToOrigin(){position = 0;character.transform.position = origin;}public void MoveToDest(){position = 1;character.transform.position = dest;}}public class Boat{public int[] BoatState;  // 记录船上的载荷位置public int num;         // 记录载荷public int position;        // 记录船的位置:0在from的位置,1在to的位置public GameObject thisboat;public Move moveto;public Boat(){num = 0;position = 0;BoatState = new int[2];BoatState[0] = BoatState[1] = 0;thisboat = Object.Instantiate(Resources.Load("Boat", typeof(GameObject)), new Vector3(0, -4, 5), Quaternion.identity, null) as GameObject;thisboat.name = "boat";moveto = thisboat.AddComponent(typeof(Move)) as Move;}
}

view

场景类的接口是IUserAction

public interface IUserAction 
{void display();void setLose();void setWin();
}

UserGUI实现了该场景类的接口,用来作为GUI显示button、label等部件。同时通过跟controller的交互传递信息,改变状态。

public class UserGUI : MonoBehaviour, IUserAction {private int state;public UserGUI(){state = -1;}public void setWin(){state = 1;}public void setLose(){state = 0;}public void display(){OnGUI();}void OnGUI(){GUIStyle style = new GUIStyle{border = new RectOffset(10, 10, 10, 10),fontSize = 50,fontStyle = FontStyle.BoldAndItalic,};// normal:Rendering settings for when the component is displayed normally.style.normal.textColor = new Color(200 / 255f, 180 / 255f, 150 / 255f);    // 需要除以255,因为范围是0-1GUI.Label(new Rect(240, 10, 200, 80), "Priests and Devils", style);if (GUI.Button(new Rect(50, 10, 100, 50), "Reset")){Director.GetInstance().scene.Reset();}if (state == 0){style.normal.textColor = new Color(255 / 255f, 0 / 255f, 0 / 255f);GUI.Label(new Rect(330, 350, 200, 80), "You Lose!", style);}if (state == 1){style.normal.textColor = new Color(255 / 255f, 0 / 255f, 0 / 255f);GUI.Label(new Rect(340, 350, 200, 80), "You Win!", style);}}
}

基本的框架搭建完毕后,再在此基础上进行游戏逻辑的设计(游戏的动作、输赢逻辑等)就简单多了,在此不做赘述,详细见github(https://github.com/JennySRH/3DGame/tree/master/PriestsAndDevils)。

实现效果如下所示:

在这里插入图片描述

这篇关于3D游戏编程——牧师和恶魔的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C#反射编程之GetConstructor()方法解读

《C#反射编程之GetConstructor()方法解读》C#中Type类的GetConstructor()方法用于获取指定类型的构造函数,该方法有多个重载版本,可以根据不同的参数获取不同特性的构造函... 目录C# GetConstructor()方法有4个重载以GetConstructor(Type[]

Python开发围棋游戏的实例代码(实现全部功能)

《Python开发围棋游戏的实例代码(实现全部功能)》围棋是一种古老而复杂的策略棋类游戏,起源于中国,已有超过2500年的历史,本文介绍了如何用Python开发一个简单的围棋游戏,实例代码涵盖了游戏的... 目录1. 围棋游戏概述1.1 游戏规则1.2 游戏设计思路2. 环境准备3. 创建棋盘3.1 棋盘类

无人叉车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)