委托(delegate)

news2025/1/18 8:49:43

委托(delegate)

委托概述

将方法调用者和目标方法动态关联起来,委托是一个类,所以它和类是同级的,可以通过委托来掉用方法,不要误以为委托和方法同级的,方法只是类的成员。委托定义了方法的类型(定义委托和与之对应的方法必须具有相同的参数个数,并且类型相同,返回值类型相同),使得可以将方法当作另一个方法的参数来进行传递,这种将方法动态地赋给参数的做法,可以避免在程序中大量使用If-Else(Switch)语句,同时使得程序具有更好的可扩展性。

基础委托(Delegate)

在.Net中声明委托使用关键词delegate,委托具有多种使用方式(以下均为同步委托调用):

在这里插入代码片

同步委托&异步委托

同步委托

委托的Invoke方法用来进行同步调用。同步调用也可以叫阻塞调用,它将阻塞当前线程,然后执行调用,调用完毕后再继续向下进行。

异步委托

异步调用不阻塞线程,而是把调用塞到线程池中,程序主线程或UI线程可以继续执行。委托的异步调用通过BeginInvoke和EndInvoke来实现。

以下为异步委托调用方式:

在这里插入代码片

异步回调(Callback)

异步回调通过设置回调函数,当调用结束时会自动调用回调函数,可以在回调函数里触发EndInvoke,这样就释放掉了线程,可以避免程序一直占用一个线程。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;

namespace 委托概述
{
    class Program
    {
        public delegate void DelegateNoReturnWithParameters(string o);
        public delegate void DelegateReturnWithParameters(string o);
        static void Main(string[] args)
        {
            Console.WriteLine($"主线程执行{Thread.CurrentThread.ManagedThreadId}");

            /*
           BeginInvoke方法参数个数不确定, 最后两个参数含义固定,如果不使用的话,需要赋值null
           委托的方法无参数,这种情况下BeginInvoke中只有两个参数。
           此外,委托的方法有几个参数,BeginInvoke中从左开始,对应响应的参数。
          1.倒数第二个参数:是有一个参数值无返回值的委托,它代表的含义为,该线程执行完毕后的回调。
           2.倒数第一个参数:向即回调中传值,用AsyncState来接受。
         3.其它参数:对应委托方法的参数。
          */
            DelegateReturnWithParameters methord = new DelegateReturnWithParameters(Test);
            IAsyncResult asyncResult = methord.BeginInvoke("demo-ok",
                new AsyncCallback(Callback),
                "AsycState:给回调函数的参数传递在此处出传值");
            Console.WriteLine($"主线程执行{Thread.CurrentThread.ManagedThreadId}-AsyncState--{asyncResult.AsyncState}");
            Console.WriteLine($"主线程执行{Thread.CurrentThread.ManagedThreadId}-IsCompleted:--{asyncResult.IsCompleted}");
 
            Console.ReadKey();
        }


       
        private static void Callback(IAsyncResult asyncResult)
        {
            /*
           *asyncResult为回调前异步调用方法返回值
          *AsyncResult 是IAsyncResult接口的一个实现类,引用空间:System.Runtime.Remoting.Messaging
          *AsyncDelegate 属性可以强制转换为定义的委托类型
           */
            DelegateReturnWithParameters methord = (DelegateReturnWithParameters)((System.Runtime.Remoting.Messaging.AsyncResult)asyncResult).AsyncDelegate;
            methord.EndInvoke(asyncResult);

            Console.WriteLine($"主线程执行{Thread.CurrentThread.ManagedThreadId}-AsyncState--{asyncResult.AsyncState}");
            Console.WriteLine($"主线程执行{Thread.CurrentThread.ManagedThreadId}-IsCompleted:--{asyncResult.IsCompleted}");

        }
        /// <summary>
        /// 定义有参无返回值测试方法
        /// </summary>
        /// <param name="o"></param>
        private static void Test(string o)
        {
            Console.WriteLine("有参无返回值:{0}", o);
        }
    }
}

注意:

1.异步调用只能调用一次EndInvoke,否则会报错。

2.如果不回调函数中执行EndInvoke,请在异步调用后手动执行EndInvoke方法释放资源。

