Unity控制逻辑(移动,翻滚,瞬移)

2023-10-31 12:31

本文主要是介绍Unity控制逻辑(移动,翻滚,瞬移),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

1.案例的前期准备

2.移动的实现

3.翻滚的实现

4.瞬移的实现


1.案例的前期准备

1.创建2D项目

 2.搭建场景

分别创建player(玩家),background(背景图),barrier(障碍物)。

 3.创建C#脚本

在对象player上创建CharacterController2D脚本

2.移动的实现

public class CharacterController2D : MonoBehaviour
{private const float MOVE_SPEED = 5f;private void Awake(){rigidbody2D = GetComponent<Rigidbody2D>();}private void Update(){float moveX = 0f;float moveY = 0f;if (Input.GetKey(KeyCode.W)){// transform.position += new Vector3(0,+1);moveY = +1f;}if (Input.GetKey(KeyCode.S)){// transform.position += new Vector3(0,-1);moveY = -1f;}if (Input.GetKey(KeyCode.A)){// transform.position += new Vector3(-1,0);moveX = -1f;}if (Input.GetKey(KeyCode.D)){// transform.position += new Vector3(+1,0);moveX = +1f;}moveDir = new Vector3(moveX, moveY).normalized;}private void FixedUpdate(){rigidbody2D.velocity = moveDir * MOVE_SPEED;}
}

实现方法:方向向量*移动速度

1,使用const储存物体速度(float)

2,在Update()中使用Input.GetKey(keyCode.按键)方法读取玩家相应按键 

3,使用moveX,moveY分别保存在X,Y轴的移动增量,以此创建Vector

4,使用normalized获取Vector对象的方向向量(单位向量)赋给moveDir              

Rigidbody2D.velocity

描述

刚体的线性速度,采用单位/秒形式。

在移动或旋转一个物体时,往往会直接使用Transform来执行这些操作。这种方法对于不具物理特性的GameObject来说,是可行的。但是一旦GameObject上附带有Rigidbody2D,这种方式就会带来性能的损失

3.翻滚的实现

public class CharacterController2D : MonoBehaviour
{//构建枚举类区分行走状态和翻滚状态private enum State{Normal, //正常行走Rolling, //翻滚}[SerializeField] private LayerMask dashLayerMask;private Rigidbody2D rigidbody2D;private Vector3 moveDir;private Vector3 rollDir; //翻滚的方向private Vector3 lastMoveDir; //保存最后一次非静止状态的Vectorprivate float rollSpeed; //翻滚的初始速度private State state; private void Awake(){rigidbody2D = GetComponent<Rigidbody2D>();state = State.Normal;}private void Update(){switch (state){case State.Normal:float moveX = 0f;float moveY = 0f;if (Input.GetKey(KeyCode.W)){// transform.position += new Vector3(0,+1);moveY = +1f;}if (Input.GetKey(KeyCode.S)){// transform.position += new Vector3(0,-1);moveY = -1f;}if (Input.GetKey(KeyCode.A)){// transform.position += new Vector3(-1,0);moveX = -1f;}if (Input.GetKey(KeyCode.D)){// transform.position += new Vector3(+1,0);moveX = +1f;}moveDir = new Vector3(moveX, moveY).normalized;//保存最后一次非静止状态的Vector,防止原地翻滚if(moveX != 0f || moveY != 0){lastMoveDir = moveDir;}//按下空格键进入翻滚状态   if (Input.GetKey(KeyCode.Space)){//最后一次非静止状态的Vector,即翻滚的方向rollDir = lastMoveDir;//翻滚的初始速度rollSpeed = 25f;state = State.Rolling;}break;case State.Rolling://翻滚是一个减速过程//减速系数float rollSpeedDropMultiplier = 5f;//按帧减速rollSpeed -= rollSpeed * rollSpeedDropMultiplier * Time.deltaTime;//设置下限float rollSpeedMinimum = 5f;//小于下限切换至行走状态if(rollSpeed < rollSpeedMinimum){state=State.Normal;}break;}}private void FixedUpdate(){switch (state){case State.Normal:rigidbody2D.velocity = moveDir * MOVE_SPEED;break;case State.Rolling:rigidbody2D.velocity = rollDir * rollSpeed;break;}}
}

4.瞬移的实现

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class CharacterController2D : MonoBehaviour
{private const float MOVE_SPEED = 5f;private enum State{Normal,Rolling,}[SerializeField] private LayerMask dashLayerMask;//设置射线与哪些层级发生碰撞,此项为可序列化的private Rigidbody2D rigidbody2D;private Vector3 moveDir;private Vector3 rollDir;private Vector3 lastMoveDir;private float rollSpeed;private bool isDashButtonDown;//瞬移状态切换private State state;private void Awake(){rigidbody2D = GetComponent<Rigidbody2D>();state = State.Normal;}private void Update(){switch (state){case State.Normal:float moveX = 0f;float moveY = 0f;if (Input.GetKey(KeyCode.W)){// transform.position += new Vector3(0,+1);moveY = +1f;}if (Input.GetKey(KeyCode.S)){// transform.position += new Vector3(0,-1);moveY = -1f;}if (Input.GetKey(KeyCode.A)){// transform.position += new Vector3(-1,0);moveX = -1f;}if (Input.GetKey(KeyCode.D)){// transform.position += new Vector3(+1,0);moveX = +1f;}moveDir = new Vector3(moveX, moveY).normalized;if(moveX != 0f || moveY != 0){lastMoveDir = moveDir;}//按F进入瞬移状态if (Input.GetKey(KeyCode.F)){isDashButtonDown = true;}if (Input.GetKey(KeyCode.Space)){rollDir = lastMoveDir;rollSpeed = 25f;state = State.Rolling;}break;case State.Rolling:float rollSpeedDropMultiplier = 5f;rollSpeed -= rollSpeed * rollSpeedDropMultiplier * Time.deltaTime;float rollSpeedMinimum = 5f;if(rollSpeed < rollSpeedMinimum){state=State.Normal;}break;}}private void FixedUpdate(){switch (state){case State.Normal:rigidbody2D.velocity = moveDir * MOVE_SPEED;if (isDashButtonDown){//瞬移距离float dashAmount = 0.5f;//无障碍物情况下的瞬移量Vector3 dashPosition = transform.position + lastMoveDir * dashAmount;//使用射线判断瞬移路径上是否有障碍物,有则被阻碍RaycastHit2D raycastHit2D = Physics2D.Raycast(transform.position, lastMoveDir, dashAmount, dashLayerMask);//射线碰撞的物体不为空if (raycastHit2D.collider != null){//射线与物体碰撞的点即为瞬移的最终位置dashPosition = raycastHit2D.point;}rigidbody2D.MovePosition(dashPosition);//切换至行走状态isDashButtonDown = false;}break;case State.Rolling:rigidbody2D.velocity = rollDir * rollSpeed;break;}}
}

​​​RigidBody2D.MovePosition

描述

将刚体移动到 /position/。

通过计算在下一次物理更新期间将刚体移动到指定 position 所需的适当线速度来将刚体移动到该位置。在移动过程中,重力或线性阻力都不会影响刚体。这使得对象能够快速从现有位置穿过世界移动到指定的 /position/。

由于该功能允许刚体穿过世界快速移动到指定的 /position/,因此附加到刚体的任何碰撞体都将按预期作出反应,也就是说,它们将产生碰撞和/或触发。这也意味着如果碰撞体产生碰撞,则将影响到刚体的运动,并可能阻止刚体在下一次物理更新期间到达指定的 /position/。如果是运动刚体,则任何碰撞都不影响刚体本身,只会影响任何其他动态碰撞体。

2D 刚体对其移动速度有固定限制,因此在短时间内尝试移动较远的距离会导致刚体无法在下一次物理更新期间到达指定 /position/。建议仅将该函数用于相对较短距离的移动。

请务必注意, 实际的位置更改只在下一次物理更新期间进行, 因此重复调用该方法而不等待下一次物理更新将导致使用最后一次调用。 因此,建议在 FixedUpdate 回调期间调用该函数。

Rigidbody2D.velocity:将钢体线性移动至某一位置

RigidBody2D.MovePosition:将钢体线性瞬移至某一位置

如何理解本案例中的射线RaycastHit2D

若没有使用射线进行判定,这钢体可能可以穿越碰撞体

RaycastHit2D raycastHit2D = Physics2D.Raycast(transform.position, lastMoveDir, dashAmount, dashLayerMask);

本案例的射线分别有如下参数

1.射线的起始位置

2.射线的方向

3.射线的距离

4.射线应于哪些层级发生碰撞

ps:若不设置此项,则与所有层级发生碰撞。可能导致本案例中的瞬移无法实现

这篇关于Unity控制逻辑(移动,翻滚,瞬移)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

Unity新手入门学习殿堂级知识详细讲解(图文)

《Unity新手入门学习殿堂级知识详细讲解(图文)》Unity是一款跨平台游戏引擎,支持2D/3D及VR/AR开发,核心功能模块包括图形、音频、物理等,通过可视化编辑器与脚本扩展实现开发,项目结构含A... 目录入门概述什么是 UnityUnity引擎基础认知编辑器核心操作Unity 编辑器项目模式分类工程

C#和Unity中的中介者模式使用方式

《C#和Unity中的中介者模式使用方式》中介者模式通过中介者封装对象交互,降低耦合度,集中控制逻辑,适用于复杂系统组件交互场景,C#中可用事件、委托或MediatR实现,提升可维护性与灵活性... 目录C#中的中介者模式详解一、中介者模式的基本概念1. 定义2. 组成要素3. 模式结构二、中介者模式的特点

mybatisplus的逻辑删除过程

《mybatisplus的逻辑删除过程》:本文主要介绍mybatisplus的逻辑删除过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录myBATisplus的逻辑删除1、在配置文件中添加逻辑删除的字段2、在实体类上加上@TableLogic3、业务层正常删除即

把Python列表中的元素移动到开头的三种方法

《把Python列表中的元素移动到开头的三种方法》在Python编程中,我们经常需要对列表(list)进行操作,有时,我们希望将列表中的某个元素移动到最前面,使其成为第一项,本文给大家介绍了把Pyth... 目录一、查找删除插入法1. 找到元素的索引2. 移除元素3. 插入到列表开头二、使用列表切片(Lis

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

MySQL逻辑删除与唯一索引冲突解决方案

《MySQL逻辑删除与唯一索引冲突解决方案》本文探讨MySQL逻辑删除与唯一索引冲突问题,提出四种解决方案:复合索引+时间戳、修改唯一字段、历史表、业务层校验,推荐方案1和方案3,适用于不同场景,感兴... 目录问题背景问题复现解决方案解决方案1.复合唯一索引 + 时间戳删除字段解决方案2:删除后修改唯一字

浅析Spring如何控制Bean的加载顺序

《浅析Spring如何控制Bean的加载顺序》在大多数情况下,我们不需要手动控制Bean的加载顺序,因为Spring的IoC容器足够智能,但在某些特殊场景下,这种隐式的依赖关系可能不存在,下面我们就来... 目录核心原则:依赖驱动加载手动控制 Bean 加载顺序的方法方法 1:使用@DependsOn(最直

Spring如何使用注解@DependsOn控制Bean加载顺序

《Spring如何使用注解@DependsOn控制Bean加载顺序》:本文主要介绍Spring如何使用注解@DependsOn控制Bean加载顺序,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录1.javascript 前言2. 代码实现总结1. 前言默认情况下,Spring加载Bean的顺

HTML5实现的移动端购物车自动结算功能示例代码

《HTML5实现的移动端购物车自动结算功能示例代码》本文介绍HTML5实现移动端购物车自动结算,通过WebStorage、事件监听、DOM操作等技术,确保实时更新与数据同步,优化性能及无障碍性,提升用... 目录1. 移动端购物车自动结算概述2. 数据存储与状态保存机制2.1 浏览器端的数据存储方式2.1.