C#基础之委托,事件

文章目录

  • 1 委托
    • 1.1 简介
    • 1.2 操作使用
      • 1.2.1 声明委托(Delegate)
      • 1.2.2 实例化委托(Delegate)
      • 1.2.3 直接调用和invoke
      • 1.2.4 Invoke 和 BeginInvoke
    • 1.3 委托的多播
    • 1.4 委托的匿名和lambda
      • 1.4.1 匿名方法
      • 1.4.2 lambda 表达式
    • 1.5 内置委托
      • 1.5.1 Action系列
      • 1.5.2 Func 系列
      • 1.5.3 Predicate
    • 1.6 示例
  • 2 事件
    • 2.1 简介
    • 2.2 原理
      • 2.2.1 讲解
      • 2.2.2 add 和 remove 访问器
    • 2.3 使用原生委托
    • 2.4 自定义委托
      • 2.4.1 声明
      • 2.4.2 操作
        • 2.4.2.1 示例一
        • 2.4.2.2 示例二

1 委托

1.1 简介

C# 中的委托(Delegate)类似于 C 或 C++ 中函数的指针。委托(Delegate) 是存有对某个方法的引用的一种引用类型变量。引用可在运行时被改变。
委托(Delegate)特别用于实现事件回调方法。所有的委托(Delegate)都派生自 System.Delegate 类。

1.2 操作使用

1.2.1 声明委托(Delegate)

委托声明决定了可由该委托引用的方法。委托可指向一个与其具有相同标签的方法。
例如,假设有一个委托:public delegate int MyDelegate (string s);
上面的委托可被用于引用任何一个带有一个单一的 string 参数的方法,并返回一个 int 类型变量

声明委托的语法如下:

delegate <return type> <delegate-name> <parameter list>

1.2.2 实例化委托(Delegate)

一旦声明了委托类型,委托对象必须使用 new 关键字来创建,且与一个特定的方法有关。当创建委托时,传递到 new 语句的参数就像方法调用一样书写,但是不带有参数。例如:

public delegate void printString(string s);
...
printString ps1 = new printString(WriteToScreen);
printString ps2 = new printString(WriteToFile);

下面的实例演示了委托的声明、实例化和使用,该委托可用于引用带有一个整型参数的方法,并返回一个整型值。

