Box2D物理碰撞基础知识

发表于2016-07-08
评论0 2.4k浏览
一、Box2D简介
  Box2D是一个用于游戏的2D刚体仿真库。它可以使游戏中物体的运动更加逼真。 Box2d有C++,flash和Java等版本。
    Box2D会自动管理各个物体的碰撞,弹跳等物理状态,我们只需要创建各种刚体。创建刚体可以通过b2BodyDef.userData来创建。b2Shape.SetAsBox(width,height)指的是物体的半宽和半高,所以在计算时都要乘以2。由于Box2D本身的限制,运算时要进行长度换算。

二、Box2D核心概念
刚体(rigid body)
  一块十分坚硬的物质,它上面的任何两点之间的距离都是完全不变的。它们就像钻石那样坚硬。
形状(shape)
  一块严格依附于物体(body)的 2D 碰撞几何结构(collisiongeometry)。形状具有摩擦(friction)和恢 复(restitution)的材料性质。
约束(constraint)
  一个约束(constraint)就是消除物体自由度的物理连接。在 2D中,一个物体有 3 个自由度。如果我 们把一个物体钉在墙上(像摆 那样) ,那我们就把它约束到了墙上。这样,此物体就只能绕着这个钉子旋 转,所以这个约束消除了它 2 个自由度。
接触约束(contact constraint)
  一个防止刚体穿透,以及用于模拟摩擦(friction)和恢复(restitution)的特殊约束。你永远都不必创建 一个接触约束,它们会自动被 Box2D 创建。
关节(joint)
  它是一种用于把两个或多个物体固定到一起的约束。Box2D支持的关节类型有:旋转,棱柱,距离等 等。关节可以支持限制(limits)和马达(motors)。
关节限制(joint limit)
  一个关节限制(joint limit)限定了一个关节的运动范围。例如人类的胳膊肘只能做某一范围角度的运动。
关节马达(joint motor)
  一个关节马达能依照关节的自由度来驱动所连接的物体。例如,你可以使用一个马达来驱动一个肘的旋转。
世界(world)
  一个物理世界就是物体,形状和约束相互作用的集合。Box2D 支持创建多个世界,但这通常是不必要 的。
积分器(integrator)
  积分器在离散的时间点上模拟物理方程,它将 与游戏动画循环一同运行。通常来说游戏物理引擎需要至少  60Hz 的速度,也就是 1/60 的时间步。
约束求解器(constraint solver)
  约束求解器用于解决模拟中的所有 约束,一次一个。要得到良好的解,需要迭代所有约束多次。建议的  Box2D 迭代次数是 10 次。

三、Box2D注意事项
  全局的对象的构造函数作了三件事情:
1、一个在b2AABB类中的实例构建的坐标系统  
2、一个定义重力的向量,这是一个b2Vec2类构建的实例。  
3、一个布尔变量来定义对象是否"沉迷"。(如果你设置为true,对象将会沉迷)。
4、执行Step()函数,每一帧都会更新所有的Body在world中的位置。 
5、Box2D中的单位为米,30像素==1米。所以在box2D中经常会看到 坐标点乘以30,这样就不奇怪为什么用30而不是15或者20的了。

四、Box2D   Body
    创建好World后,可以向World内部添加任何球体或者盒子,以及你想到的任何形状的东西,那我就需要定义一个Body。
  一个body体大概需要做2-4件事情:
1、定义一个形状
2、一个(x,y)的位置
3、角度
4、一个预制的Sprite对象
    其中3、4是可选操作。这里我可能不会讲到Box2D内设的一些画图类库,因为我们在实际的操作中,可能用的都是自定义的。内设的类库一般用来模拟比较好,进行实际开发可能不适合,所以大家有想研究内设画图类库的可能需要自己去研究一下了。

五、Box2D      b2ShapeDef
  如果你想在你的游戏或者其他的什么中具有一些有特色的东西,你可以通过综合形状定义Body来制作一个Sprite。形状的定义,有3种类型的形状定义,他们都是扩展的b2ShapeDef基类。
  b2BoxDef类具有4个重要的属性:
