我试图覆盖C#中的等号(==)运算符来处理将任何类型与自定义类型进行比较(自定义类型实际上是一个包装/框在零).
所以我有这个:
internal sealed class nothing
{
public override bool Equals(object obj)
{
if (obj == null || obj is nothing)
return true;
else
return false;
}
public static bool operator ==(object x,nothing y)
{
if ((x == null || x is nothing) && (y == null || y is nothing))
return true;
return false;
}
...
}
现在如果我打个电话:
nothing n = new nothing(); bool equal = (10 == n);
它工作得很好但是,如果我尝试通过一个Linq表达式来做同样的事情:
exp = Expression.Equal(
Expression.Constant(10),Expression.Constant(new nothing(),typeof(nothing))
);
它抛出异常:
System.ArgumentException : Expression of type 'system.int32' cannot be used for parameter of type 'System.Object' of method 'Boolean op_Equality(System.Object,PARTSFinder.Rules.Runtime.Rulesnothing)'
at System.Linq.Expressions.Expression.ValidateArgumentTypes(MethodInfo method,ReadOnlyCollection`1& arguments)
at System.Linq.Expressions.Expression.ValidateCallArgs(Expression instance,MethodInfo method,ReadOnlyCollection`1& arguments)
at System.Linq.Expressions.Expression.Call(Expression instance,IEnumerable`1 arguments)
at System.Linq.Expressions.Expression.Call(Expression instance,Expression[] arguments)
at System.Linq.Expressions.ExpressionCompiler.GenerateBinaryMethod(ILGenerator gen,BinaryExpression b,StackType ask)
任何想法为什么基础系统可以将Int32转换为Object,但Linq不能,或者我如何解决这个问题?
这整个事情盯着,因为Linq也不能比较Int32到Object的第一位:
exp = Expression.Equal(
Expression.Constant(10),Expression.Constant(null)
);
抛出一个异常,指出“system.int32”和“System.Object”没有比较运算符.
快速跟进:
以下工作没有问题:
exp = Expression.Equal(
Expression.Constant(10,typeof(object)),typeof(nothing))
);
exp = Expression.Equal(
Expression.Constant(10,Expression.Constant(null)
);
所以特意把所有的东西投射到对象. Linq也不在内部处理继承?这很讨厌…
后续#2:
我也试过使用自定义的比较方法:
exp = Expression.Equal(
Expression.Constant(10),Expression.Constant(null),false,this.GetType().getmethod("ValueEquals",BindingFlags.Public | BindingFlags.Static)
);
public static bool ValueEquals(object x,object y)
{
if (x == null && y == null)
return true;
if (x.GetType() != y.GetType())
return false;
return x == y;
}
这也引发了一个例外:
system.invalidOperationException : The operands for operator 'Equal' do not match the parameters of method 'ValueEquals'.
at System.Linq.Expressions.Expression.getmethodBasedBinaryOperator(ExpressionType binaryType,Expression left,Expression right,Boolean liftToNull)
但是再次直接将所有内容投射到对象作品中:
exp = Expression.Equal(
Expression.Constant(10,Expression.Constant(null,BindingFlags.Public | BindingFlags.Static)
);
所以我想我有我的解决方法…投掷所有东西来对象,并使用一个自定义的比较方法.我还是很惊讶,Linq不像C#一样正常地进行转换.
解决方法
null有什么问题?重新删除int vs null,try int?:
exp = Expression.Equal(
Expression.Constant(10,typeof(int?)),typeof(int?))
);