异步委托线程等待

1.【Delegate】.EndInvoke(推荐)

1   public delegate void NoReturnWithParameters(string o);
2   NoReturnWithParameters noReturnWithParameters = new NoReturnWithParameters(...);
3        ......
4   noReturnWithParameters.EndInvoke(asyncResult);

2.【IAsyncResult】.AsyncWaitHandle.WaitOne(可以定义等待时间,超过等待时间不继续等待向下执行)

1  IAsyncResult asyncResult = null;
2  asyncResult.AsyncWaitHandle.WaitOne(2000);//等待2000毫秒,超时不等待
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;

namespace 委托概述
{
    class Program
    {
        public delegate void DelegateNoReturnWithParameters(string o);
        public delegate void DelegateReturnWithParameters(string o);
        static void Main(string[] args)
        {
            Console.WriteLine($"主线程执行{Thread.CurrentThread.ManagedThreadId}");

            /*
           BeginInvoke方法参数个数不确定, 最后两个参数含义固定,如果不使用的话,需要赋值null
           委托的方法无参数,这种情况下BeginInvoke中只有两个参数。
           此外,委托的方法有几个参数,BeginInvoke中从左开始,对应响应的参数。
          1.倒数第二个参数:是有一个参数值无返回值的委托,它代表的含义为,该线程执行完毕后的回调。
           2.倒数第一个参数:向即回调中传值,用AsyncState来接受。
         3.其它参数:对应委托方法的参数。
          */
            DelegateReturnWithParameters methord = new DelegateReturnWithParameters(Test);
            IAsyncResult asyncResult = methord.BeginInvoke("demo-ok",
                new AsyncCallback(Callback),
                "AsycState:给回调函数的参数传递在此处出传值");
            ;
            while (!asyncResult.IsCompleted&& asyncResult.AsyncWaitHandle.WaitOne(2000))
            {
                Console.WriteLine($"主线程执行等待中。。。{Thread.CurrentThread.ManagedThreadId}-AsyncState--{asyncResult.IsCompleted}");
            }

            Console.WriteLine($"主线程执行{Thread.CurrentThread.ManagedThreadId}-IsCompleted:--{asyncResult.IsCompleted}");
            Console.WriteLine($"主线程执行{Thread.CurrentThread.ManagedThreadId}-AsyncState--{asyncResult.AsyncState}");
         
 
            Console.ReadKey();
        }


       
        private static void Callback(IAsyncResult asyncResult)
        {
            /*
           *asyncResult为回调前异步调用方法返回值
          *AsyncResult 是IAsyncResult接口的一个实现类,引用空间:System.Runtime.Remoting.Messaging
          *AsyncDelegate 属性可以强制转换为定义的委托类型
           */
            DelegateReturnWithParameters methord = (DelegateReturnWithParameters)((System.Runtime.Remoting.Messaging.AsyncResult)asyncResult).AsyncDelegate;
            methord.EndInvoke(asyncResult);

            Console.WriteLine($"执行委托的方法的线程状态为:{Thread.CurrentThread.ManagedThreadId}-AsyncState--{asyncResult.AsyncState}");
            Console.WriteLine($"执行委托的方法的线程完成状态为:{Thread.CurrentThread.ManagedThreadId}-IsCompleted:--{asyncResult.IsCompleted}");

        }
        /// <summary>
        /// 定义有参无返回值测试方法
        /// </summary>
        /// <param name="o"></param>
        private static void Test(string o)
        {
            Thread.Sleep(2000);

            Console.WriteLine($"执行委托的方法的线程为:{Thread.CurrentThread.ManagedThreadId}");
        }
    }
}

在这里插入图片描述

3.【IAsyncResult】.IsCompleted(是IAsyncResult对象的一个属性,该值指示异步操作是否已完成。不推荐)

1  IAsyncResult asyncResult = xxx.BeginInvoke(...);
2  while (!asyncResult.IsCompleted)
3  {
4      //正在等待中
5  }

内置委托(泛化委托)

