Unity使用物理引擎实现多旋翼无人机的模拟飞行

作者:weixin_44411555 时间:2021-09-25 04:11:40 

内容简介

最近在用Unity实现无人机的模拟飞行,但发现站里基本没有完整介绍如何实现该功能的博客,因时间紧迫,就自己简单做了一个仿真(不是完全按照现实物理情景来做,即通过各个螺旋桨旋转产生力带动机体飞行)。下面我会简述完全按现实物理情景实现模拟飞行,并详细描述我自己做的模拟飞行(不完全仿真),给各位提供参考。

现实物理情景的实现——简述(以四旋翼无人机为例)

因为我没有实现1:1仿真,所以这里只介绍思路,希望能给到读者一点启发。

  1. 构建一个四旋翼无人机模型(可以网上下载。或用Maya等建一个)

  2. 将模型设置为Rigidbody和BoxCollider(即实现一个物理模型,记得要先Create一个Plane承住你的模型)

  3. 给四个螺旋桨各写一个脚本,各产生一个向上的力

  4. 通过螺旋桨产生的升力即可带动模型飞行,通过控制不同螺旋桨产生升力的不同,即可实现无人机的各种飞行姿态(这步是最难的,也最花时间)

简化物理情景的实现——详述(以四旋翼无人机为例)

背景:上面的现实模拟最麻烦之处在于要对四个螺旋桨(电机)实现控制并达到平衡,太麻烦了,调试要很久,网上又没有现成的代码(至少免费的我没有发现)。所以我自己简化了一下模型,并实现所需功能。
思路:简化模型的区别在于我不用四个螺旋桨(电机)来各自产生升力,形成合力来带动机体。而是直接将力加在机身上,即跳过了前面那一步,直接用合力来实现无人机的飞行。
实现步骤:以下为实现步骤,考虑到网上好像没有相关的博客和视频教程,我会尽量详细来叙述,所以会有点多。

1.导入一个无人机模型,并给整体设置Rigidbody和BoxCollider(注意最好Collider设成和模型差不多大)。

Unity使用物理引擎实现多旋翼无人机的模拟飞行

2. 写脚本1(Plane_Text)(总共需要写两个脚本,记为脚本1和脚本2),脚本1使其产生力带动机体运动


using System.Collections;
using System.Collections.Generic;
using System.IO.Pipes;
using System.Runtime.CompilerServices;
using System.Security.Cryptography.X509Certificates;
using UnityEngine;
using UnityEngine.UI;

public class Plane_Text : MonoBehaviour
{
private Rigidbody m_Rigidbody;
public float speed = 5;
public float angle = 0;
private float x, y, z;
public Transform target;

void Start()
{
m_Rigidbody = gameObject.GetComponent<Rigidbody>();//获取组件
}

void Update()
{
var vec = new Vector3(x, y, z);
m_Rigidbody.AddForce(vec * speed, ForceMode.Force);//无人机朝vec向量方向运动,至于vec的取值由下面各个按键确定
/*******无人机飞机控制*********/
/*******W加油门(上升);S减油门(下降);A左转;D右转*********/
if (Input.GetKeyUp(KeyCode.W))
speed++;
if (Input.GetKeyDown(KeyCode.W))
x = 0;y = 1;z = 0;
if (Input.GetKeyUp(KeyCode.S))
speed--;
if (Input.GetKeyDown(KeyCode.D))
angle++;
this.transform.Rotate(Vector3.up * angle);//按下D键则一直右转,下同
if (Input.GetKeyUp(KeyCode.D))
angle = 0;
this.transform.Rotate(Vector3.up * angle);//松开D键停止右转,下同
if (Input.GetKeyDown(KeyCode.A))
angle--;
this.transform.Rotate(Vector3.up * angle);
if (Input.GetKeyUp(KeyCode.A))
angle = 0;
this.transform.Rotate(Vector3.up * angle);
/*******↑前倾;↓后倾;←左倾;→右倾*********/
if (Input.GetKeyDown(KeyCode.LeftArrow))//按下←键朝向量(-1,2,0)方向运动,即左倾,但该左倾只是运动轨迹上左倾,模型是不会往左倾的,所以需要下脚本2里设置
x = -1; y = 2; z = 0; //这个取值随你定,但x一定是负的且z一定是0,至于x和y的大小就看你想飞行轨迹左倾多少来决定了
if (Input.GetKeyUp(KeyCode.LeftArrow))//当松开←键时,不再左倾,而是按原来(0,1,0)方向,即y轴向上,继续移动
x = 0; y = 1; z = 0;//同样因为是y轴,x和z的值必须为0,y的值看你心情
/**************************************************/
if (Input.GetKeyDown(KeyCode.RightArrow))//按下→键
x = 1; y = 2; z = 0;                     //这些我就不再解释了,跟上面“←键”的是一样的,只是倾斜方向不一样而已
if (Input.GetKeyUp(KeyCode.RightArrow))
x = 0; y = 1; z = 0;
/**************************************************/
if (Input.GetKeyDown(KeyCode.UpArrow))//按下↑键
x = 0;y = 2;z = 1;
if (Input.GetKeyUp(KeyCode.UpArrow))
x = 0; y = 1; z = 0;
/**************************************************/
if (Input.GetKeyDown(KeyCode.DownArrow))//按下↓键
x = 0;y = 2;z = -1;
if (Input.GetKeyUp(KeyCode.DownArrow))
x = 0; y = 1; z = 0;
/**************************************************/
}
}

3.写脚本2(Plane_Move):该脚本是用来实现模型的前后左右倾的


using System.Collections;
using System.Collections.Generic;
using System.IO.Pipes;
using System.Runtime.CompilerServices;
using System.Security.Cryptography.X509Certificates;
using UnityEngine;
using UnityEngine.UI;

