Unity游戏制作(二)

这篇博客介绍了如何使用Unity和MVC框架来开发一款名为‘魔鬼与牧师’的游戏。作者详细阐述了游戏的角色分配,包括导演、场记和群众,并提供了代码实现。游戏逻辑通过SSDirector、ISceneController接口和UserGUI类来控制,同时利用Click类处理用户交互。实验总结中提到,作者参考了他人解决方案来修复游戏中存在的bug。
摘要由CSDN通过智能技术生成

Unity游戏制作(二)

实验内容

  1. 编程实践:魔鬼与牧师

实验环境

  • Windows
  • Unity 2020.3.18

技术日记

一、MVC框架

模拟人类组织管理社会的方法,根据不同人拥有资源、知识与技能的不同,赋予不同人(或对象)特定的职责。再按一定结构(如设计模式),将它们组织起来。

MVC的游戏框架如下
在这里插入图片描述

其中,游戏框架中的角色分别为

  • 导演,1名(仅要一个)
    • 类型:SSDriector
    • 职责:
      • 获取当前游戏的场景
      • 控制场景运行、切换、入栈与出栈
      • 暂停、恢复、退出
      • 管理游戏全局状态
      • 设定游戏的配置
      • 设定游戏全局视图
  • 场记若干,每场需要一个。
    • 抽象类型(角色):ISceneController
    • 具体类型:FirstController
    • 职责:
      • 管理本次场景所有的游戏对象
      • 协调游戏对象(预制件级别)之间的通讯
      • 响应外部输入事件
      • 管理本场次的规则(裁判)
      • 各种杂务
  • 群众,1个
    • 抽象类型(角色):IUserAction
    • 具体类型:UserGUI

二、实验部分

1.项目配置过程

新建3d-unity的文件,然后直接把gitee上Assets文件夹替换新项目的Assets文件夹,把Scenes里面的SampleScenc然拖动出来后,直接点击运行即可开始游戏。

2. 实现思路及核心算法

为实现魔鬼与牧师,采用了MVC的架构,并设置了六个函数,其模块如下图所示:

在这里插入图片描述

具体如下:

1) 需要设置一个导演SSDirector.cs类:
public class SSDirector : System.Object{
	private static SSDirector _instance;
	public ISceneController CurrentScenceController { get; set; }
	
	public static SSDirector GetInstance(){
		if (_instance == null){
			_instance = new SSDirector();
		}
		return _instance;
	}
}

通过ISceneController接口访问不同场的场记。

2) 设置一个场记,其中场记的接口函数如下:
namespace Interface{
	public interface ISceneController{
		void LoadResources();
	}

	public interface IUserAction{
		void MoveBoat();           //移动小船                      
		void Restart();            //重新开始游戏                        
		void MoveRole(RoleModel role);       //移动人物              
		int Check();                   //判断游戏是否结束                    
	}
}

LoadResources()函数中,加载并初始化了长方形、正方形、球及其色彩代表游戏中的对象,如下所示:

public void LoadResources(){
		Debug.Log("DD");
		GameObject water = Instantiate(Resources.Load("Prefabs/Water", typeof(GameObject)), new Vector3(0,-1,0), Quaternion.identity) as GameObject;
		water.name = "water";
		start_land = new LandModel("start");
		end_land = new LandModel("end");
		boat = new BoatModel();
		role = new RoleModel[6];

		for (int i = 0; i < 3; i++){
			RoleModel r = new RoleModel("priest");
			r.SetName("priest" + i);
			r.SetPosition(start_land.GetEmptyPosition());
			r.GoLand(start_land);
			start_land.AddRole(r);
			role[i] = r;
		}

		for (int i = 3; i < 6; i++){
			RoleModel r = new RoleModel("devil");
			r.SetName("devil" + i);
			r.SetPosition(start_land.GetEmptyPosition());
			r.GoLand(start_land);
			start_land.AddRole(r);
			role[i] = r;
		}
3) 设置游戏逻辑与用户交互的门面 UserGUI.cs类:
public class UserGUI : MonoBehaviour {

	private IUserAction action;
	public int sign = 0;

	void Start()
	{
		action = SSDirector.GetInstance().CurrentScenceController as IUserAction;
	}
	void OnGUI(){
		if (sign == 1){
			GUI.Label(new Rect(Screen.width / 2-20, Screen.height/2-60, 100, 50), "Lost!");
			if (GUI.Button(new Rect(Screen.width/2-50, Screen.height/2+30, 100, 50), "Restart")){
				action.Restart();
				sign = 0;
			}
		}
		else if (sign == 2){
			GUI.Label(new Rect(Screen.width/2-20, Screen.height/2-60, 100, 50), "Win!");
			if (GUI.Button(new Rect(Screen.width/2-50, Screen.height/2+30, 100, 50), "Restart")){
				action.Restart();
				sign = 0;
			}
		}
	}	
}
还有Click.cs类捕捉用户的点击事件:
public class Click : MonoBehaviour{
	IUserAction action;
	RoleModel role = null;
	BoatModel boat = null;

	public void SetRole(RoleModel role){
		this.role = role;
	}

	public void SetBoat(BoatModel boat){
		this.boat = boat;
	}

	void Start(){
		action = SSDirector.GetInstance().CurrentScenceController as IUserAction;
	}

	void OnMouseDown(){
		if (boat == null && role == null){
			return;
		}

		if (boat != null){
			action.MoveBoat();
		}
		
		else if(role != null){
			action.MoveRole(role);
		}
	}
}
4)模型的控制类ModelController.cs

负责每个对象具体的控制,包含了四个类

  • BoatModel
  • LandModel
  • RoleModel
  • Move

实验总结

由于我自己写的游戏最后还是有许多bug没有解决,最终是跟着这个师兄的思路完成的:https://blog.csdn.net/TempterCyn/article/details/101110379

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值