1、SetAsBox(设定边框):这是一个向量,本质上说他就是一个形状的中心坐标
2、Density(密度):在碰撞的等式中使用密度*面积=质量,密度如果是0或者null,将会是一个静止的对象。  
3、Friction(摩擦力):这用来计算两个对象之间的摩擦,可以在0.0-1.0之间调整它们。
4、Restitution(弹性):这是调整对象弹性程度的属性,可以在0.0-1.0之间调整它们。       b2CircleDef类中有一个不同的属性,代替SetAsBox是他的Radius(半径)。
     b2PolyDef类具有一个顶点数组(最大是8)来代替SetAsBox和Radius。这些顶点都是b2Vec2类型的对象。  
  以下通过实例来讲述Box2D的基本用法,在打开源文件进行测试时,需要把Box2D的类库放在目录下面,程序才能正常运行。

六、Box2D    Hello World
  Hello World实例中会讲述Box2D 的基础用法。
1、准备好Box2D类库
2、在画自己的形状时,如果不想自己给自己添乱,定义的形状width或者height尽量都能被30整除,这样便于我们的计算
3、按前面的讲述,我们需要创建的全局变量里面要有一个World,记分器、约束求解器。同样还要准备一个盒子(b2AABB)、重力(gravity)、是否能睡眠(doSleep)
4、添加形状需要用到b2Body、b2BodyDef、b2PolygonDef(多边形)、b2CircleDef(圆形)这几个常用的类。
     Hello World里面的难点应该就是第4点,希望大家能注意一下,如果自己做例子时Rect与Circle应该怎样添加。例子中创建了静态地面与动态的物体。upData函数让整个世界运转起来,你可能不需要知道for循环里面到底是怎么运作的,只需要知道这个for循环让这个对象开始在世界中进行模拟就可以了(这个for循环是必须的)。Box2D  鼠标与刚体交互之移除选中刚体
  此示例在Hello World的基础之上进行扩展,关键操作为getBodyAtMouse()函数,早起看过C++版本而写的此函数。现在网络上可能也能搜到类似的,但是都讲的不是很详细。不过这个函数你有可能不需要到底是怎样运行,仅仅知道这个函数返回与鼠标点相交的刚体就可以了。下面解释一下getBodyAtMouse()函数的机制:
     当鼠标点击时,得到当前鼠标的坐标点,在此鼠标点产生一个很小的刚体,半径在0.001内就有效,然后遍历整个世界内部与小刚体产生碰撞的刚体,最后返回碰撞的刚体。返回的刚体都是唯一的,说唯一是因为世界内部的对象都是刚体,既然是刚体,就不会产生两个刚体重叠或者相交的情况,所以大家不要感到我说的唯一 很奇怪。

七、Box2D  鼠标与刚体交互之拖拽刚体
  由于Box2D是不直接与鼠标交互的,而是通过鼠标关节
  b2MouseJoint交互的。
  交互的过程由四个步骤完成:
  第一步:获取鼠标单击处的刚体。
  第二步:创建鼠标关节。
  第三步:控制鼠标关节。
  第四步:销毁鼠标关节。
  获取刚体:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
public function getBodyAtMouse(world:b2World,stage:Stage,includeStatic:Boolean=true):b2Body
 
             {
 
                            var mouseb2Vec:b2Vec2 = new b2Vec2(stage.mouseX / 30, stage.mouseY / 30);
 
                            var aabb:b2AABB = new b2AABB();
 
                            aabb.lowerBound.Set(mouseb2Vec.x - 0.001, mouseb2Vec.y - 0.001);
 
                            aabb.upperBound.Set(mouseb2Vec.x + 0.001, mouseb2Vec.y + 0.001);
 
                            var maxCount:int = 10;
 
var shapesArray:Array = new Array();
 
                            var count:int = world.Query(aabb, shapesArray, maxCount);
 
                            var body:b2Body;
 
                            for (var i:int = 0; i < count;i++ )
 
                            {
 
                                if (!shapesArray[i].m_body.IsStatic()||includeStatic)
 
                                {
 
                                               var tShape:b2Shape = shapesArray[i] as b2Shape;
 
                                               var inside:Boolean = tShape.TestPoint(tShape.m_body.GetXForm(), mouseb2Vec);
 
                                               if (inside)
 
                                               {
 
                                                        body = tShape.m_body;
 
                                                        break;
 
                                               }
 
                                }
 
                            }
 
                            return body;
 
             }

  创建关节:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
