1,委托列子

internal delegate void Feedback(int value);
class DelegateRef
{
public static void StaticDelegateDeomo()
{
Console.WriteLine("---------------static delegate Demo ------------");
Counter(1, 3, null);
Counter(1, 3, FeedBackToConsole);
Counter(1, 3, FeedBackToMsgBox);
Console.WriteLine();
}
public static void InstanceDelegateDemo()
{
Console.WriteLine("----------------Instance Delegate Demo ---------");
DelegateRef p = new DelegateRef();
Counter(1, 3, p.FeedBackToFile);
Console.WriteLine();
}
public static void ChainDelegateDemo1(DelegateRef p)
{
Console.WriteLine("----------------Chain Delegate Demo ----------");
Feedback chain = FeedBackToConsole;
chain += FeedBackToMsgBox;
chain += p.FeedBackToFile;
Counter(1, 2, chain);
Console.WriteLine();
chain -= FeedBackToMsgBox;
Counter(1, 2, chain); }
private static void Counter(int from, int to, Feedback fb)
{
for (int val = from; val <= to; val++)
{
if (fb != null)
fb(val);
}
} private static void FeedBackToConsole(int value)
{
Console.WriteLine("Item=" + value);
}
private static void FeedBackToMsgBox(int value)
{
MessageBox.Show("Item=" + value);
}
private void FeedBackToFile(int value)
{
using(StreamWriter sw=new StreamWriter("Status", true))
{
sw.WriteLine("Item=" + value);
}
}
public static void CallDelegateDemo()
{
StaticDelegateDeomo();
InstanceDelegateDemo();
ChainDelegateDemo1(new DelegateRef());
}
}

2,委托揭秘

在 Conter之中,实际上 +=是创建了Delegate的实列,并且调用了Combine函数,触发调用了invoke函数,移除调用了,remove函数

