首页 > 其他分享 >Unity 3D定点数物理引擎实战系列4

Unity 3D定点数物理引擎实战系列4

时间:2023-10-27 15:14:51浏览次数:40  
标签:Convert 代码 Entity Unity 源码 定点数 3D 物理

1.4基于Unity整合BEPUphysicsint物理引擎实战

 

上一节我们详细的讲解BEPUphysicsint 的物理事件。此物理引擎会产生了碰撞事件与非碰撞事件,碰撞事件大家好理解,非碰撞事件例如: 物理Entity的update事件,Entity的activation/deactivation事件等。本节课来实战如何编译BEPUphysicsint源码到自己的项目, 如何整合物理引擎与Unity图形渲染。本文分成4个部分:

 

1: 源码编译, 将github上的源码下载下来编译到Unity项目;

2: 在Unity中初始化BEPUphysicsint的物理世界;

3: 编写PhyBoxEntity组件创建物理Entity并同步Unity图像;

4: 使用Unity自带碰撞器组件,来生成创建PhyBoxEntity;

 

 

创建项目,源码编译BEPUphysicsint

 

  准备源码,我们从github上面下载下来BEPUphysicsint的源码,下载地址为:

https://github.com/sam-vdp/bepuphysics1int

下载下来以后,解压开来代码截图,如图1.4-1: 

 

 

下载下来源码后怎么用到Unity(百度也百度不到),我的方法是先分析文件夹,搞清楚每个文件夹的大概的功能与作用,看下项目,打开vs的工程BEPUphysics.sln看下源码项目如何组织的,如图1.4-2:

 

 

 

 

结合我们的文件夹,我们大胆来猜测分析下代码的作用,哪些我们可能需要的,哪些要删除的。分析时,我一般会先根据名字来猜测,再核对一下里面的代码。BEPUbenchmark/ BEPUfloatBenchmark,我先查找名字benchmark的意思,benchmark的英文有类似单元测试,实例代码的意思,所以这两个文件夹下的代码应该是供我们整合到项目做的实例参考,可以阅读里面的源码,但是不用放Unity源码编译。进一步核实一下里面的代码验证一下自己的判断,打开入口文件Benchmark.cs,里面讲解的就是如何setup物理世界等。如图1.4-3所示:

 

 

 

BEPUik/ BEPUphysics: ik与physics看上去就是物理引擎的核心代码库,打开代码大概验证一下,ik里面有IKJoint.cs相关代码, physics里面有Entity, Space等代码,所以这个应该是核心代码,等会要放入Unity项目工程。如图1.4-4所示

 

 

 

 

经过类似的分析名字与内容,我们发现BEPUutilities(工具代码)+ BEPUik+ BEPUphysics+ FixedMath.Net文件夹所对应的代码为引擎编译所依赖的必须代码, ConversionHelper这里是向量定点与浮点的转换代码,在Unity中可能用到或参考,可以考虑放项目中。

 

  接下来创建Unity项目来编译源码(参考unity 2020.3.33f1版本)。项目创建完成后分好文件夹,这里我分了3个一级目录文件夹: AssetsPackage, Scenes, Scripts,Scripts又分为了3rd, Framework, 两个文件夹,如图1.4-5:

 

 

 

其中3rd用来存放第三方的代码,我在里面又新建了一个BEPU的文件夹,拷贝我们分析的必须代码到Unity项目。

 

不出意外的话,大量的报错,主要分成几类, AssemblyInfo.cs 里面的代码报错,打开这个代码一看,没有什么用,直接连同里面的” Properties”一起删除掉。又发现源码文件夹里面”bin”与”obj”文件夹,这个是源码工程的编译时生成的目录,删掉。如图1.4-7:

 

 

接着就是BoxBoxCollider.cs代码报错,打开代码一看,对比原版,发现原版里面有个编译的宏开关ALLOWUNSAFE,原版本打开了,Unity版本没有,我们在Unity PlayerSetting里面加上这个宏,如图1.4-8:

 

 

 

 

经过这些操作,代码很神奇的就全部编译过了。接下来往Unity里来构建我们的物理世界了。

 

 

在Unity中初始化BEPUphysicsint的物理世界

 

这个简单多了,打开BEPUphysicsint的benchmark代码参考,copy过来就可以了。初始化的时,主要做的事情有: 构建物理世界, 配置物理世界重力,设置物理迭代的参数,关闭原来Unity自带物理引擎, Update中迭代物理世界。这里是全局执行一次,新建一个BEPUPhyMgr.cs的全局单例,用来做初始化。代码实现在Framework/BEPUWrapper文件夹下,直接上代码:

 

 

public class BEPUPhyMgr : MonoBehaviour

 

{

 

    public BEPUphysics.Space space;

 

    public static BEPUPhyMgr Instance = null;

 

 

 

