Unity经典游戏编程之:球球大作战

版权声明:

  • 本文原创发布于博客园"优梦创客"的博客空间(网址:http://www.cnblogs.com/raymondking123/)以及微信公众号"优梦创客"
  • 您可以自由转载,但必须加入完整的版权声明!

首先:

  • alt + 鼠标拖拽 视角旋转
  • 按住鼠标中间,并移动鼠标,可以平移视图
  • 按住鼠标右键,wasdqe 可以在6个方向上移动

  • 在 Hierarchy视图中,鼠标左键先选中一个 物体,按下 shift ,再 鼠标左键选中另外一个 物体,即可选中 多个 物体,进行操作。此时把在右侧Inspector添加组件就可为所有GameObject添加Component

  • unity默认游戏单位为米
  • plane默认长宽是10米 Capsule默认是2个1米的半球加一个1m的圆柱体组成 cube Sphere
  • unity中的Skybox也会提供一些天光(各个方向上微弱的光)

  • Global坐标系永远和场景的坐标系保持一致 (随着世界坐标系选择)
  • local坐标系总是与物体自身方向保持平行

  • persp是透视模式,近大远小
  • Iso是平行投影

  • materials 材质

  • 灯光的颜色与物体的颜色向乘得到最后显示给用户的颜色(颜色分量相乘,物体之所以显示红色是因为其反射红色的光)

  • Time.timeScale //时间流逝的规模。这可以用于慢动作效果
  • 当timeScale为1.0时,时间与实时一样快。当timeScale为0.5时,时间比实时慢2倍。
  • 当timeScale设置为零时,如果所有功能都与帧速率无关,则游戏基本暂停。
  • 除了realtimeSinceStartup,timeScale影响Time类的所有时间和delta时间测量变量。
  • 如果降低timeScale,Time.fixedDeltaTime也会降低相同的量。设置为零时,不会调用- fixedupdate

  • 小球移动:
    private void FixedUpdate()
    {
        Time.timeScale =0.5f;      //子弹时间,加速减速均可用该值实现
        if (Input.GetKey(KeyCode.LeftArrow))            //unity事先就检测到了所有按键,所以在这可以直接判断按键是否按下,用if判断就可以同时施加不同方向上的力   //GetKey是检测按键的持续情况,GetKeyUp和GetKeyDown是只有在按键的上、下边缘才会检测,可以做触发器,但不宜实时控制运动
        {
            rigidbody.AddForce(-moveForce * Time.fixedDeltaTime, 0, 0);   //乘以FixedDeltaTime的意义就在于时间可以缩放,加了个权重
        }
        if (Input.GetKey(KeyCode.RightArrow))
        {
            rigidbody.AddForce(moveForce * Time.fixedDeltaTime, 0, 0);
        }
        if (Input.GetKey(KeyCode.UpArrow))
        {
            rigidbody.AddForce(0, 0, moveForce * Time.fixedDeltaTime);
        }
        if (Input.GetKey(KeyCode.DownArrow))
        {
            rigidbody.AddForce(0, 0, -moveForce * Time.fixedDeltaTime);
        }
        if (Input.GetKey(KeyCode.Space))
        {
            rigidbody.AddForce(0, jumpForce * Time.fixedDeltaTime, 0);
        }
    }
  • Physic Material //物理学材料是用来调整摩擦和碰撞对象的反弹效应。 //属于collider一个属性
  • 属性: 功能:
  • 动态摩擦 已经移动时使用的摩擦力。通常是从0到1的值。零值感觉像冰一样,值1将使它非常快地停下来,除非大量的力或重力推动物体。
  • 静摩擦力 物体静止在表面上时使用的摩擦力。通常是0到1之间的值。零值感觉像冰,值1会使对象移动变得非常困难。
  • 反弹力 表面有多么有弹性?值0不会反弹。值1将在没有任何能量损失的情况下反弹,但是可以预期某些近似值,但这可能会为模拟增加少量能量。
  • 摩擦组合 如何组合两个碰撞物体的摩擦力。
    - 平均 两个摩擦值是平均值。
    - 最低 使用这两个值中最小的一个。
    - 最大 使用这两个值中最大的一个。
    - 乘以 摩擦值相互相乘。
  • 弹跳组合 如何组合两个碰撞对象的弹性。它具有与摩擦组合模式相同的模式
  • 摩擦力是防止表面相互滑落的量。尝试堆叠对象时,此值很重要。摩擦有两种形式,动态和静态。物体静止时使用静摩擦力。它会阻止对象开始移动。如果对物体施加足够大的力,它将开始移动。此时动态摩擦将发挥作用。动态摩擦力现在会在与另一个物体接触时尝试减慢物体的速度。
  • 当两个物体接触时,根据所选择的模式对它们两者施加相同的弹性和摩擦效果。当两个接触的碰撞器具有不同的组合模式设置时,存在一种特殊情况。在这种特殊情况下,使用具有最高优先级的功能。优先顺序如下:平均 < 最小 < 乘以 < 最大值。例如,如果一个材质具有平均值但另一个具有最大值,则要使用的组合函数为最大值,因为它具有更高的优先级。 //力的作用是相互的,所以当两个物体的摩擦力系数不同时,有不同的计算摩擦力的方法,但是加在这两个物体上的摩擦力大小相等方向相反

  • 让摄像机跟着小球动,如果作为子节点,会收到小球移动和旋转的影响,我们不希望收到旋转影响,所以我们只需要摄像机跟随小球的位置移动就可: //一个平行四边形的移动

public class MainCamera : MonoBehaviour
{
    public Transform player;

    private Vector3 offset;
    private Vector3 playerStartPosition;
    private Vector3 cameraStartPosition;
    // Start is called before the first frame update
    void Start()
    {
        playerStartPosition = player.position;
        cameraStartPosition = this.transform.position;

        offset = this.transform.position - player.position;
    }

    // Update is called once per frame
    void Update()
    {
   //     this.transform.position = cameraStartPosition + (player.position - playerStartPosition);            //用Offset和相机起始位置偏移均可,因为是个平行四边形
        this.transform.position = player.position + offset;
    }
}
  • 为了不影响小球在吃道具时的速度,需要把道具设置为trigger

  • collider性能优化: //collider相当于mesh的网格模型,只不过不是用来画图而是用来计算碰撞的,所有的静态物体生成一个collider静态网格,当有动态物体碰撞到静态物体时,根据该网格计算动态物体该如何进行反应
  • unity中计算场景中所有的静态碰撞器的碰撞体并把他们保持在缓冲中,这使得所有的静态碰撞体不能够移动并节省每帧计算碰撞体的开销 //unity保存到缓存后,不会每帧都再次计算静态碰撞体
  • 我们的错误在于让道具旋转了
  • 任何时候,当我们移动、旋转、缩放一个静态碰撞器时,unity会再次重新计算所有的静态碰撞器,并更新静态碰撞器的缓存,而重新计算缓存是有开销的
  • 我们可以移动、旋转或缩放动态碰撞器并且unity不会重新缓存任何碰撞器的碰撞体 //动态碰撞器实时计算
  • unity鼓励我们移动碰撞体,只要我们在移动之前声明该碰撞器是动态的即可
  • 要做到这点,我们只要使用Rigidbody组件,任何带有Collider和Rigidbody的游戏对象都被认为是动态的,任何不带rigidbody而带有Collider的组件的游戏对象都被认为是静态的,而我们的道具就被认为是静态的,因此unity会每帧重新计算静态碰撞器的缓存
  • 解决方案:
  • 为道具添加Rigidbody,其实变为动态碰撞器,
  • 此时立方体会掉落,因为Rigidbody是有重力的,我们可以设置rigidbody.useGravity = false,这会阻止立方体下落,但是此时立方体虽然不受重力了,但是任然会受其他类型物理力的影响且会带来物理开销
  • 更好的解决方案为设置:Rigidbody.IsKinemaitc = true;
  • 一个Kinematic的刚体不受物理力的影响,而只受图形变换的影响(如动画和Transform组件)
    这一点用在带有触发器或碰撞器的物体上很好,如电梯或平台的自动移动就是用图形动画(而非物理运算)实现的

  • 总结:
    • 静态物体不应具有Rigidbody
    • 动态物体应具有Rigidbody
    • 受物理力影响的Rigidbody应使用默认设置(Kinematic为false)
    • 受图形变换影响的Rigidbody应设置Kinematic为true;

转载于:https://www.cnblogs.com/raymondking123/p/11314972.html

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值