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

相关文章

基于Python开发Windows屏幕控制工具

《基于Python开发Windows屏幕控制工具》在数字化办公时代,屏幕管理已成为提升工作效率和保护眼睛健康的重要环节,本文将分享一个基于Python和PySide6开发的Windows屏幕控制工具,... 目录概述功能亮点界面展示实现步骤详解1. 环境准备2. 亮度控制模块3. 息屏功能实现4. 息屏时间

Python远程控制MySQL的完整指南

《Python远程控制MySQL的完整指南》MySQL是最流行的关系型数据库之一,Python通过多种方式可以与MySQL进行交互,下面小编就为大家详细介绍一下Python操作MySQL的常用方法和最... 目录1. 准备工作2. 连接mysql数据库使用mysql-connector使用PyMySQL3.

如何搭建并配置HTTPD文件服务及访问权限控制

《如何搭建并配置HTTPD文件服务及访问权限控制》:本文主要介绍如何搭建并配置HTTPD文件服务及访问权限控制的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、安装HTTPD服务二、HTTPD服务目录结构三、配置修改四、服务启动五、基于用户访问权限控制六、

MySQL精准控制Binlog日志数量的三种方案

《MySQL精准控制Binlog日志数量的三种方案》作为数据库管理员,你是否经常为服务器磁盘爆满而抓狂?Binlog就像数据库的“黑匣子”,默默记录着每一次数据变动,但若放任不管,几天内这些日志文件就... 目录 一招修改配置文件:永久生效的控制术1.定位my.cnf文件2.添加核心参数不重启热更新:高手应

双系统电脑中把Ubuntu装进外接移动固态硬盘的全过程

《双系统电脑中把Ubuntu装进外接移动固态硬盘的全过程》:本文主要介绍如何在Windows11系统中使用VMware17创建虚拟机,并在虚拟机中安装Ubuntu22.04桌面版或Ubunt... 目录一、首先win11中安装vmware17二、磁盘分区三、保存四、使用虚拟机进行系统安装五、遇见的错误和解决

使用FileChannel实现文件的复制和移动方式

《使用FileChannel实现文件的复制和移动方式》:本文主要介绍使用FileChannel实现文件的复制和移动方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录使用 FileChannel 实现文件复制代码解释使用 FileChannel 实现文件移动代码解释

SpringBoot请求参数接收控制指南分享

《SpringBoot请求参数接收控制指南分享》:本文主要介绍SpringBoot请求参数接收控制指南,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring Boot 请求参数接收控制指南1. 概述2. 有注解时参数接收方式对比3. 无注解时接收参数默认位置

Spring Security+JWT如何实现前后端分离权限控制

《SpringSecurity+JWT如何实现前后端分离权限控制》本篇将手把手教你用SpringSecurity+JWT搭建一套完整的登录认证与权限控制体系,具有很好的参考价值,希望对大家... 目录Spring Security+JWT实现前后端分离权限控制实战一、为什么要用 JWT?二、JWT 基本结构

Android实现两台手机屏幕共享和远程控制功能

《Android实现两台手机屏幕共享和远程控制功能》在远程协助、在线教学、技术支持等多种场景下,实时获得另一部移动设备的屏幕画面,并对其进行操作,具有极高的应用价值,本项目旨在实现两台Android手... 目录一、项目概述二、相关知识2.1 MediaProjection API2.2 Socket 网络

用js控制视频播放进度基本示例代码

《用js控制视频播放进度基本示例代码》写前端的时候,很多的时候是需要支持要网页视频播放的功能,下面这篇文章主要给大家介绍了关于用js控制视频播放进度的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言html部分:JavaScript部分:注意:总结前言在javascript中控制视频播放