    public void Awake() {

 

        if (BEPUPhyMgr.Instance != null) {

 

            return;

 

        }

 

 

 

        Physics.autoSimulation = false; // 关闭原来物理引擎迭代;

 

        // Physics.autoSyncTransforms = false; // 关闭射线检测功能

 

 

 

        BEPUPhyMgr.Instance = this; // 初始化单例

 

        this.space = new BEPUphysics.Space(); // 创建物理世界

 

        this.space.ForceUpdater.gravity = new BEPUutilities.Vector3(0, -9.81m, 0); // 配置重力

 

        this.space.TimeStepSettings.TimeStepDuration = 1 / 60m; // 设置迭代时间间隔

 

 

 

    }

 

 

 

 

 

    public void Update() {

 

        this.space.Update(); // 模拟迭代物理世界

 

    }

 

}

 

在Unity中创建一个GameApp空节点,挂这个BEPUPhyMgr组件来做初始化(这里是测试,具体在项目框架中做初始化)。创建一个Cube与Plane, 删除掉它原来的物理碰撞器。如图1.4-9所示:

 

 

 

创建物理Entity并同步Unity图像

 

  构建完物理世界后,我们就是要往物理世界里面放物理的Entity, 物理世界迭代计算Entity,然后再把Entity里面的位置,旋转等信息同步到Unity的图像节点的Transform中,这样物理的Entity就可以带着图像的节点移动了。这个需要我们开发者来自己开发,同时Entity的类型有很多,我们以Box Entity为例,主要做2件事情:

(1) 根据用户给定的物体大小与物体是否可移动,来创建对应的物理Entity;

(2) 对于可移动的物体,我们每次update同步Entity位置等到Unity Transform组件;

新建一个PhyBoxEntity.cs的组件到Framework/BEPUWrapper文件夹下。构建物理Entity Box需要长,宽,高,我们也设定相关参数,物体是否可以移动,給一个bool給用户选择。有时候我们要调整物理的中心(与标准Unity物理引擎类似),我们加个相当于Unity节点的中心偏移:

    BEPUphysics.Entities.Prefabs.Box box;

 

    public bool isStatic = false;

    private float width = 1;

    private float height = 1;

    private float length = 1;

private float centerX = 0, centerY = 0, centerZ = 0;

 

根据这些设定,我们来使用BEPUphysics来创建Box的Entity,其它的类似,并加入到物理世界。

if (isStatic) {

            this.box = new BEPUphysics.Entities.Prefabs.Box(new BEPUutilities.Vector3(0, 0, 0), System.Convert.ToDecimal(this.width), System.Convert.ToDecimal(this.height), System.Convert.ToDecimal(this.length));

        }

        else {

            this.box = new BEPUphysics.Entities.Prefabs.Box(new BEPUutilities.Vector3(0, 0, 0), System.Convert.ToDecimal(this.width), System.Convert.ToDecimal(this.height), System.Convert.ToDecimal(this.length), 1);

        }

 

BEPUPhyMgr.Instance.space.Add(box);

 

 

 

接下来同步Unity的图像节点的Transform到物理Entity,直接把物体Transform的位置同步设置到Entity里面。

       Vector3 pos = this.transform.position;

        this.box.position = new BEPUutilities.Vector3(System.Convert.ToDecimal(pos.x + this.centerX), System.Convert.ToDecimal(pos.y + this.centerY), System.Convert.ToDecimal(pos.z + this.centerZ));

最后对于移动的物体,我们在LateUpdate里面同步物理Entity位置到Unity Transform。    public void LateUpdate() {

        if (this.isStatic) {

            return;

        }

        BEPUutilities.Vector3 worldPos = this.box.position;

        // Debug.Log(worldPos.ToString());

        

        double x = System.Convert.ToDouble((decimal)worldPos.X);

        double y = System.Convert.ToDouble((decimal)worldPos.Y);

        double z = System.Convert.ToDouble((decimal)worldPos.Z);

 

        

        this.transform.position = new Vector3((float)x - this.centerX, (float)y - this.centerY, (float)z - this.centerY);

    }

挂一个BoxPhyEntity组件到Cube节点,你会神奇的发现Cube在物理引擎的迭代作用下掉下去了。

 

使用Unity自带碰撞器组件,来生成创建PhyBoxEntity

 

  这里还有一个问题,就是物理Entity的大小设置无法像Unity自带的物理引擎一样,可视化编辑,其实这个问题很好解决,我们的策略是添加Unity对应的物理碰撞器组件 (如: BoxCollider),利用Unity做好的功能特性来编辑物理Entity大小与中心,运行的时候读取数据即可。这里我做了一个简单的转化,代码如下:

[RequireComponent(typeof(BoxCollider))]

添加PhyBoxEntity组件的时候,必要要添加BoxCollider组件。

