null
。所以在使用委托变量前最好做非空校验 weituo!=null
。委托,像 C++ 的函数指针。
static void Main() { MyDelegate md = new MyDelegate(M1); md(); MyDelegate md1 = M1; // 这种定义的方法是 // 和上面是一样的,省略了 new。 md1.Invoke(); // 这种调用的方法是上面写法的完整版。 // Invoke(/*有参数的话可以在括号中传递。*/) } static void M1() { Console.WriteLine("M1 函数。"); } public delegate void MyDelegate();
输出:
我是 M1 函数 我是 M1 函数 请按任意键继续.
public class MyClass { public void Say() { /* ...... 这里是要执行的代码,而这段代码的不确定性很大。根据调用者的不同可能执行的 代码不一样。这时候就可以把这段代码封装起来,封装成一个委托类型。通过委托 变量来代替,将来你传递一个什么委托变量,这个地方就执行什么方法。 这样的意思,代码提前写好了,在这个地方先占一个位,这里执行的代码是活的。 是由传递过来的方法确定的。 ......*/ } }
委托就是事先占了一个位置,执行的方法待定,由传递过来的方法决定,较的灵活。
练习3: // 把邮箱名替换成※号 zxh@itcast.cn → ***@itcast.cn steve_zhao@163.com → **********@163.com sk@codeedu.com → **@codeedu.com
static void Main() { while(true) { Console.WriteLine("请输入一个邮箱地址:"); string email = Console.ReadLine(); email = Regex.Replace(email, @"(\w+)(@\w+\.\w)", ReplaceMethod, RegexOptions.ECMAScript); // ▲ 第三个参数直接传递一个返回值为 string 类型的方法, // 这里就用的委托,不是定死的,是活的。 Console.WriteLine(email); } } static string ReplaceMethod(Match match) { string uid = match.Groups[1].Value; string others = match.Groups[2].Value; StringBuilder sb = new StringBuilder(); for (int i = 0; i < uid.Length; i++) { sb.Append("*"); } return sb.ToString() + others; }
delegate
返回值类型 委托类型名(参数) 比如 delegate void StringProcess(string s);
注意这里的除了前面的 delegate
,剩下部分和声明一个函数一样,但是StringProcess
不是函数名,而是委托类型名。int
、Person
一样,如果要用的话还要声明委托类型的变量,声明委托类型变量的方式:StringProcess f1;
StringProcess sp = new StringProcess(SayHello)
,这样就可以像调用普通函蒙一样把sp
当成函教用了.委托可以看做是函数的指针.整数可以用整数变量指向它,对象可以用对象变量指向它,函数也可以用委托变量指向它.和直接调用函数的区别:用委托就可以指向任意的函数,哪怕是之前没定义的都可以,而不使用受限于那几种。StringProcess sp = SayHelo
,编译器帮我们进行了new
.但是不能sp = Printit()
,因为这样就成了函数调用。1.link-01 // C#.Net基础加强十一天-委托 事件 反射