.Net Framework 提供两个支持泛型的内置委托,分别是Action<>和Func<>,在System命名空间中定义,结合lambda表达式,可以提高开发效率。
Net Framework 提供两个支持泛型的内置委托,分别是Action<>和Func<>,在System命名空间中定义,结合lambda表达式,可以提高开发效率。

在这里插入图片描述
在这里插入图片描述

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace 委托概述
{
    class 内置委托泛化委托
    {
        static void Main(string[] args)
        {
            //使用Action声明委托
            Action<string> action = TestAction;
            action.Invoke("action-demo-ok");
            //使用Func声明委托
            Func<string, string> func = TestFunc;
            string result = func.Invoke("func-demo-ok");
            Console.WriteLine(result);
            Console.ReadKey();
        }
        private static void TestAction(string o)
        {
            Console.WriteLine("TestAction方法执行成功:{0}", o);
        }
        private static string TestFunc(string o)
        {
            return "TestFunc方法执行成功:" + o;
        }
}
}

Action:无返回值的泛型委托,目前.NET Framework提供了17个Action委托,它们从无参数到最多16个参数。public delegate void Action

-----------
Action -无返回值的泛型委托
Action<int,string> -传入参数int、string,无返回值的委托
Action<int,string,bool> -传入参数int,string,bool,无返回值的委托
Action<bool,bool,bool,bool> -传入4个bool型参数,无返回值的委托
Action最少0个参数,最多16个参数,无返回值。

Func:有返回值的泛型委托,.NET Framework提供了17个Func函数,允许回调方法返回值。

-----------
public delegate TResult Func
Func --无参,返回值为int的委托
Func<int,string>-- 传入参数int,返回值为string类型的委托
Func<object,string,bool> --传入参数为object, string 返回值为bool类型的委托
Func<T1,T2,T3,int> --表示 传入参数为T1,T2,T3(类型)返回值为int类型的委托
Func最少0个参数,最多16个参数,根据返回值泛型返回。必须有返回值,不可为void。

在这里插入图片描述
除此之外还有Predicate,它是固定返回值为bool类型的泛型委托。Action和Func足够使用这里不做介绍。

注意:

1.委托定义不要太多,微软仅在MSCorLib.dll中就有近50个委托类型,而且.NET Framework现在支持泛型,所以我们只需几个泛型委托(在System命名空间中定义)就能表示需要获取多达16个参数的方法。

2.如需获取16个以上参数,就必须定义自己的委托类型。所以建议尽量使用内置委托,而不是在代码中定义更多的委托类型,这样可以减少代码中的类型数量,同时简化编码。

3.如需使用ref或out关键字以传引用的方式传递参数,就需要定义自己的委托。

内置委托(泛化委托)参数协变&逆变

协变(out):假定S是B的子类,如果X(S)允许引用转换成X(B),那么称X为协变类。(支持“子类”向“父类”转换)
逆变(in):假定S是B的子类,如果X(B)允许引用转换成X(X),那么称X为协变类。(支持“父类”向“子类”转换)

正如泛化接口,泛型委托同样支持协变与逆变

委托揭秘

委托看似很容易使用,通过delegate关键词定义,用熟悉的new构造委托实例,熟悉的方式调用回调函数,但实际上编译器和CLR在幕后做了大量工作来隐藏其复杂性。
重新审视上面计算器的一段代码:

1     public delegate int MulticastInstance(int inputA, int inputB);

事实上通过反编译可看到:
在这里插入图片描述
编译器相当于定义了一个完整的类(继承自System.MulticastDelegate,定义四个方法:构造函数、Invoke、BeginInvoke和EndInvoke):

所有委托类型都派生自System.MulticastDelegate类,System.MulticastDelegate派生自System.Delegate,后者又派生自System.Object。历史原因造成有两个委托类。
创建的所有委托类型都将MulticastDelegate作为基类,个别情况下仍会用到Delegate。Delegate类的两个静态方法Combine和Remove的签名都指出要获取Delegate参数。由于创建的委托类型派生自MulticastDelegate,后者又派生自Delegate,所以委托类型的实例是可以传递给这两个方法的。

MulticastDelegate的三个重要非公共字段

结语

  • 同步委托将阻塞当前线程,等待方法执行完毕继续执行程序,相当于直接调用方法。
  • 异步委托是将方法放入线程池中执行并不阻塞主线程。