.method public hidebysig static void  ChainDelegateDemo1(class ClrFromCSharp_2_2.LearnDelegate.DelegateRef p) cil managed
// SIG: 00 01 01 12 60
{
// 方法在 RVA 0x2ad4 处开始
// 代码大小 122 (0x7a)
.maxstack 3
.locals init ([0] class ClrFromCSharp_2_2.LearnDelegate.Feedback chain)
IL_0000: /* 00 | */ nop
IL_0001: /* 72 | (70)000207 */ ldstr "----------------Chain Delegate Demo ----------"
IL_0006: /* 28 | (0A)00001C */ call void [mscorlib]System.Console::WriteLine(string)
IL_000b: /* 00 | */ nop
IL_000c: /* 14 | */ ldnull
IL_000d: /* FE06 | (06)000059 */ ldftn void ClrFromCSharp_2_2.LearnDelegate.DelegateRef::FeedBackToConsole(int32)
IL_0013: /* 73 | (06)000051 */ newobj instance void ClrFromCSharp_2_2.LearnDelegate.Feedback::.ctor(object,
native int)
IL_0018: /* 0A | */ stloc.0
IL_0019: /* 06 | */ ldloc.0
IL_001a: /* 14 | */ ldnull
IL_001b: /* FE06 | (06)00005A */ ldftn void ClrFromCSharp_2_2.LearnDelegate.DelegateRef::FeedBackToMsgBox(int32)
IL_0021: /* 73 | (06)000051 */ newobj instance void ClrFromCSharp_2_2.LearnDelegate.Feedback::.ctor(object,
native int)
IL_0026: /* 28 | (0A)00003C */ call class [mscorlib]System.Delegate [mscorlib]System.Delegate::Combine(class [mscorlib]System.Delegate,
class [mscorlib]System.Delegate)
IL_002b: /* 74 | (02)000017 */ castclass ClrFromCSharp_2_2.LearnDelegate.Feedback
IL_0030: /* 0A | */ stloc.0
IL_0031: /* 06 | */ ldloc.0
IL_0032: /* 02 | */ ldarg.0
IL_0033: /* FE06 | (06)00005B */ ldftn instance void ClrFromCSharp_2_2.LearnDelegate.DelegateRef::FeedBackToFile(int32)
IL_0039: /* 73 | (06)000051 */ newobj instance void ClrFromCSharp_2_2.LearnDelegate.Feedback::.ctor(object,
native int)
IL_003e: /* 28 | (0A)00003C */ call class [mscorlib]System.Delegate [mscorlib]System.Delegate::Combine(class [mscorlib]System.Delegate,
class [mscorlib]System.Delegate)
IL_0043: /* 74 | (02)000017 */ castclass ClrFromCSharp_2_2.LearnDelegate.Feedback
IL_0048: /* 0A | */ stloc.0
IL_0049: /* 17 | */ ldc.i4.1
IL_004a: /* 18 | */ ldc.i4.2
IL_004b: /* 06 | */ ldloc.0
IL_004c: /* 28 | (06)000058 */ call void ClrFromCSharp_2_2.LearnDelegate.DelegateRef::Counter(int32,
int32,
class ClrFromCSharp_2_2.LearnDelegate.Feedback)
IL_0051: /* 00 | */ nop
IL_0052: /* 28 | (0A)000049 */ call void [mscorlib]System.Console::WriteLine()
IL_0057: /* 00 | */ nop
IL_0058: /* 06 | */ ldloc.0
IL_0059: /* 14 | */ ldnull
IL_005a: /* FE06 | (06)00005A */ ldftn void ClrFromCSharp_2_2.LearnDelegate.DelegateRef::FeedBackToMsgBox(int32)
IL_0060: /* 73 | (06)000051 */ newobj instance void ClrFromCSharp_2_2.LearnDelegate.Feedback::.ctor(object,
native int)
IL_0065: /* 28 | (0A)00003F */ call class [mscorlib]System.Delegate [mscorlib]System.Delegate::Remove(class [mscorlib]System.Delegate,
class [mscorlib]System.Delegate)
IL_006a: /* 74 | (02)000017 */ castclass ClrFromCSharp_2_2.LearnDelegate.Feedback
IL_006f: /* 0A | */ stloc.0
IL_0070: /* 17 | */ ldc.i4.1
IL_0071: /* 18 | */ ldc.i4.2
IL_0072: /* 06 | */ ldloc.0
IL_0073: /* 28 | (06)000058 */ call void ClrFromCSharp_2_2.LearnDelegate.DelegateRef::Counter(int32,
int32,
class ClrFromCSharp_2_2.LearnDelegate.Feedback)
IL_0078: /* 00 | */ nop
IL_0079: /* 2A | */ ret
} // end of method DelegateRef::ChainDelegateDemo1
}

3,利用GetInvocationList()的方式,显示调用每一个委托,这样可以显式的处理每个委托的返回值和异常,否则

  • 当出现异常的时候,链表里面的下面的委托不会执行
  • 当出现阻塞时候,链表里面的下面的委托不会执行

4,委托和反射