public class Plane_Move : MonoBehaviour
{
   private Rigidbody m_Rigidbody;
   void Start()
   {
       m_Rigidbody = gameObject.GetComponent<Rigidbody>();//获取组件
   }

void Update()
   {
/*******↑前倾;↓后倾;←左倾;→右倾*********/
if (Input.GetKeyDown(KeyCode.LeftArrow))
transform.rotation = Quaternion.Euler(0, 0, 10);//绕z轴转10°,即左倾,不一定是10,看你想倾多少就倾多少,别太大就行,下同
if (Input.GetKeyUp(KeyCode.LeftArrow))
transform.rotation = Quaternion.Euler(0, 0, 0);//松开按键恢复竖直向上飞,下同
/**************************************************/
if (Input.GetKeyDown(KeyCode.RightArrow))
this.transform.rotation = Quaternion.Euler(0, 0, -10);
if (Input.GetKeyUp(KeyCode.RightArrow))
this.transform.rotation = Quaternion.Euler(0, 0, 0);
/**************************************************/
if (Input.GetKeyDown(KeyCode.UpArrow))
this.transform.rotation = Quaternion.Euler(10, 0, 0);
if (Input.GetKeyUp(KeyCode.UpArrow))
this.transform.rotation = Quaternion.Euler(0, 0, 0);
/**************************************************/
if (Input.GetKeyDown(KeyCode.DownArrow))
this.transform.rotation = Quaternion.Euler(-10, 0, 0);
if (Input.GetKeyUp(KeyCode.DownArrow))
this.transform.rotation = Quaternion.Euler(0, 0, 0);
/**************************************************/
}
}

4.完成上面3步,我们就会得到两个脚本

Unity使用物理引擎实现多旋翼无人机的模拟飞行

5.然后就是把这两个脚本拖到无人机上(注意两个都要拖!),接着就是运行了(如果是运行不了肯定是你复制了我的代码,但脚本名又和我不一样的缘故)

6.最后运行成功(原谅我不会截动图给你们看运行结果)

7.当然有人会说螺旋桨怎么不会转啊,这也简单;首先给四个螺旋桨的顶点各建一个空物体(每个螺旋桨要绕着它对应的空物体旋转),然后再写一个脚本(Power_all),我这里就不区分正反桨了。(你要区分也行,就是把Vector3.down改成Vector3.up就是另一对桨)


using System.Collections.Generic;
using System.Collections;
using System.Globalization;
using UnityEngine;
using System;
using UnityEngine.UI;
//using  UnityEngine.Debug;

public class Power_all : MonoBehaviour
{
public GameObject target01;//声明顶点
private float n = 10;

// Update is called once per frame
void Update()
{

this.transform.RotateAround(target01.transform.position, Vector3.down, 100f * n * Time.deltaTime);//该函数通过放入一个顶点,以及围绕这个顶点要旋转的x轴并给定旋转速度
 //Vector3向量,在三维坐标系中带有方向和大小的数据,Vector3.up=(0,1,0)、Vector3.down=(0,-1,0)、Vector3.left=(-1,0,0)、Vector3.right=(1,0,0)

}
}

8.将上面这个螺旋桨的脚本分别拖到各个桨叶模型上,然后将各个空物体拉到各个声明的顶点处即可。

9.上述即完成了简化的多旋翼无人机模拟飞行。可能有人会说我第2、3点谁规定了只能前倾一个角度,其实我是这样理解的,如果按照现实情景来完全模拟,确实前倾任意角度,甚至是180°,但前倾角度过大导致的结果是坠机,所以我设置在10°可以理解为安全倾角,当然你也可以自己该代码来实现自己想要的功能,我这里只是提供一个参考。

来源:https://blog.csdn.net/weixin_44411555/article/details/115585726

标签:Unity,物理引擎,飞行
0
投稿

猜你喜欢

  • 从 JVM 中深入探究 Synchronized作用及原理

    2023-07-28 17:22:33
  • C#/VB.NET 实现彩色PDF转为灰度PDF

    2021-07-23 08:23:49
  • Android利用属性动画实现优酷菜单

    2022-06-15 13:16:44
  • Java中StringUtils工具类进行String为空的判断解析

    2022-10-16 06:56:53
  • 浅谈Java封装、继承、多态特性

    2023-10-09 13:59:35
  • Java实现窗体程序显示日历

    2022-09-14 11:01:59
  • Spring Boot 利用 XML 方式整合 MyBatis

    2023-02-09 23:01:18
  • Android串口通信之串口读写实例

    2023-11-04 00:11:03
  • JavaWeb入门教程之分页查询功能的简单实现

    2021-11-11 21:52:23
  • SpringBoot在一定时间内限制接口请求次数的实现示例

    2021-10-12 04:28:52
  • java jdbc连接和使用详细介绍

    2023-01-13 18:12:52
  • HashMap 和 Hashtable的区别

    2022-11-01 06:34:47
  • Android 内存泄漏的几种可能总结

    2022-02-27 21:43:26
  • C#获取Windows进程监听的TCP/UDP端口实例

    2021-11-20 13:06:21
  • Java Comparable 和 Comparator 的详解及区别

    2023-07-05 10:41:44
  • Spring Boot conditional注解用法详解

    2022-03-19 02:32:25
  • Android实现页面滑动切换动画

    2023-12-26 11:47:54
  • C#实现洗牌游戏实例

    2021-07-07 00:45:29
  • springmvc用于方法鉴权的注解拦截器的解决方案代码

    2022-06-02 10:30:18
  • Mybatis 如何传入字符串参数,分割并遍历

    2021-10-08 01:14:45
  • asp之家 软件编程 m.aspxhome.com