using System;delegate int NumberChanger(int n);
namespace DelegateAppl
{class TestDelegate{static int num = 10;public static int AddNum(int p){num += p;return num;}public static int MultNum(int q){num *= q;return num;}public static int getNum(){return num;}static void Main(string[] args){// 创建委托实例NumberChanger nc1 = new NumberChanger(AddNum);NumberChanger nc2 = new NumberChanger(MultNum);// 使用委托对象调用方法nc1(25);Console.WriteLine("Value of Num: {0}", getNum());nc2(5);Console.WriteLine("Value of Num: {0}", getNum());Console.ReadKey();}}
}结果:
Value of Num: 35
Value of Num: 175

注意:调用委托对应方法一般是通过invoke方法,但是从 C# 2.0 开始,委托的调用可以直接使用方法调用语法,而不需要显式调用 Invoke 方法。

1.2.3 直接调用和invoke

虽然委托调用底层实际上是通过 Invoke 方法实现的,但语法上允许直接调用委托,就像调用普通方法一样。换句话说,调用委托和直接调用 Invoke 方法是等效的。

假设我们有一个 Action 类型的委托:

Action action = () => Console.WriteLine("Hello, Delegate!");直接调用委托:
action(); // 输出: Hello, Delegate!通过 Invoke 方法调用:
action.Invoke(); // 输出: Hello, Delegate!

两种方式的结果完全一样,因为 () 是对委托对象 Invoke 方法的简化语法糖

为什么允许直接调用?

  • 简洁性:如果每次调用都必须写 .Invoke,代码显得冗长。因此,C# 提供了直接调用语法,增强代码可读性。
  • 语法糖:编译器在编译时会自动将直接调用委托语法转换为 Invoke 方法的调用。
    即:action(); 实际被编译为:action.Invoke();
  • 优先推荐直接调用
    直接调用的方式更加简洁可读,因此在大多数情况下,推荐使用 action() 而不是显式调用 action.Invoke()

为什么保留 Invoke 方法?虽然直接调用语法更方便,但在某些特殊场景下,显式调用 Invoke 方法可能更合适:

  • 反射场景:通过反射调用委托时,需要使用 Invoke 方法。
  • 动态场景:在动态生成代码或动态委托时,Invoke 方法更明确。
using System;
using System.Reflection;class Program
{static void Main(){Action action = PrintMessage;// 使用反射调用 InvokeMethodInfo invokeMethod = action.GetType().GetMethod("Invoke");invokeMethod.Invoke(action, null);}static void PrintMessage(){Console.WriteLine("Hello, Reflection!");}
}
输出:
Hello, Reflection!

1.2.4 Invoke 和 BeginInvoke

委托的 InvokeBeginInvoke 方法分别用于同步异步调用委托。它们的主要区别体现在调用方式、线程管理和返回结果的处理上。

Invoke 和 BeginInvoke 的区别

特性InvokeBeginInvoke
调用类型同步调用异步调用
线程阻塞当前线程会阻塞,直到方法执行完成当前线程不会阻塞
返回结果直接返回方法的返回值返回 IAsyncResult 对象,通过 EndInvoke 获取返回值
异常处理异常会直接在调用线程中抛出异常在调用 EndInvoke 时抛出
线程使用在调用线程上执行方法在线程池中执行方法
使用场景方法较快且调用线程不能被中断时方法较慢且需要异步执行时
  • Invoke:同步调用
    定义:Invoke 是同步调用,当前线程会等待方法执行完毕后再继续执行后续代码。
    特点:
    • 阻塞调用:调用线程会被阻塞,直到被调用的方法完成。
    • 返回结果:直接返回被调用方法的返回值(如果有)。
    • 异常处理:如果被调用的方法抛出异常,异常会在调用线程中传播。
// 定义一个委托
delegate int AddDelegate(int x, int y);AddDelegate add = (x, y) => x + y;// 同步调用
int result = add.Invoke(3, 4);
Console.WriteLine($"Result: {result}"); // 输出:Result: 7
  • BeginInvoke:异步调用
    定义:BeginInvoke 是异步调用,立即返回一个 IAsyncResult 对象,并不会阻塞调用线程。
    特点:
    • 非阻塞调用:调用线程可以继续执行其他代码,而被调用的方法在后台线程中执行。
    • 回调机制:可以通过传递回调方法或轮询 IAsyncResult 对象来获取结果。
      需要显式调用 EndInvoke 方法以获取结果或处理异常。
// 定义一个委托
delegate int AddDelegate(int x, int y);
AddDelegate add = (x, y) =>
{Console.WriteLine("Adding...");System.Threading.Thread.Sleep(2000); // 模拟耗时操作return x + y;
};// 异步调用
IAsyncResult asyncResult = add.BeginInvoke(3, 4, null, null);
// 主线程继续执行其他任务
Console.WriteLine("Doing other work...");
// 获取异步调用结果
int result = add.EndInvoke(asyncResult);
Console.WriteLine($"Result: {result}"); // 输出:Result: 7

BeginInvoke 的回调,可以通过回调函数在异步操作完成后处理结果:

void CallbackMethod(IAsyncResult ar)
{// 获取委托实例AddDelegate add = (AddDelegate)ar.AsyncState;// 获取结果int result = add.EndInvoke(ar);Console.WriteLine($"Result in Callback: {result}");
}AddDelegate add = (x, y) =>
{Console.WriteLine("Adding...");System.Threading.Thread.Sleep(2000);return x + y;
};// 异步调用并指定回调函数
add.BeginInvoke(5, 7, CallbackMethod, add);
// 主线程继续工作
Console.WriteLine("Doing other work...");

注意事项:

  • BeginInvoke 使用线程池中的线程来执行方法,因此需要注意线程池的资源消耗。
    必须调用 EndInvoke:
  • 调用 BeginInvoke 后,无论是否需要结果,都必须调用 EndInvoke,否则可能会导致资源泄漏。
  • 推荐使用 Task 和 async/await:
  • 在现代 C# 中,推荐使用 Task 和 async/await 替代 BeginInvokeEndInvoke,因为它们更易读且不易出错。

1.3 委托的多播

委托对象可使用 + 运算符进行合并。一个合并委托调用它所合并的两个委托。只有相同类型的委托可被合并。- 运算符可用于从合并的委托中移除组件委托。
使用委托的这个有用的特点,可以创建一个委托被调用时要调用的方法的调用列表。这被称为委托的 多播(multicasting),也叫组播
+- 运算符确实可以直接用于委托对象的合并和移除,但这和 +=-= 的用法有所不同。它们的区别主要在于运算场景赋值方式。具体来说

  • +- 运算符:用于直接创建新的委托对象,不影响原始委托。它们不会修改原始委托,而是生成一个新的多播委托对象。
    • 使用 + 合并两个委托对象,生成一个新的多播委托。
    • 使用 - 从多播委托中移除一个委托,生成一个新的委托对象。
  • +=-= 运算符:用于修改已有的委托实例,直接在原始变量上添加或移除委托。
    • += 将一个委托添加到现有委托链上,结果赋给原变量。
    • -= 从现有委托链中移除一个委托,结果赋给原变量。

下面的程序演示了委托的多播:

using System;delegate int NumberChanger(int n);
namespace DelegateAppl
{class TestDelegate{static int num = 10;public static int AddNum(int p){num += p;return num;}public static int MultNum(int q){num *= q;return num;}public static int getNum(){return num;}static void Main(string[] args){// 创建委托实例NumberChanger nc;NumberChanger nc1 = new NumberChanger(AddNum);NumberChanger nc2 = new NumberChanger(MultNum);nc = nc1;nc += nc2;// 调用多播nc(5);Console.WriteLine("Value of Num: {0}", getNum());Console.ReadKey();}}
}结果:
Value of Num: 75

注意
在C#中,当使用+=操作符向委托添加方法时,有两种方式是等效的:

  • 显式地创建一个新的委托实例并将其添加到现有的委托链中
    myDelegate += new NumberChanger(AddNum);
  • 省略new 部分,直接添加方法。C#编译器会自动为您处理委托的实例化(如果必要的话):myDelegate += AddNum
    这两种方式在功能上是完全相同的。从C# 2.0开始,第二种方式(省略new关键字和委托类型)变得更加流行,因为它更简洁,并且减少了不必要的代码。

1.4 委托的匿名和lambda

二者比较:

特性匿名方法 (delegate)Lambda 表达式
语法简洁性较繁琐,需要显式写出 delegate 关键字和参数列表更简洁,直接用 (参数) => {} 表达逻辑
表达式形式支持不支持表达式形式,必须用 {} 包裹逻辑块支持表达式形式,单行逻辑可以省略 {} 和 return
捕获外部变量(闭包)支持支持
语法风格更接近传统 C# 方法声明更现代、函数式编程风格
语义清晰性delegate 明确表明它是匿名方法使用 => 运算符,强调简洁和函数式思想

1.4.1 匿名方法

匿名方法是通过使用 delegate 关键字创建委托实例来声明的。
语法

delegate(parameters) { statement; }

例如:

delegate void NumberChanger(int n);
...
NumberChanger nc = delegate(int x)
{Console.WriteLine("Anonymous Method: {0}", x);
};

代码块 Console.WriteLine("Anonymous Method: {0}", x); 是匿名方法的主体。

委托可以通过匿名方法调用,也可以通过命名方法调用,即,通过向委托对象传递方法参数。

using System;delegate void NumberChanger(int n);
namespace DelegateAppl
{class TestDelegate{static int num = 10;public static void AddNum(int p){num += p;Console.WriteLine("Named Method: {0}", num);}public static void MultNum(int q){num *= q;Console.WriteLine("Named Method: {0}", num);}static void Main(string[] args){// 使用匿名方法创建委托实例NumberChanger nc = delegate(int x){Console.WriteLine("Anonymous Method: {0}", x);};// 使用匿名方法调用委托nc(10);// 使用命名方法实例化委托nc =  new NumberChanger(AddNum);// 使用命名方法调用委托nc(5);// 使用另一个命名方法实例化委托nc =  new NumberChanger(MultNum);// 使用命名方法调用委托nc(2);Console.ReadKey();}}
}

1.4.2 lambda 表达式

在 C# 2.0 及更高版本中,引入了 lambda 表达式,它是一种更简洁的语法形式,用于编写匿名方法。并且 从 C# 2.0 开始对委托的实例化做了简化,委托类型的实例化在某些情况下可以省略显式使用 new 关键字
使用 lambda 表达式:

using System;delegate void NumberChanger(int n);namespace DelegateAppl
{class TestDelegate{static int num = 10;public static void AddNum(int p){num += p;Console.WriteLine("Named Method: {0}", num);}public static void MultNum(int q){num *= q;Console.WriteLine("Named Method: {0}", num);}static void Main(string[] args){// 使用 lambda 表达式创建委托实例NumberChanger nc = x => Console.WriteLine($"Lambda Expression: {x}");// 使用 lambda 表达式调用委托nc(10);// 使用命名方法实例化委托nc = new NumberChanger(AddNum);// 使用命名方法调用委托nc(5);// 使用另一个命名方法实例化委托nc = new NumberChanger(MultNum);// 使用命名方法调用委托nc(2);Console.ReadKey();}}
}

1.5 内置委托

C# 提供了一些内置的泛型委托,可以覆盖大部分常见场景,主要包括以下几个

1.5.1 Action系列

Action 是一个用于定义没有返回值的方法的委托。支持最多 16 个参数的重载。

Action action = () => Console.WriteLine("No parameters");
action();Action<int, string> actionWithParams = (x, y) => Console.WriteLine($"x: {x}, y: {y}");
actionWithParams(10, "hello");

1.5.2 Func 系列

Func 是一个带有返回值的泛型委托。最多支持 16 个输入参数,最后一个泛型参数是返回值的类型,前面的泛型参数表示输入参数

Func<int, int, int> add = (x, y) => x + y;
int result = add(3, 5);
Console.WriteLine(result); // 输出 8

1.5.3 Predicate

Predicate<T> 是一个返回 bool 的泛型委托,常用于过滤或条件判断。

Predicate<int> isEven = x => x % 2 == 0;
bool check = isEven(4);
Console.WriteLine(check); // 输出 True

1.6 示例

下面的实例演示了委托的用法。委托 printString 可用于引用带有一个字符串作为输入的方法,并不返回任何东西。

我们使用这个委托来调用两个方法,第一个把字符串打印到控制台,第二个把字符串打印到文件:

using System;
using System.IO;namespace DelegateAppl
{class PrintString{static FileStream fs;static StreamWriter sw;// 委托声明public delegate void printString(string s);// 该方法打印到控制台public static void WriteToScreen(string str){Console.WriteLine("The String is: {0}", str);}// 该方法打印到文件public static void WriteToFile(string s){fs = new FileStream("c:\\message.txt", FileMode.Append, FileAccess.Write);sw = new StreamWriter(fs);sw.WriteLine(s);sw.Flush();sw.Close();fs.Close();}// 该方法把委托作为参数,并使用它调用方法public static void sendString(printString ps){ps("Hello World");}static void Main(string[] args){printString ps1 = new printString(WriteToScreen);printString ps2 = new printString(WriteToFile);sendString(ps1);sendString(ps2);Console.ReadKey();}}
}结果:
The String is: Hello World

2 事件

2.1 简介

C# 事件(Event)是一种成员,用于将特定的事件通知发送给订阅者。事件通常用于实现观察者模式,它允许一个对象将状态的变化通知其他对象,而不需要知道这些对象的细节。

事件(Event) 基本上说是一个用户操作,如按键、点击、鼠标移动等等,或者是一些提示信息,如系统生成的通知。应用程序需要在事件发生时响应事件。例如,中断。
C# 中使用事件机制实现线程间的通信。

关键点:

  • 声明委托:定义事件将使用的委托类型。委托是一个函数签名。
  • 声明事件:使用 event 关键字声明一个事件。
  • 触发事件:在适当的时候调用事件,通知所有订阅者。
  • 订阅和取消订阅事件:其他类可以通过 +=-= 运算符订阅和取消订阅事件。

事件模型五个组成部分:

  • 事件的拥有者
  • 事件成员
  • 事件的响应者
  • 事件处理器
  • 事件订阅–把事件处理器与事件关联在一起,本质是一种以委托类型为基础的

2.2 原理

2.2.1 讲解

事件在类中声明且生成,且通过使用同一个类或其他类中的委托事件处理程序关联。包含事件的类用于发布事件。这被称为 发布器(publisher) 类。其他接受该事件的类被称为 订阅器(subscriber) 类。事件使用 发布-订阅(publisher-subscriber) 模型。

  • 发布器(publisher) 是一个包含事件和委托定义的对象。事件和委托之间的联系也定义在这个对象中。发布器类的对象调用这个事件,并通知其他的对象。
  • 订阅器(subscriber) 是一个接受事件并提供事件处理程序的对象。在发布器类中的委托调用订阅器(subscriber)类中的方法(事件处理程序)
    在C#中,通常使用 += 操作符来订阅事件,使用 -= 操作符来取消订阅事件

2.2.2 add 和 remove 访问器

自己定义事件的 addremove 访问器,从而控制事件订阅和取消订阅的具体行为。
下面的 EventHandler 是系统自带 事件,不用声明

public class EventDemo
{private EventHandler _myEvent;// 自定义事件public event EventHandler MyEvent{add{Console.WriteLine("Adding a subscriber");_myEvent += value;}remove{Console.WriteLine("Removing a subscriber");_myEvent -= value;}}public void TriggerEvent(){_myEvent?.Invoke(this, EventArgs.Empty);}
}class Program
{static void Main(){EventDemo demo = new EventDemo();EventHandler handler = (sender, e) => Console.WriteLine("Event triggered!");// 订阅事件demo.MyEvent += handler;  // 输出: Adding a subscriber// 触发事件demo.TriggerEvent();      // 输出: Event triggered!// 取消订阅事件demo.MyEvent -= handler;  // 输出: Removing a subscriber}
}

自定义 add 和 remove 访问器通常在以下场景中使用:

  • 自定义订阅逻辑:需要记录订阅者或对订阅者进行筛选时。
  • 线程安全:确保事件的订阅和取消订阅在多线程环境下安全。
  • 限制订阅数量:控制最多只能有特定数量的订阅者。
  • 日志记录或调试:每次事件订阅或取消时记录相关信息。

注意事项:

  • 事件是委托的包装:事件是基于委托的,但它对委托的直接访问进行了限制,提供了一种更安全的机制来管理委托调用。
  • 事件默认行为:如果不需要特殊逻辑,直接使用默认的 add 和 remove,即可满足大部分场景。
  • 不要直接对事件赋值:只能通过 +=-= 访问事件。直接赋值(如 MyEvent = null)是不允许的,除非是在声明类内部。

2.3 使用原生委托

namespace EventExample
{class Program{MyForm form = new MyForm();// 写此处原生对应的 事件可以先写此处名字,让visualstudio 自动生成对应参数类型的 事件form.Click += form.FormClicked;form.ShowDialog();}class MyForm : Form{internal void FormClicked(object sender,EventArgs e){this.Text = DataTime.Now.ToString();}}
}

2.4 自定义委托

2.4.1 声明

在类的内部声明事件,首先必须声明该事件的委托类型。
例如:

public delegate void BoilerLogHandler(string status);

然后,声明事件本身,使用 event 关键字:

// 基于上面的委托定义事件
public event BoilerLogHandler BoilerEventLog;

上面的代码定义了一个名为 BoilerLogHandler 的委托和一个名为 BoilerEventLog 的事件,该事件在生成的时候会调用委托。

2.4.2 操作

2.4.2.1 示例一

以下示例展示了如何在 C# 中使用事件:

using System;
namespace EventDemo
{// 定义一个委托类型,用于事件处理程序public delegate void NotifyEventHandler(object sender, EventArgs e);// 发布者类public class ProcessBusinessLogic{// 声明事件public event NotifyEventHandler ProcessCompleted;// 触发事件的方法protected virtual void OnProcessCompleted(EventArgs e){ProcessCompleted?.Invoke(this, e);}// 模拟业务逻辑过程并触发事件public void StartProcess(){Console.WriteLine("Process Started!");// 这里可以加入实际的业务逻辑// 业务逻辑完成,触发事件OnProcessCompleted(EventArgs.Empty);}}// 订阅者类public class EventSubscriber{public void Subscribe(ProcessBusinessLogic process){process.ProcessCompleted += Process_ProcessCompleted;}private void Process_ProcessCompleted(object sender, EventArgs e){Console.WriteLine("Process Completed!");}}class Program{static void Main(string[] args){ProcessBusinessLogic process = new ProcessBusinessLogic();EventSubscriber subscriber = new EventSubscriber();// 订阅事件subscriber.Subscribe(process);// 启动过程process.StartProcess();Console.ReadLine();}}
}

说明

  • 定义委托类型:
    public delegate void NotifyEventHandler(object sender, EventArgs e);
    这是一个委托类型,它定义了事件处理程序的签名。通常使用 EventHandlerEventHandler<TEventArgs> 来替代自定义的委托。
  • 声明事件:
    public event NotifyEventHandler ProcessCompleted;
    这是一个使用 NotifyEventHandler 委托类型的事件。
  • 触发事件:
protected virtual void OnProcessCompleted(EventArgs e)
{ProcessCompleted?.Invoke(this, e);
}

这是一个受保护的方法,用于触发事件。使用 ?.Invoke 语法来确保只有在有订阅者时才调用事件。

  • 订阅和取消订阅事件:
    process.ProcessCompleted += Process_ProcessCompleted;
    订阅者使用 += 运算符订阅事件,并定义事件处理程序 Process_ProcessCompleted。
2.4.2.2 示例二
using System;
namespace SimpleEvent
{using System;/***********发布器类***********/public class EventTest{private int value;public delegate void NumManipulationHandler();public event NumManipulationHandler ChangeNum;protected virtual void OnNumChanged(){if ( ChangeNum != null ){ChangeNum(); /* 事件被触发 */}else {Console.WriteLine( "event not fire" );Console.ReadKey(); /* 回车继续 */}}public EventTest(){int n = 5;SetValue( n );}public void SetValue( int n ){if ( value != n ){value = n;OnNumChanged();}}}/***********订阅器类***********/public class subscribEvent{public void printf(){Console.WriteLine( "event fire" );Console.ReadKey(); /* 回车继续 */}}/***********触发***********/public class MainClass{public static void Main(){EventTest e = new EventTest(); /* 实例化对象,第一次没有触发事件 */subscribEvent v = new subscribEvent(); /* 实例化对象 */e.ChangeNum += new EventTest.NumManipulationHandler( v.printf ); /* 注册 */e.SetValue( 7 );e.SetValue( 11 );}}
}结果:
event not fire
event fire
event fire

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

DreamCamera2相机预览变形的处理

最近遇到一个问题&#xff0c;相机更换了摄像头后&#xff0c;发现人像角度顺时针旋转了90度&#xff0c;待人像角度正常后&#xff0c;发现 预览时图像有挤压变形&#xff0c;最终解决。在此记录 一人像角度的修改 先放示意图 设备预览人像角度如图1所示&#xff0c;顺时针旋…

鸿蒙修饰符

文章目录 一、引言1.1 什么是修饰符1.2 修饰符在鸿蒙开发中的重要性1.3 修饰符的作用机制 二、UI装饰类修饰符2.1 Styles修饰符2.1.1 基本概念和使用场景2.1.2 使用示例2.1.3 最佳实践 2.2 Extend修饰符2.2.1 基本概念2.2.2 使用示例2.2.3 Extend vs Styles 对比2.2.4 使用建议…

phpmyadmin导出wordpress数据教程

网站搬家或网站修改&#xff0c;需要导出数据或备份数据&#xff0c;一般主机控制面板最常用phpmyadmin。下面这个是一个有详细图文操作步骤的phpmyadmin导出wordpress数据教程。 1、先登陆到你主机的管理面板&#xff0c;找到数据库&#xff0c;再找到phpmyadmin&#xff0c;…

阅读笔记——SVD本质+计算+应用

摘要&#xff1a;本文讨论了正交相似对角化的方法&#xff0c;几何含义&#xff1b;方阵&#xff0c;非方阵的奇异值分解的计算&#xff0c;矫正方法以及与正交相似对角化存在区别&#xff1b;最后讨论了奇异值分解的应用。 1.实对称矩阵A的变换是一种线性变换&#xff0c;对应…

Linux网络_网络协议_网络传输_网络字节序

一.协议 1.概念 协议&#xff08;Protocol&#xff09; 是一组规则和约定&#xff0c;用于定义计算机网络中不同设备之间如何进行通信和数据交换。协议规定了数据的格式、传输方式、传输顺序等详细规则&#xff0c;确保不同设备和系统能够有效地互联互通。 在网络通信中&#…

数据结构 (16)特殊矩阵的压缩存储

前言 特殊矩阵的压缩存储是数据结构中的一个重要概念&#xff0c;它旨在通过找出特殊矩阵中值相同的矩阵元素的分布规律&#xff0c;把那些呈现规律性分布的、值相同的多个矩阵元素压缩存储到一个存储空间中&#xff0c;从而节省存储空间。 一、特殊矩阵的定义 特殊矩阵是指具有…

ehr系统建设方案,人力资源功能模块主要分为哪些,hrm平台实际案例源码,springboot人力资源系统,vue,JAVA语言hr系统(源码)

eHR人力资源管理系统&#xff1a;功能强大的人力资源管理工具 随着企业规模的不断扩大和业务需求的多样化&#xff0c;传统的人力资源管理模式已无法满足现代企业的需求。eHR人力资源管理系统作为一种先进的管理工具&#xff0c;能够为企业提供高效、准确、实时的人力资源管理。…

搭建AD域服务器

搭建AD域服务器 使用深信服HCI搭建AD域服务器 1、新建虚拟机 2、填写参数 3、省略安装过程 4、进入服务器管理器 5、 6、 7、 8、 9、 10、 11、 12、 13、 14、 15、 16、 17、 18、 19、 20、 21、 22、 23、

MOH: MULTI-HEAD ATTENTION AS MIXTURE-OFHEAD ATTENTION

当前的问题 多头注意力使用多个头部可以提高模型的精度。然而&#xff0c;并不是所有的注意力头都具有同样的重要性。一些研究表明&#xff0c;许多注意力头可以被修剪而不影响准确性。 此外&#xff0c;在多头注意中&#xff0c;每个注意头并行操作&#xff0c;最终输出是所…

Spring boot之BeanDefinition介绍

在spring框架中IOC容器进行bean的创建和管理。Bean的创建是一个比较复杂的过程&#xff0c;它并不像我们创建对象一样只是直接new一下就行&#xff0c;虽然有些bean确实就是New一下。但在Spring中可以通过一些途径对bean进行增强扩展。在这个过程中&#xff0c;BeanDefinition作…

Ubuntu 服务器部署 Tomcat 并配置 SSL/TLS 证书

本文目录 准备登陆云服务器安装 Java下载 tomcat 包配置防火墙浏览器访问 Tomcat 默认页面以服务的形式运行 Tomcat创建 Tomcat 用户和组创建 systemd 服务文件启动 tomcat 服务 Tomcat webapps 文件目录部署一个静态网站tomcat 的配置文件 将域名解析到服务器Tomcat 配置 SSL/…

C++小问题

怎么分辨const修饰的是谁 是限定谁不能被改变的&#xff1f; 在C中&#xff0c;const关键字的用途和位置非常关键&#xff0c;它决定了谁不能被修改。const可以修饰变量、指针、引用等不同的对象&#xff0c;并且具体的作用取决于const的修饰位置。理解const的规则能够帮助我们…

PPT不能编辑,按钮都是灰色,怎么办?

PPT文件打开之后&#xff0c;发现无法编辑&#xff0c;再仔细查看发现工具栏中的功能按钮都是灰色的&#xff0c;无法使用&#xff0c;这是什么原因&#xff1f;该如何解决&#xff1f; 原因&#xff1a;无法编辑PPT文件&#xff0c;并且功能按钮都是灰色&#xff0c;这是因为…

相交链表和环形链表

&#xff08;一&#xff09;相交链表 相交链表 思路&#xff1a;先分别计算出A列表和B列表的长度&#xff0c;判断它们的尾节点是否相等&#xff0c;如果不相等就不相交&#xff0c;直接返回空。然后让两个列表中的长的列表先走它们的差距步&#xff0c;然后再一起走&#xff…

ARM架构下安装新版docker及docker-compose

一、常见CPU 架构&#xff1a; 二、环境信息 CPU架构操作系统配置HUAWEI Kunpeng 920 5220 aarch64openEuler 22.03 (LTS-SP3)64C128g15T 三、安装docker 3.1 二进制包下载 docker-ce 社区下载地址&#xff1a; wget https://mirrors.nju.edu.cn/docker-ce/linux/static/s…

LeetCode-315. Count of Smaller Numbers After Self

目录 题目描述 解题思路 【C】 【Java】 复杂度分析 LeetCode-315. Count of Smaller Numbers After Selfhttps://leetcode.com/problems/count-of-smaller-numbers-after-self/description/ 题目描述 Given an integer array nums, return an integer array counts whe…

【NLP 4、数学基础】

此去经年&#xff0c;应是良辰美景虚设 —— 24.11.28 一、线性代数 1.标量和向量 ① 标量 Scalar 一个标量就是一个单独的数 ② 向量 Vector 一个向量是一列数 可以把向量看作空间中的点&#xff0c;每个元素是不同坐标轴上的坐标 向量中有几个数&#xff0c;就叫作几维…

VideoBooth: Diffusion-based Video Generation with Image Prompts

VideoBooth: Diffusion-based Video Generation with Image Prompts 概括 文章提出了一个视频生成模型VideoBooth&#xff0c;输入一张图片和一个文本提示词&#xff0c;即可输出保持图片中物体且符合文本提示词要求的视频。 方法 粗-细两阶段设计&#xff1a;1&#xff09;…

Graphy 是一款终极、易于使用、功能齐全的 FPS 计数器、统计监视器和调试器,适用于您的 Unity 项目。

主要特点&#xff1a; Graph & Text: 图文&#xff1a; FPSMemory 记忆Audio 声音的Advanced device information 高级设备信息Debugging tools 调试工具 GitHub - Tayx94/graphy:Graphy 是适用于 Unity 项目的终极、易于使用、功能丰富的 FPS 计数器、统计监视器和调试…

ASP.NET Core 负载/压力测试

文章目录 一、第三方工具二、使用发布版本进行负载测试和压力测试 负载测试和压力测试对于确保 web 应用的性能和可缩放性非常重要。 尽管负载测试和压力测试的某些测试相似&#xff0c;但它们的目标不同。 负载测试&#xff1a;测试应用是否可以在特定情况下处理指定的用户负…