Unity中状态机的使用

Unity中状态机的使用

在游戏中,人物的状态是不断变化的,所以写个FSM来管理状态是必要的。
一个有限状态机是一个设备,或者是一个设备模型,具有有限数量的状态,它可以在任何给定的时间根据输入进行操作,使得一个状态变换到另一个状态,或者是使一个输入或者一种行为的发生。一个有限状态机在任何瞬间只能处在一种状态。

设置状态的常量

我们这里设置了几个普通的idle,run,death的常量

public class ConstStrings
{
    //状态 常量
    public const string St_Born = "born";//出生
    public const string St_Idle = "idle";//静止
    public const string St_FunnyIdle = "funnyidle";//休闲动作
    public const string St_Run = "run";//移动
    public const string St_Dead = "death";//死亡
    public const string St_Relive = "relive";//复活
    public const string St_Jump = "jump";//跳跃

}

定义状态图

/// 
/// 状态图,存储角色可以切换的所有状态
/// 
public class CharStateGraph
{
    /// 
    /// 状态图的名称
    /// 
    public string StateGraphName;
    /// 
    /// 存储角色所有的状态
    /// 
    public Dictionary<string, CharSateBase> states = new Dictionary<string, CharSateBase>();

    /// 
    /// 仅用作当前状态的定时器
    /// 切换状态时清空
    /// 
    public uint[] framtimers;
    public uint[] timers;

    public CharStateGraph()
    {
        framtimers = new uint[10];
        timers = new uint[10];
    }
    ~CharStateGraph()
    {
        StopTimers();
        framtimers = null;
        timers = null;
    }

下面来定义一般的状态图,继承于上面的状态图。

/// 
/// 一般状态图
/// 
public class SGCommon : CharStateGraph
{
    public SGCommon()
    {
        CommonStates.AddBorn(states);
        CommonStates.AddIdle(states);
        CommonStates.AddFunnyIdle(states);
        CommonStates.AddAttack(states);
        CommonStates.AddMoveAttack(states);
        CommonStates.AddRun(states);
        CommonStates.AddDeath(states);
    }
}

状态的切换

在状态图里面,常需要的就是切换状态的行为,改变人物的状态。
这个了类是用来管理状态机的,就是在实际应用时候去调用的了。

/// 
/// 状态图实例,用于实现状态切换等功能
/// 
public class StateGraphInstance
{
    /// 
    /// 状态图
    /// 
    public CharStateGraph sg;
    /// 
    /// 当前状态
    /// 
    public CharSateBase currentState;
    /// 
    /// 前一个状态
    /// 
    public CharSateBase preState;
    /// 
    /// 角色实体
    /// 
    public BaseController inst;
    public Delegate OnStop;
    public Delegate OnStart;

    public StateGraphInstance(CharStateGraph sg, BaseController inst)
    {
        this.sg = sg;
        this.inst = inst;
    }

    ~StateGraphInstance()
    {
        sg = null;
        currentState = null;
        preState = null;
        OnStop = null;
        OnStart = null;
        inst = null;
    }
    public void Start()
    {
        if (OnStart != null)
            OnStart.DynamicInvoke();
    }
    public void Update()
    {
        if (currentState != null)
        {
            currentState.Update(inst, Time.time);
        }
    }
    public void Stop()
    {
        sg.StopTimers();
        sg = null;
        if (OnStop != null)
            OnStop.DynamicInvoke();

        currentState = null;
        preState = null;
        inst = null;
        OnStop = null;
        OnStart = null;
    }
    /// 
    /// 进入状态
    /// 
    /// 
    /// 
    public void GoToState(string statename, params object[] objs)
    {
        CharSateBase state;
        if (sg.states.TryGetValue(statename, out state))
        {
            preState = currentState;
            if (preState != null)
                preState.ClearAddTags();
            if (currentState != null && currentState.onexit != null)
            {
                currentState.Exit(inst, objs);
            }
            sg.StopTimers();
            currentState = state;
            currentState.Enter(inst, objs);
        }
    }

角色的状态

另外还定义了角色的状态,角色状态的名字,角色的状态的进入和退出。
其实这里的 CharStateGraph就是状态机,里面保存状态图的简单信息,名字,停止时间等。
CharStateBase就是普通的状态。
在状态图里面,我们使用来管理状态图的。相当于这个就是一个状态机,状态机和状态的关系是一对多的关系。

在CharStateBase里面有三个常用的状态的改变,状态进入Enter,更新Update,退出Exit。
Unity中状态机的使用_第1张图片

/// 
/// 角色状态
/// 
public class CharSateBase
{
    /// 
    /// 状态名字
    /// 
    public string statename;
    /// 
    /// 状态开始的时间
    /// 
    private float starttime;
    /// 
    /// 状态的标签
    ///  
    // busy   表示上层是否可以操控  // atomic 表示状态之间是否不被打断 // canrotate // norotate // canmove // nomove
    public HashSet<string> tags;
    /// 
    /// 状态额外加的标签
    /// 
    public HashSet<string> addTags;
    /// 
    /// 状态进入是的委托
    /// 
    public Delegate onenter;
    /// 
    /// 状态退出的委托
    /// 
    public Delegate onexit;
    /// 
    /// 状态更新的事件
    /// 
    public Actionfloat> onupdate;
    /// 
    /// 状态中监听的事件
    /// 

