Unity3D Odin Inspector Attribute回调的实现原理

环境:Unity2021.1.14 Odin3.0.4 语言:C#

面向:Odin进阶开发人员

问题

    在Odin Attribute的使用过程中,比如OnValueChanged,通常会传入一个字符串指定回调函数。

    通过字符串指定回调函数的方式是否能通过Odin的API主动调用?带着这样的疑问,我开始了探索。

情况

    使用OnValueChanged的Attribute可以指定回调函数,在age值改变的时候会触发OnAgeChanged:

public class Test1 : MonoBehaviour
{
	[OnValueChanged("OnAgeChanged")]
	public int age = 18;

	public void OnAgeChanged()
	{
		Debug.LogError("我的年龄:" + age);
	}
}

    我们可以看到回调函数指定是通过字符串形式传递的。

原理

    OnValueChangedAttributeDrawer中使用字符串进行创建。

    创建ActionResolver,ActionResolver.DoAction调用回调函数:

this.onChangeAction = ActionResolver.Get(this.Property, this.Attribute.Action);
if (this.Attribute.InvokeOnInitialize && !this.onChangeAction.HasError)
{
    this.onChangeAction.DoActionForAllSelectionIndices();
}

    ActionResolverCreator进行创建,在InitResolver时,通过内部保存的Resolvers数组逐个尝试创建ResolvedAction:

var array = ActionResolverCreators;
ResolvedAction action = null;
for (int i = 0; i < array.Length; i++)
{
    var resolverCreator = array[i].ResolverCreator;
    if (resolverCreator == null) break;
    action = resolverCreator.TryCreateAction(ref resolver.Context);
    if (action != null)
        break;
}

    3种Odin默认提供的字符串解析:

MethodPropertyActionResolverCreator

MethodReferenceActionResolverCreator

ExpressionActionResolverCreator

(需要自定义字符串解析可以通过打RegisterDefaultActionResolver标签)

    我们先看ExpressionActionResolverCreator类。

    ExpressionUtility.ParseExpression解析字符串,需要字符串以@开头:

var method = ExpressionUtility.ParseExpression(expression, isStatic, context.ParentType, parameterTypes, parameterNames, out compileError, true);
var parameterValues = new object[parameterCount + (isStatic ? 0 : 1)];
GetExpressionLambda(method, isStatic, context.ParentType.IsValueType, parameterValues);

    返回的函数最外层使用lambda表达式进行包裹,并将创建好的Delegate传入,最终使用Delegate. DynamicInvoke进行调用。

    Emitter.EmitMethod构建出方法,根据ASTParser. Parse()所解析出来的Token:

Tokenizer.SetExpressionString(expression);
Emitter.EmitMethod("$Expression(" + expression + ")_" + Guid.NewGuid().ToString(), Parser.Parse(), context, delegateType);

    MethodReferenceActionResolverCreator,通过Type反射出对应的MethodInfo进行调用:

MethodInfo method = GetCompatibleMethod(contextType, memberName, flags, ref context.NamedValues, ref argSetup, out errorMessage);
GetMethodInvoker(method as MethodInfo, argSetup, context.ParentType.IsValueType);

方法

    最简单的方式是创建ActionResolver直接进行解析,这也是AttributeDrawer调用的方式:

var action = ActionResolver.Get(PropertyTree.Create(this).RootProperty, "OnAgeChanged");
action.DoActionForAllSelectionIndices();

    这里底层使用的是Type进行的反射,我们可以传递@this.OnAgeChanged()主动调用表达式的方式:

var action = ActionResolver.Get(PropertyTree.Create(this).RootProperty, "@this.OnAgeChanged()");
action.DoActionForAllSelectionIndices();

    使用ExpressionUtility直接解析表达式,并调用:

var action = ExpressionUtility.ParseExpression("this.OnAgeChanged()", false, typeof(Test1), null, null, out var compileError, true);
if (!string.IsNullOrEmpty(compileError))
{
	Debug.LogError(compileError);
}
action.DynamicInvoke(this);

    以上的方法解析表达式大概会耗费0.0180秒的时间,反射方式耗费0.0010秒的时间,可以看到解析表达式是比较耗时的,所以在真实开发的时候推荐初始化时就先解析并缓存起来。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值