internal static class DelegateReflection {
// Here are some different delegate definitions
private delegate Object TwoInt32s(Int32 n1, Int32 n2);
private delegate Object OneString(String s1); public static void Go(String[] args) {
if (args.Length < 2) {
String fileName = Path.GetFileNameWithoutExtension(Assembly.GetEntryAssembly().Location);//获取执行文件的名称,但是没用 String usage =//提示用法
@"Usage:" +
"{0}{1} delType methodName [Arg1] [Arg2]" +
"{0} where delType must be TwoInt32s or OneString" +
"{0} if delType is TwoInt32s, methodName must be Add or Subtract" +
"{0} if delType is OneString, methodName must be NumChars or Reverse" +
"{0}" +
"{0}Examples:" +
"{0} {1}TwoInt32s Add 123 321" +
"{0} {1}TwoInt32s Subtract 123 321" +
"{0} {1}OneString NumChars \"Hello there\"" +
"{0} {1}OneString Reverse \"Hello there\"";
Console.WriteLine(usage, Environment.NewLine, "DelegateReflection+");
return;
} // Convert the delType argument to a delegate type Type delType = Type.GetType(args[0]);//1,利用string返回类型Type...注意:DelegateReflection+TwoInt32s才是其TYPE
if (delType == null) {
Console.WriteLine("Invalid delType argument: " + args[0]);
return;
} Delegate d;
try {
// Convert the Arg1 argument to a method//注意,在某个类里面获取某个方法..方法名是arg[1],Add
MethodInfo mi = typeof(DelegateReflection).GetMethod(args[1], BindingFlags.NonPublic | BindingFlags.Static);//2,获取类中方法 // Create a delegate object that wraps the static method
d = Delegate.CreateDelegate(delType, mi);//3,利用该方法将类的静态方法--->传送至委托.
}
catch (ArgumentException) {
Console.WriteLine("Invalid methodName argument: " + args[1]);
return;
} // Create an array that that will contain just the arguments
// to pass to the method via the delegate object
Object[] callbackArgs = new Object[args.Length - 2]; if (d.GetType() == typeof(TwoInt32s)) {//类型比较
try {
// Convert the String arguments to Int32 arguments
for (Int32 a = 2; a < args.Length; a++)
callbackArgs[a - 2] = Int32.Parse(args[a]);
}
catch (FormatException) {
Console.WriteLine("Parameters must be integers.");
return;
}
} if (d.GetType() == typeof(OneString)) {
// Just copy the String argument
Array.Copy(args, 2, callbackArgs, 0, callbackArgs.Length);
} try {
// Invoke the delegate and show the result
Object result = d.DynamicInvoke(callbackArgs);//4,调用委托,动态给与参数.参数是object[]---必须匹配.
Console.WriteLine("Result = " + result);
}
catch (TargetParameterCountException) {
Console.WriteLine("Incorrect number of parameters specified.");
}
} // This callback method takes 2 Int32 arguments
private static Object Add(Int32 n1, Int32 n2) {
return n1 + n2;
} // This callback method takes 2 Int32 arguments
private static Object Subtract(Int32 n1, Int32 n2) {
return n1 - n2;
} // This callback method takes 1 String argument
private static Object NumChars(String s1) {
return s1.Length;
} // This callback method takes 1 String argument
private static Object Reverse(String s1) {
Char[] chars = s1.ToCharArray();
Array.Reverse(chars);
return new String(chars);
}
}
  1. 新建委托TYPE
  2. 从类中查找静态方法
  3. 利用Delegate.CreateDelegate建立类中静态方法的委托.
  4. 利用d.DynamicInvoke(object[])来调用委托,其中参数是一个数组匹配原静态方法的参数.如果不一致会报参数不匹配错误.
  5. TargetParameterCountException