BoxCollider boxPhy = this.GetComponent<BoxCollider>();

        this.width = boxPhy.size.x;

        this.height = boxPhy.size.y;

        this.length = boxPhy.size.z;

 

        this.centerX = boxPhy.center.x;

        this.centerY = boxPhy.center.y;

        this.centerZ = boxPhy.center.z;

初始化的时候,从碰撞器中读取大小数据,这样就可以关联到Unity自带的形状。后续你开发PhyBoxEntity, PhySphereEntity等都可以通过这样的方式来整合。同时这些Entity可以继承一个基类PhyBaseEntity, 来做些设计上的调整。最后上下效果图,我添加一个Cube与一个平面到物理世界,效果如下:

 

 

标签:Convert,代码,Entity,Unity,源码,定点数,3D,物理
From: https://www.cnblogs.com/rainy1unity/p/17792387.html

相关文章

  • Unity DOTS系列之Aspect核心机制分析
      最近DOTS发布了正式的版本,我们来分享一下DOTS里面Aspect机制,方便大家上手学习掌握UnityDOTS开发。Aspect 机制概述当我们使用ECS开发的时候,编写某个功能可能需要某个entity的一些组件,如果我们一个个组件的查询出来,可能参数会写很长。如果我们编写某个功能的时候,需要enti......
  • 循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(6) -- 窗口控
    在我们窗口新增、编辑状态下的时候,我们往往会根据是否修改过的痕迹-也就是脏数据状态进行跟踪,如果用户发生了数据修改,我们在用户退出窗口的时候,提供用户是否丢弃修改还是继续编辑,这样在一些重要录入时的时候,可以避免用户不小心关掉窗口,导致窗口的数据要重新录入的尴尬场景。本篇随......
  • 循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(4) -- 实现Da
    在我们设计软件的很多地方,都看到需要对表格数据进行导入和导出的操作,主要是方便客户进行快速的数据处理和分享的功能,本篇随笔介绍基于WPF实现DataGrid数据的导入和导出操作。1、系统界面设计在我们实现数据的导入导出功能之前,我们在主界面需要提供给客户相关的操作按钮,如下界面所示......
  • 循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(5) -- 树列表
    在我们展示一些参考信息的时候,有所会用树形列表来展示结构信息,如对于有父子关系的多层级部门机构,以及一些常用如字典大类节点,也都可以利用树形列表的方式进行展示,本篇随笔介绍基于WPF的方式,使用TreeView来洗实现结构信息的展示,以及对它的菜单进行的设置、过滤查询等功能的实现逻辑......
  • BEPUphysicsint定点数3D物理引擎介绍
    帧同步的游戏中如果用物理引擎,为了保证不同设备上的结果一致,需要采用定点数来计算迭代游戏过程中的物理运算。也就是我们通常说的定点数物理引擎(确定性物理引擎)。本系列教程给大家详细的讲解如何在你的项目中内置一个确定性物理引擎。确定性物理引擎我们使用github上开源的物理......
  • Unity ECS内存分配器原理详解
    ECS为什么会高效,性能好,Entity的内存布局与分配就是非常重要的部分,今天我们一起来分析一下UnityECS架构里面如何来做高效的内存分配器。这种思路也可以给我们做内存分配提供很好的一个思路。1:ECS 里面基本的一些概念UnityECS框架里面有几个重要的概念:Entity,ComponentD......
  • Unity RVO动态避障技术方案介绍
    我们在开发游戏的时候经常会遇到这样的问题,当我们寻路的时候,其它人也在寻路,如何避免不从其它人的位置穿过。这个叫做动态避障,目前主流的解决方案就是RVO。本节我们来介绍一些Unity实现RVO的相关资料,后续在详细的讲解ROV算法的原理。先給大家介绍一个RVO2Library的项目,这个项......
  • CF1883D In Love
    思路如果每一次加或者删一个区间,再去暴力找有没有互不相交的区间的话,铁定TLE。那么,我们考虑维护有多少对互不相交的区间,那么每次加或者删一个区间,就去算这个区间对答案的贡献,然后再看答案是否为\(0\)即可快速判断有没有互不相交的区间。现在考虑如何计算一个新加入或者删去......
  • Civil 3d 创建装配并添加、镜像部件
    英文论坛中有人提的问题,问题我也看不懂,就截图回复了一下,后来他进一步描述了问题,我大致清楚了,虽然我之前使用代码创建装配并添加过部件,但之前的操作仅添加一个部件,如何添加更多的部件并镜像部件,确实没有尝试过,于是试着写了以下测试代码,分享给大家。publicvoidC_AssTe......
  • 3DCAT+东风日产:共建线上个性化订车实时云渲染方案
    近年来,随着5G网络和云计算技术的不断发展,交互式3D实时云看车正在成为一种新的看车方式。与传统的到4S店实地考察不同,消费者可以足不出户,通过网络与终端设备即可实现全方位展示、自选汽车配色、模拟效果、快捷选车并进行个性化定制。3DCAT实时渲染云作为一家专注于为汽车行业提供......