最近一直在研究unity3d,很强大的一款3d引擎。本篇介绍基于此引擎的一个物理赛车驱动算法。
建模丑了点...
首先,我们需要先弄懂汽车的工作原理。每个汽车的动力来自引擎。引擎动力的量化我们用力矩来表示。引擎通过变速箱把力传给轮子,这样轮子就转了起来,整个汽车就可以动了。
需要注意的有以下几个点:
1,加减档系统。所谓变速箱其实可以抽象为一个数组,每个数表示引擎力与轮子力的线性对应关系。汽车一般为6个档位,那么我们数组的大小也应该是6。
动力来自引擎,通过当前档位计算出轮子的力矩,从而使轮子转动。
另一方面我们捕获轮子的转速,通过当前档位再计算出引擎当前的转速。我们限定了引擎的最大与最小转速,规定:引擎达到最大转速的时候,加档;减速到最小转速的时候,减档。
这样就实现了汽车的自动加减档,当然也可以做成手动的。
2,unity3d的wheelCollider(轮子碰撞器)。我们可以通过给wheelCollider一个力,它就可以自动实现滚动的物理效果,同时,我们可以通过代码捕获它的转速,这恰恰符合我们的要求。
更强大的是,我们可以设置轮子的横向与纵向摩擦力,设置轮子的悬挂系统参数,通过这些设置,可以模拟汽车的避震,漂移等效果。
suspension Spring 悬挂系统;
forwardFriction 轮子的纵向摩擦力(车子前进后退方向的摩擦力);
sideWays Friction 轮子的横向摩擦力;
Extremun Slip;Extremum Value;Asymptote Slip;Asymptote Value这四个值其实是坐标系上的两个坐标,他们确定了一个滑动距离与摩擦力的关系曲线。
stiffness Factor可以整体调节摩擦曲线的倍数。
3,unity3d的Rigidbody(刚体)。我们可以赋予刚体质量,并且刚体是受重力影响的。车重的不同也会影响汽车的行驶效果!我们一般按真实世界的数值来,例如汽车的mass我们可以设为6000左右(单位为kg),包括模型的尺寸,也最好与真实尺寸一致(单位为米)。这样我们才能获得更真实的物理效果。
下面我们分享代码(来源csdn):
using UnityEngine; using System.Collections; public class CarControl : MonoBehaviour { //操纵前轮,用于转向 public WheelCollider FrontLeftWheel; public WheelCollider FrontRightWheel; public WheelCollider BackLeftWheel; public WheelCollider BackRightWheel; //齿轮数组 public float[] GearRatio; //当前档位 public int CurrentGear=0; public float EngineTorgue=600.0f; public float MaxEngineRPM=3000.0f; public float MinEngineRPM=1000.0f; private float EngineRPM=0.0f; // Use this for initialization void Start () { //设置车的重心,使车更稳定 Vector3 centerOfMass=rigidbody.centerOfMass; centerOfMass.y=-1.5f; rigidbody.centerOfMass=centerOfMass; } // Update is called once per frame void Update () { //限制车的最大速度,调整阻力可能不是最好的做法。但它很简单,而且不会干扰物理系统的运行。 rigidbody.drag=rigidbody.velocity.magnitude/250; //通过两个轮子的平均rpm,计算引擎rpm,然后切换档位 EngineRPM=(FrontLeftWheel.rpm+FrontRightWheel.rpm)/2*GearRatio[CurrentGear]; ShiftGears(); //设置换档的声音 audio.pitch=Mathf.Abs(EngineRPM/MaxEngineRPM)+1.0f; if(audio.pitch>2.0) { audio.pitch=2.0f; } //最后设置轮子转动力矩。引擎力矩除以当前档位,乘以用户输入值。 //轮子力矩提供一个汽车前进的力。轮子的转动又会提高档位。 BackLeftWheel.motorTorque=EngineTorgue/GearRatio[CurrentGear]*Input.GetAxis("Vertical"); BackRightWheel.motorTorque=EngineTorgue/GearRatio[CurrentGear]*Input.GetAxis("Vertical"); //转动角度是任意数乘以用户输入值 FrontLeftWheel.steerAngle=20*Input.GetAxis("Horizontal"); FrontRightWheel.steerAngle=20*Input.GetAxis("Horizontal"); } void ShiftGears() { int AppropriateGear=CurrentGear; if(EngineRPM>=MaxEngineRPM) { AppropriateGear=CurrentGear; for(int i=0;i<GearRatio.Length;i++) { if(FrontLeftWheel.rpm*GearRatio[i]<MaxEngineRPM) { AppropriateGear=i; break; } } CurrentGear=AppropriateGear; } if(EngineRPM<=MaxEngineRPM) { AppropriateGear=CurrentGear; for(int j=GearRatio.Length-1;j>=0;j--) { if(FrontLeftWheel.rpm*GearRatio[j]>MinEngineRPM) { AppropriateGear=j; break; } } CurrentGear=AppropriateGear; } } }