异步委托从根本上说并不是多线程技术(任务Task也一样),就算异步委托内部将方法塞给线程池去执行也并不能说是开辟新线程执行方法,(线程池一定开辟新线程)这种说法并不严谨。

  • 委托本质是将调用者和目标方法动态关联起来,这是或许是我所理解的委托存在的最根本目的吧。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/78518.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Flutter - DecoratedBox(装饰容器)及内部控件使用详解

DecoratedBox. 可以在其子组件绘制前(或后)绘制一些装饰&#xff08;Decoration&#xff09;&#xff0c;如背景、边框、渐变等 比如 这样 这样 这样 // DecoratedBox 的源码 const DecoratedBox({super.key,required this.decoration, // Decorationthis.position Decor…

产品新说 | 重磅,刚刚正式发布

在当代术加持和需求迭代的背景下&#xff0c;运维人员更需要透过现象看本质&#xff0c;也就是通过复杂的表象数据去挖掘其背后的信息价值。而运维数据经过各种技术手段的治理后&#xff0c;通常的呈现方式是数据空间的形式&#xff0c;比如表格、纯文字等。为了更形象地表达数…

层次遍历二叉树

层次遍历二叉树 文章目录层次遍历二叉树♥ 做法♥算法构思♥ 数据结构设计♥ 层次遍历过程♥ 算法实现应用:用层次遍历求路径之逆♥ 问题♥ 解题思路:♥ 算法框架&#xff1a;♥ 算法实现♥ 做法 ▪ 逐层进行访问 ▪ 对某一层的节点访问完后,再按照其访问次序对各个节点的左、…

[附源码]Python计算机毕业设计SSM基于的在线怀旧电影歌曲听歌系统(程序+LW)

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

MATLAB算法实战应用案例精讲-【神经网络】扩散模型(Diffusion Models)

前言 广播模型、扩散模型和传染模型可以用于分析信息、技术、行为、信念和传染病在人群中的传播。这些模型在通信科学、市场营销学和流行病学的研究中发挥着核心作用。 在计算机视觉中,生成模型是一类能够生成合成图像的模型。例如,一个被训练来生成人脸的模型,每次都会生…

Java项目:SSM电子书网站管理系统

作者主页&#xff1a;源码空间站2022 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文末获取源码 项目介绍 本项目包含管理员、用户两种角色&#xff1b; 管理员角色包含以下功能&#xff1a; 登录页面,管理员管理书籍,用户管理等功能。 用户角色包含…

Docker学习笔记4(狂神)

DockerFile 很多官方镜像都是基础包&#xff0c;很多功能没有&#xff0c;我们通常会自己搭建自己的镜像&#xff01; 官方既然可以制作镜像&#xff0c;那我们也可以&#xff01; DockerFile的构建过程 很多指令&#xff1a; 创建一个自己的centos: 我们可以看到我们从docke…

代码随想录训练营第四天

专题&#xff1a;链表 题目&#xff1a;两两交换链表中的节点 给定一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后的链表。 你不能只是单纯的改变节点内部的值&#xff0c;而是需要实际的进行节点交换。 例如&#xff1a; 题目分析 要两两交换链表中的结…

python opencv 找到圆点标定板所有点后通过距离找两个角点6

先对大图中的标定板框选&#xff0c;然后再标定计算 工程目录结构 如果提示没有win32gui则 pip install pywin32 如果是conda的环境则 conda install pywin32 drawROI.py import copy import jsonimport cv2 import numpy as np import os import matplotlib.pyplot as plt f…

【计算机视觉】Keras API和Tensorflow API的讲解(超详细必看)

觉得有帮助麻烦点赞关注收藏~~~ 一、Keras API Keras是一个用Python编写的高级神经网络API&#xff0c;它能够以Tensorflow、CNTK或者Theano作为后端运行&#xff0c;是一个模块化&#xff0c;最小化并且非常容易扩展的架构&#xff0c;它的开发者Francois Chollet说&#xff…

ESP32 ESP-IDF LVGL8.3.3 ST7735颜色修正