clr via c# delegate的更多相关文章

  1. 公共语言运行库(CLR)开发系列课程(2):Pinvoke 进阶 学习笔记

    上一章地址 API版本 具有字符串参数的API通常有两种版本 GetWindowText GetWindowTextA GetWindowTextW 缺省情况下CLR会自动寻找合适的匹配 CharSe ...

  2. LuaInterface简介

    Lua是一种很好的扩展性语言,Lua解释器被设计成一个很容易嵌入到宿主程序的库.LuaInterface则用于实现Lua和CLR的混合编程. (一)Lua from the CLR 测试环境:在VS2 ...

  3. [整理]Unity3D游戏开发之Lua

    原文1:[Unity3D]Unity3D游戏开发之Lua与游戏的不解之缘(上) 各位朋友,大家好,我是秦元培,欢迎大家关注我的博客,我地博客地址是blog.csdn.net/qinyuanpei.如果 ...

  4. LuaInterface简单介绍

    LuaInterface简单介绍 Lua是一种非常好的扩展性语言.Lua解释器被设计成一个非常easy嵌入到宿主程序的库.LuaInterface则用于实现Lua和CLR的混合编程. (一)Lua f ...

  5. ILRuntime 学习

    ILRuntime: https://github.com/Ourpalm/ILRuntime Demo: https://github.com/Ourpalm/ILRuntimeU3D 中文在线文档 ...

  6. CLR via C#(12)-委托Delegate

    本来按照进度应该学习事件了,可总觉得应该委托在前,事件在后,才好理解. 委托是一个类,它提供了回调函数机制,而且是类型安全的.使用委托可以将方法当作另一个方法的参数来进行传递,这种将方法动态地赋给参数 ...

  7. 《CLR.via.C#第三版》第二部分第12章节 泛型 读书笔记(六)

    终于讲到泛型了.当初看到这个书名,最想看的就是作者对泛型,委托,反射这些概念的理解.很多人对泛型的理解停留在泛型集合上,刚开始我也是,随着项目越做越多,对待泛型的认识也越来越深刻. 泛型的概念:泛型是 ...

  8. CLR VIA C# 学习笔记

    第19章 可空类型 1)使用Nullable<T>可将int32的值类型设置为Null,CLR会在Null时默认赋值为0; 如:Nullable<T> x=null; //使用 ...

  9. CLR via C# 3rd - 08 - Methods

       Kinds of methods        Constructors      Type constructors      Overload operators      Type con ...

随机推荐

  1. initramfs打包集成rootfs到image镜像及linux rootfs的正常启动

    最近的项目中需要在仿真机haps及VDK上集成rootfs,中间遇到一些问题,在此整理记录以备忘. rootfs里面集成的busybox版本1.29.3 (buildroot环境中自带) kernel ...

  2. .net core 常见设计模式-IChangeToken

    场景 一个对象A,希望它的某些状态在发生改变时通知到B(或C.D),常见的做法是在A中定义一个事件(或直接用委托),当状态改变时A去触发这个事件.而B直接订阅这个事件 这种设计有点问题B由于要订阅A的 ...

  3. 开发工具篇:Git和Github

    开发工具篇:Git和Github Git是什么? Git是目前世界上最先进的分布式版本控制系统.工作原理 / 流程: Workspace:工作区 Index / Stage:暂存区 Repositor ...

  4. docker-Gitlab、GitLab Runner安装

    以下操作均在CentOs下操作 1.Gitlab install ① 启动gitlab docker run --detach \ --hostname 115.30.149.35 \ --publi ...

  5. Windows版Redis主从配置

    一.下载 从github上下载Redis的zip包,地址:https://github.com/MicrosoftArchive/redis/releases Redis本身不支持windows,这是 ...

  6. Zabbix添加SNMP自定义监控项OID出现“No Such Instance currently exists at this OID”

    原因:zabbix 是用snmpget来获取指定的OID数据,snmpwalk是遍历某个OID下的数据. 所以一定要用snmpget来验证某个OID是否正确. snmptranslate 获取的OID ...

  7. HttpClient介绍和使用

    HttpClient介绍和使用 今天有一个需求:后台访问一个接口,获取返回的数据.于是找到了HttpClient 1.介绍 SpringCloud中服务和服务之间的调用全部是使用HttpClient, ...

  8. composer intall 报错

    报错 [Composer\Exception\NoSslException] The openssl extension is required for SSL/TLS protection but ...

  9. 一份中规中矩的 iOS笔试题

    一.背景 因为一些原因,我从公司离职,在此之前,我需要帮忙招聘一个新人来做我的工作,于是就有了这篇文章. 由于公司项目16年就已经上线了,前同事写的项目中,有用xib.storyBoard.代码自动布 ...

  10. react FileReader读取TXT文件并保存 split切割字符串 map()分别渲染切割后的数组内的所有字符串

    //class my_fileReader( e ) {         console.log(e.target.files[0]);         const reader = new File ...