    /// 
    /// 进入状态
    /// 
    public void Enter(BaseController bc, params object[] objs)
    {
        starttime = Time.time;
        if (onenter != null)
        {
            onenter.DynamicInvoke(bc, objs);
        }
    }

    public void Update(BaseController bc,float curtime)
    {
        if (onupdate != null)
        {
            onupdate.DynamicInvoke(bc, curtime - starttime);
        }
    }

    /// 
    /// 退出状态
    /// 
    public void Exit(BaseController bc,params object[] objs)
    {
        if (onexit != null)
        {
            onexit.DynamicInvoke(bc,objs);
        }
    }

其实上面的是状态机的名字的改变,实质上还是普通的状态机,下面看下常见的状态机模式。

常见的状态机模式

Istate

Istate是状态机的抽象基类。以后的很多的状态就可以去继承这个基类的。

public abstract class IState
{
    public T Owner { get; set; }
    public F Fsm { get; set; }
    public IStateParam baseparam { get; private set; }
    public void SetParam(IStateParam _param)
    {
        baseparam = _param;
    }
    public void SetFSMID(F id)
    {
        Fsm = id;
    }
    public abstract bool CheckTransition();
    public abstract void Enter();
    public abstract void Execute();
    public abstract void Exit();
}

我们现在使用的是来管理摄像机的一些状态,下面只去列举一种摄像机的状态,实际上有很多种的。如若不是,实现状态机的意义也就是没有的了。

// 相机展示时装直接跟随
public class CamFashionShowFollowState : IState
{
    public CamFashionShowFollowParam mDirparam;

    public override bool CheckTransition()
    {
        return true;
    }

    public override void Enter()
    {
#if UNITY_EDITOR
        Debug.Log("CamFashionShowFollowState  " + Time.time);
#endif
        mDirparam = baseparam as CamFashionShowFollowParam;
        FreeLookCam.Instance.CanZoom = false;
        FreeLookCam.Instance.CanRotate = true;
    }
    public override void Execute()
    {
        if (FreeLookCam.Instance != null)
            FreeLookCam.Instance.DirFollowTarget();
    }
    public override void Exit()
    {
    }
}

状态机IstateMachine类

public class IStateMachine
{
    private T Owner { get; set; }
    private Dictionary> mStates;
    private IState mCurrState;
    private IState mPrevState;
    private IState mGlobalState;
    public IStateMachine(T owner)
    {
        this.Owner = owner;
        mStates = new Dictionary>();
    }
    public bool Contains(F fsmID)
    {
        return this.mStates.ContainsKey(fsmID);
    }
    public void AddState(F fsmID, IState state)
    {
        if (Contains(fsmID))
        {
            return;
        }
        state.SetFSMID(fsmID);
        state.Owner = Owner;
        mStates.Add(fsmID, state);
    }
    public IState GetState(F fsmID)
    {
        if (Contains(fsmID))
        {
            return mStates[fsmID];
        }
        return null;
    }
    public void ChangeState(IState newState)
    {
        mPrevState = mCurrState;
        mCurrState.Exit();
        mCurrState = newState;
        mCurrState.Enter();
    }
    public void ChangeState(F newFSM)
    {
        IState newState = GetState(newFSM);
        if (newState != null)
        {
            ChangeState(newState);
        }
    }
    public void SetCurrState(IState fsm)
    {
        mCurrState = fsm;
    }

    public IState GetCurState()
    {
        return mCurrState;
    }
    public F GetCurrStateID()
    {
        if (mCurrState == null)
        {
            return default(F);
        }
        return mCurrState.Fsm;
    }
    public void Start()
    {
        if (mCurrState != null)
        {
            mCurrState.Enter();
        }
        if (mGlobalState != null)
        {
            mGlobalState.Enter();
        }
    }
}

这里就是差不多就要完成的了,上文还有一个实例化状态机的类StateGraphInstance,这是用来管理状态机的,在这里就不如列举了。

你可能感兴趣的:(Untiy,C#)