zoukankan      html  css  js  c++  java
  • 委托的使用

    • 委托是一种类,所以声明的时候尽量和其他类保持平级的位置
    • 委托与所封装的方法必须“类型兼容”,也就是返回值的数据类型一致,参数列表个数和数据类型上一致

        delegate double Calc(double x,double y)//委托封装方法的类型

             double Add(double x,double y)//方法


    一般使用委托,是为了把方法作为参数传到另一个方法里,这样主方法,里面套了个方法,就能动了。有两种使用方法

    1.模板方法:借用指定的外部方法来产生结果

    • 相当于“填空题”
    • 常常位于代码中部
    • 委托带有返回值

    2.回调方法:回调方法,调用制定的外部方法

    • 相当于“流水线”
    • 常常位于代码末尾
    • 委托无返回值
    class Program
        {
            static void Main(string[] args)
            {
                //IProductFactory pizzaFactory = new PizzaFactory();
                //IProductFactory toycarFactory = new ToyCarFactory();
    
                ProductFactory productFactory = new ProductFactory();
                WrapFactory wrapFactory = new WrapFactory();
                Logger logger = new Logger();
    
                Action<Product> log = new Action<Product>(logger.Log);//Action委托实例封装了logger
    
                Func<Product> func1 = new Func<Product>(productFactory.MakePizza);//Func委托实例封装了ProductFactory的方法
                Func<Product> func2 = new Func<Product>(productFactory.MakeToyCar);
    
    
                Box box = wrapFactory.WrapProduct(func1, log);//调用模板方法(主方法)
                Box box1 = wrapFactory.WrapProduct(func2, log);
    
                //Box box = wrapFactory.WrapProduct(pizzaFactory);
                //Box box1 = wrapFactory.WrapProduct(pizzaFactory);
                 Console.WriteLine(box.Product.Name);
                 Console.WriteLine(box1.Product.Name);
                 Console.ReadLine();
            }
        }
    
        interface IProductFactory
        {
            Product Make();
        }
        class PizzaFactory : IProductFactory
        {
            public Product Make()
            {
                Product product = new Product();//
                product.Name = "Pizza";
                product.Price = 12;
                return product;
            }
        }
        class ToyCarFactory : IProductFactory
        {
            
               public Product Make()
            {
                Product product = new Product();
                product.Name = "ToyCar";
                product.Price = 100;
                return product;
            }
            
        }   
        class Logger
        { 
        public void Log(Product product)
            {
                Console.WriteLine("Product ‘{0} created at {1}.Price is {2}’",product.Name,DateTime.UtcNow,product.Price);
    
            }
        }
        class Product
        {
            public string Name{get;set;}
            public double Price { get; set; }
        }
        class Box
        {
            public Product Product { get; set; }
        }
        class WrapFactory//把产品装上盒子交给用户
        {
    
            //public Box WrapProduct(IProductFactory productFactory)//用接口重构委托
            //{
            //    Box box = new Box();
            //    Product product = productFactory.Make();
    
            //    //if (product.Price >= 50)
            //    //{
            //    //    logCallback(product);
            //    //}
            //    //box.Product = product;
            //    return box;
            //}
            public Box WrapProduct(Func<Product> getProduct, Action<Product> logCallback)//模板方法,好处在于Product,Box,都可以不动,只在生产产品时不断添加方法
            {
                Box box = new Box();
                Product product = getProduct.Invoke();
    
                if (product.Price >= 50)
                {
                    logCallback(product);
                }
                box.Product = product;
                return box;
            }
        }
        class ProductFactory
        {
            public Product MakePizza()
            {
                Product product = new Product();
                product.Name = "Pizza";
                product.Price = 12;
                return product;
            }
            public Product MakeToyCar()
            {
                Product product = new Product();
                product.Name = "ToyCar";
                product.Price = 100;
                return product;
            }
        }
    

      

     缺点

    • 方法级别的耦合,违反设计模式
    • 可读性下降,debug难度增加
    • 委托回调,异步调用和多线程放在一起,会让代码难以阅读和维护
    • 使用不当内存泄漏,委托引用了一个方法,如果这个方法是实例方法,那委托会一直占用这个实例方法导致内存无法释放实例
  • 相关阅读:
    "etc/profile" E212: Can't open file for writing
    Swift 判断是否是调试模式以及是否越狱
    密码技术之基本介绍
    算法
    App Thinning (App 瘦身)
    #pragma once vs #ifndef
    APUE学习之进程控制
    Socket编程-基础使用
    Link Script 学习
    PX4学习之-uORB msg 自动生成模板解读
  • 原文地址:https://www.cnblogs.com/1521681359qqcom/p/11216187.html
Copyright © 2011-2022 走看看