如何使用反射调用自定义运算符

32

在我的小项目中,我正在使用 System.Reflection 类来生成可执行代码。我需要调用自定义类型的 + 运算符。有人知道如何使用 C# 反射调用自定义类的自定义运算符吗?

3个回答

48

C#编译器将重载的运算符转换为函数,函数名称为op_XXXX,其中XXXX表示运算操作。例如,operator +编译为op_Addition

以下是可重载运算符及其相应方法名称的完整列表:

┌──────────────────────────┬───────────────────────┬──────────────────────────┐
│         Operator         │      Method Name      │       Description        │
├──────────────────────────┼───────────────────────┼──────────────────────────┤
│ operator +               │ op_UnaryPlus          │ Unary                    │
│ operator -               │ op_UnaryNegation      │ Unary                    │
│ operator ++              │ op_Increment          │ Unary                    │
│ operator --              │ op_Decrement          │ Unary                    │
│ operator !               │ op_LogicalNot         │ Unary                    │
│ operator +               │ op_Addition           │                          │
│ operator -               │ op_Subtraction        │                          │
│ operator *               │ op_Multiply           │                          │
│ operator /               │ op_Division           │                          │
│ operator &               │ op_BitwiseAnd         │                          │
│ operator |               │ op_BitwiseOr          │                          │
│ operator ^               │ op_ExclusiveOr        │                          │
│ operator ~               │ op_OnesComplement     │ Unary                    │
│ operator ==              │ op_Equality           │                          │
│ operator !=              │ op_Inequality         │                          │
│ operator <               │ op_LessThan           │                          │
│ operator >               │ op_GreaterThan        │                          │
│ operator <=              │ op_LessThanOrEqual    │                          │
│ operator >=              │ op_GreaterThanOrEqual │                          │
│ operator <<              │ op_LeftShift          │                          │
│ operator >>              │ op_RightShift         │                          │
│ operator %               │ op_Modulus            │                          │
│ implicit operator <type> │ op_Implicit           │ Implicit type conversion │
│ explicit operator <type> │ op_Explicit           │ Explicit type conversion │
│ operator true            │ op_True               │                          │
│ operator false           │ op_False              │                          │
└──────────────────────────┴───────────────────────┴──────────────────────────┘

要检索DateTime 结构的operator+ 方法,你需要编写:

MethodInfo mi = typeof(DateTime).GetMethod("op_Addition",
    BindingFlags.Static | BindingFlags.Public );

2
只是好奇 :) 如果我有一个相同签名的静态 op_Addition 方法会怎么样? - Şafak Gür
1
@ŞafakGür 然后你会得到一个编译器错误,说“类型 '<YourTypeName>' 已经定义了一个与参数类型相同的成员 'op_Addition'”。因为定义的运算符方法恰好被重命名为上述名称,所以你不能在同一个类中同时存在两者。 - Mohammad Dehghan
1
我已经自己找到了答案,但是为了以后参考:类型转换运算符被命名为op_Explicitop_Implicit(我认为这些名称很容易理解)。但请记住,可以定义多个类型转换运算符,因此需要通过指定参数类型或返回类型(关于转换的“方向”)来缩小搜索范围。 - Grx70
1
你忘记了 ~ 吗? - Denis535
1
@ChristabellaIrwanto 我记得我最初反编译了一段C#代码,然后在C#编译器中查找了实际的代码。微软没有在任何地方记录这一过程。 - Mohammad Dehghan
显示剩余4条评论

8
typeof(A).GetMethod("op_Addition").Invoke(null, instance1, instance2);

通过我,参数需要通过数组给出: type.GetMethod("op_Subtraction").Invoke(null, new object[] { instance1, instance2 }); - Joseph Merdrignac

0
考虑将您定制的运算符作为您的类的“属性”进行设置。然后通过“反射”访问该“属性”及其“值”。
就像这样:
PropertyInfo pinfo = obj.GetType().GetProperty("CustomOperator", BindingFlags.Public | BindingFlags.Instance);
string customOperator = pinfo.GetValue(obj,null) as string;

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接