委托
/// <summary>
/// 1.简介
/// 委托是存有对某个方法的引用的一种引用类型变量。说白了就是对方法的引用。引用可以在运行时被改变。
/// 委托特别用于实现事件和回掉方法。
/// 在使用委托前要先声明一下。
///委托类似于C++中的指针,但是更安全
/// 它不知道也不关心所引用的方法的类;只关心引用的方法是否具有与委托相同的参数和返回类型。
///
/// 2.多播委托:委托对象可以使用 + 运算符进行合并。一个合并委托调用它所合并的两个委托。
只有相同类型的委托才可合并。 - 运算符表示从合并的委托中移除组件委托
///
///3.委托是C#中最为常见的内容。与类、枚举、结构、接口一样,委托也是一种类型。类是对象的抽象,而委托则可以看成是函数的抽象。
一个委托代表了具有相同参数列表和返回值的所有函数。可以实现异步线程操作UI,可以实现观察者模式等,可以将方法当作参数操作等
/// </summary>
/*声明委托,定义参数类型*/
delegate int NumberChange(int n);
class DelegateTest
{
static int num = 10;
public static int Add(int a) { return num += a; }
public static int Minus(int a) { return num -= a; }
public static int GetNum{ get { return num; } }
static void Main(string[] args)
{
//创建委托实例,一对一型
NumberChange nc = new NumberChange(Add);
NumberChange nc1 = new NumberChange(Minus);
//调用委托
nc(20);
Console.WriteLine(GetNum);
nc1(5);
Console.WriteLine(GetNum);
//实现多播委托
NumberChange n;
NumberChange n1 = new NumberChange(Add);
NumberChange n2 = new NumberChange(Minus);
n = n1;
n += n2;
//调用多播
n(5);
Console.WriteLine(GetNum);
}
}
看完以上的基础,我们可能会有一个疑问,委托简直就是把一个函数交给另一个东西执行嘛,好像没什么突出的优点。
让我们不妨将情景切换到游戏中:
如果我们游戏中有一个核心功能SpawnEnemy(),我们怎么在很多怪物类型中产生出带有不同技能的怪物呢?疯狂用ifelse肯定是不行的,这对开发和维护都是噩梦。这时,你只需要使用委托(我们就叫它OnSpawnEnemy()吧),然后在每次新调用SpawnEnemy()时,先根据符合的条件,给委托重分配执行函数,然后再通过委托来执行某个特定的生成方法。例如:在25秒时,Manager重分配OnSpawnEnemy = CreateA来创建A怪,在50秒时,重分配OnSpawnEnemy = CreateB来创建B怪,然后我每次调用执行委托时,他重视执行创建的不同的怪物。
委托有两个极好的地方:
1.匿名委托:它可以避免你为委托写很多单独的函数,你只需要按照 OnSpawnEnemy = delegate{ //内容 }来写。但你不应该创建很复杂的匿名委托体(为了调试的缘故)。
而匿名委托真正让人喜欢的地方是,你可以通过给定的状态一览脚本的核心行为痕迹。同时,不写新的函数也能100%适合变化的上下文性质,并保持核心架构的清洁。
2.最重要的就是委托能用 += ,-= 来挂接/卸载执行函数。通常要+完后,在不是用的情况下 - 掉。
当然了,委托也有一些缺点。除非你清楚的知道你自己在做什么,不然它会引起GC问题。这里有两个特例:
1)你添加一个大量消耗内存的委托(例如Texture2D)。你把这个代表挂在一个事件上,然后就好了。回调什么的一切正常。现在由于某种原因,委托的容器消失了(持有Texture2D的游戏对象)。你从来没有删除对事件的委托的引用。 Texture2D仍然被引用。它会一直呆在,直到持有这个是事件的类消亡。由于这些类极有可能是“Manager”,所以你的纹理可能永远不会被GC化
2)当你使用匿名委托,委托引用一个Texture2D。你没有办法从事件中删除委托。这样,你就回到了情况1。 如果你打算使用C#委托,尽量避免匿名委托。在使用c#很nice的特性(如linq和foreach等)时,也要非常小心内存开销。多使用Profiler,并确保检查性能开销。
Action委托
普通委托在使用前要先声明一下,而Action委托简化了这一过程
//普通委托预先声明
//delegate void DisplayMessage(string message);
public class Name
{
private string instanceName;
public Name(string name)
{
this.instanceName = name;
}
public void DisplayToConsole()
{
Console.WriteLine(this.instanceName);
}
public void DisplayToWindow()
{
MessageBox.Show(this.instanceName);
}
}
public class testTestDelegate
{
public static void Main()
{
Name testName = new Name("Koani");
//普通委托
//ShowValue showMethod = testName.DisplayToWindow;
//Action委托
Action showMethod = testName.DisplayToWindow;
showMethod();
}
}
Action<T>委托
///<summary>
说明: Action类型的委托返回值类型必须是void,且传参必须和泛型一致。
如果想要一个参数且有返回值,用Func<T,TResult>委托
///<summary>
public class TestAction
{
public static void Main()
{
Action<string> messageTarget;
int input = int.Parse(Console.ReadLine())
if(input > 1 )
messageTarget = Console.WriteLine;
messageTarget("Hello, World!");
else
//匿名
messageTarget = delegate(string s) { Console.WriteLine(s); };
//lambda表达式:
// messageTarget = s =>Console.WriteLine(s);
messageTarget("ni hao");
}
private static void ShowMessage(string message)
{
Console.WriteLine(message);
}
}
Func委托
Func和Action委托的唯一区别在于Func要有返回值, Action没有返回值
更多Func重载