var lbf_tempBody:b2Body = getBodyAtMouse(lbf_world, stage);
 
                            if (lbf_tempBody)
 
                            {
 
                                     //设置关节
 
                                     var lbf_mouseJointDef:b2MouseJointDef = new b2MouseJointDef();
 
                                     // 设置body1为无碰撞检测形状的静态刚体
 
                                     lbf_mouseJointDef.body1 = lbf_world.GetGroundBody();
 
                                     // 设置body2为当前被检测到被点击的刚体
 
                                     lbf_mouseJointDef.body2 = lbf_tempBody;
 
                                     // 设置鼠标关节的目标位置
 
                                     lbf_mouseJointDef.target.Set(mouseX / 30, mouseY / 30);
 
                                     // 设置鼠标关节的力度
 
                                     lbf_mouseJointDef.maxForce = 10000;
 
                                     // 设置鼠标关节的时间步
 
                                     lbf_mouseJointDef.timeStep = lbf_timeStep;
 
                                     // 在世界中创建这个b2MouseJoint对象
 
                                     lbf_mouseJoint = lbf_world.CreateJoint(lbf_mouseJointDef) as b2MouseJoint
 
                                     
 
                                     lbf_mouseJointDef = null;
 
                                     lbf_tempBody = null;
 
                            }

  控制关节:

1
2
3
4
5
6
7
8
9
10
11
12
//控制鼠标关节很简单,就是在帧循环时间中不断更新鼠标关节的目标位置。
if (lbf_mouseJoint)
{
         lbf_mouseJoint.SetTarget(new b2Vec2(mouseX / 30, mouseY / 30));
}
销毁关节:
if (lbf_mouseJoint)
{
         lbf_world.DestroyJoint(lbf_mouseJoint);
 
         lbf_mouseJoint = null;
}

  具体细节请详看:
Box2D   鼠标与刚体交互之拖拽刚体

八、Box2D    键盘事件
  在Box2D中,刚体不仅会受到重力、碰撞等影响,而且也可以施加一个力对它

1
2
3
4
5
6
//造成影响。可以通过至少以下这几个方法去移动一个刚体:
public function ApplyForce(force:b2Vec2, point:b2Vec2) : void
//向目标(世界位置)施加一个力,如果这个力不是施加在刚体的质心,它将产生一个扭矩并影响其角速度,在此之前,它将自动唤醒刚体。力的单位为N。(1N=1kg*m/s^2)
public function ApplyImpulse(impulse:b2Vec2, point:b2Vec2) : void
//向目标(世界位置)施加一个冲量,它将立刻改变刚体的速度,如果这个力不是施加在刚体的质心,它将影响其角速度,在此之前,它将自动唤醒刚体。冲量的单位为kg * m/s 或 N*s。
public function SetLinearVelocity(v:b2Vec2) : void

  直接设置质心线速度,需要注意的是它是不会自动唤醒刚体的,所以你必须事先唤醒施加对象再设置质心线速度,免得以为方法不执行。遇到问题应如何具体操作:

1
2
3
4
5
6
7
8
//左右移动时:
direction.Set(-2, 0);/direction.Set(2, 0);
player.WakeUp();
player.ApplyForce(direction, player.GetWorldCenter());
//跳跃时:
direction.Set(0, -5);
player.WakeUp();
player.ApplyImpulse(direction, player.GetWorldCenter());

腾讯GAD游戏程序交流群:484290331Gad游戏开发核心用户群

如社区发表内容存在侵权行为,您可以点击这里查看侵权投诉指引

0个评论