陈拓 2022/12/07-2022/12/10 1. 概述 在《ESP32 ESP-IDF LVGL8.3.3移植(ST7735)》 ESP32 ESP-IDF LVGL8.3.3移植_晨之清风的博客-CSDN博客ESP32 ESP-IDF LVGL8.3.3移植。https://blog.csdn.net/chentuo2000/article/details/128269394?spm1001.2014.3001.5502​​​​​​​…

Python 工匠 第四章 条件分支控制流

基础知识 分支惯用写法 没必要显式和布尔值比较&#xff0c;直接&#xff1a; if user.is_active:pass省略零值判断 if containers_count 0: --> if not containers_count: # 因为bool(0): False但是两者仍有不同 前者只有为0的时候才满足条件 后者则扩大到0, None, 空…

说说Vue-Router和Vue组件中的name属性的使用区别

目录 ⏬ Vue路由匹配规则routes中的name属性的使用 1. 指定页面路由&#xff0c;并传递参数 2. 获取组件的name值&#xff0c;以供页面使用 3. 同个路由&#xff0c;渲染多个视图 ⏬ vue组件中name的使用 1、组件递归操作 2、配合keep-alive对组件缓存做限制 3、在dev-to…

SpringBoot---错误处理机制

PostManHttp请求模拟工具&#xff0c;软件下载链接如下 PostMan下载链接 如果是其他客户端&#xff0c;默认响应一个JSON数据 原理-----SpirngMVC错误处理的自动配置 可以参照ErrorMvcAutoConfiguration&#xff1b;错误处理的自动配置&#xff1b; 给容器中添加了以下组件: …

基于51单片机的多功能电子时钟设计

设计任务&#xff1a; 1、设计任务&#xff1a;利用单片机、时钟芯片 DS1302、温度传感器 DS18B20、1602 液晶 等实现日期、时间、温度的显示即一个简单的万年历。 2、设计要求 &#xff08;1&#xff09;通过 DS1302 能够准确的计时&#xff0c;时间可调并在液晶上显示出来…

RK3568平台开发系列讲解(驱动基础篇)Linux内核面向对象思想之封装

🚀返回专栏总目录 文章目录 一、链表的抽象与封装二、设备管理模型的抽象与封装三、总线设备模型的抽象与封装沉淀、分享、成长,让自己和他人都能有所收获!😄 📢Linux内核虽然是使用C语言实现的,但是内核中的很多子系统、模块在实现过程中处处体现了面向对象编程思想。…

动态规划:将题目转换为01背包问题

文章目录494. 目标和474. 一和零494. 目标和 力扣传送门&#xff1a; https://leetcode.cn/problems/target-sum/ 题目描述&#xff1a; 给你一个整数数组 nums 和一个整数 target 。 向数组中的每个整数前添加 ‘’ 或 ‘-’ &#xff0c;然后串联起所有整数&#xff0c;可以…

easyrecovery2023最新免费版电脑数据恢复软件使用教程

easyrecovery2023版能实现多种不同格式的完成修复和进程的解决&#xff0c;能进行数据的操作和保存解决完成&#xff0c;通过不同的内容进行操作&#xff0c;能解决大部分的使用问题&#xff0c;能安全的进行保存。easyrecovery免安装版对于多种格式下的内容&#xff0c;能对多…

nacos注册中心和配置中心

nacos注册中心和配置中心 nacos 一个更易于构建云原生应用的动态服务发现、配置管理和服务管理平台。 nacos官方文档&#xff1a;https://nacos.io/zh-cn/ 相关概念&#xff1a;https://nacos.io/zh-cn/docs/architecture.html nacos是AP架构,注重可用性和分区容错性&#…

腾讯云双十二服务器2核2G、2核4G、4核8G、8核16G、16核32G配置价格表出炉

现在腾讯云服务器2核2G、2核4G、4核8G、8核16G、16核32G配置价格表已经出来了&#xff0c;大家可以参考一下。腾讯云轻量应用服务器为轻量级的云服务器&#xff0c;使用门槛低&#xff0c;按套餐形式购买&#xff0c;轻量应用服务器套餐自带的公网带宽较大&#xff0c;4M、6M、…