• 2011 年最佳代码


    c# 扩展方法奇思妙用滥用篇一:改进 2011 年最佳代码

    2011-08-08 18:39 by 鹤冲天, 2219 visits, 收藏, 编辑

    今天从老赵 的文章《谈谈年度最佳代码“不管你们信不信,反正我信了”》中学习了 2011 年最佳代码:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    try
    {
        if (you.believe(it) || !you.believe(it))
        {
            I.believe(it);
        }
    }
    catch (Exception ex)
    {
        throw new Exception("It's a miracle!");
    }
    finally
    {
        it.justHappened();
    }

    老赵 在文中强调了语义编程,建议大家去看看。

    在我看来,第三行代码可以改进一些。

    信不信,引申出来听不听、做不做、想不想,隐约中有”或者不”的意味,于是 OrNot 扩展方法在我大脑中出现了。

    不过下手之前,还得准备一下:

    准备工作

    先来准备两个接口,表示上面代码中 you、I、it:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    public interface IPerson
    {
        bool Believe(IThing thing);
    }
    
    public interface IThing
    {
        void JustHappened();
    }

    这三个变量可如下声明:

    1
    2
    3
    IThing it = null;
    IPerson you = null;
    IPerson I = null;

    OrNot 扩展方法

    我最初的想法是,既然是个扩展,就应该让它应用尽量广泛,于是写出的下面的泛型扩展:

    1
    2
    3
    4
    5
    6
    7
    public static class Extensions
    {
        public static bool OrNot<T>(this Predicate<T> predicate, T t)
        {
            return predicate(t) || !predicate(t);
        }
    }

    不过用起来不方便,如把代码写成下面的样子,是编译不通过的:

    1
    2
    3
    4
    if(you.Believe.OrNot(it))
    {
        I.Believe(it);
    }

    提示错误:

    'IPerson.Believe(IThing)' is a 'method', which is not valid in the given context

    只好写成为:

    1
    2
    3
    4
    5
    Predicate<IThing> youBelieve = you.Believe;
    if(youBelieve.OrNot(it))
    {
        I.Believe(it);
    }

    或者:

    1
    2
    3
    4
    if (((Predicate<IThing>)you.Believe).OrNot(it))
    {
        I.Believe(it);
    }

    这两种方式我都看不上眼,不过第二种方式给了一些灵感,于是做了些改进。

    改进

    上面代码使用了 Predicate<T> 委托,代表输入一个 T 返回一个布尔值。

    在本应用中,是输入一个 IThing 返回一个布尔值,可表示为 Predicate<IThing>,写起来太烦索了。

    何不就当前语境定义一个新的委托呢?如下:

    1
    public delegate bool WhatEver (IThing obj);

    相应扩展方法修改为:

    1
    2
    3
    4
    5
    6
    7
    public static class Extensions
    {
        public static bool OrNot(this WhatEver func, IThing t)
        {
            return func(t) || !func(t);
        }
    }

    代码可写成:

    1
    2
    3
    4
    if (((WhatEver)you.Believe).OrNot (it))
    {
        I.Believe(it);
    }

    准确的的表达出了”不管你们不信“的意思。

    完整代码:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    try
    {
        if (((WhatEver)you.Believe).OrNot(it))
        {
            I.Believe(it);
        }
    }
    catch (Exception ex)
    {
        throw new Exception("It's a miracle!");
    }
    finally
    {
        it.JustHappened();
    }

    再次改进

    回复中有朋友问是否可写成 ((WhatEver)you.Believe(it).OrNot(),读起来顺多了。

    这样可能不恰当,写成如下道是可以的:

    1
    2
    3
    4
    if (whatEver(you.Believe)(it).OrNot())
    {
        I.Believe(it);
    }

    这时 whatEver 是个静态变量:

    1
    public static Func<Predicate<IThing>, Func<IThing, Func<bool>>> whatEver = p => i => (() => p(i));

    这种方式在我的文章《借助委托精简代码》中有提及。

    配合如下扩展方法:

    1
    2
    3
    4
    5
    6
    7
    public static class Extensions
    {
        public static bool OrNot(this Func<bool> func)
        {
            return func() || !func();
        }
    }

    完整代码如下:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    namespace ImproveBestCodeOf2011
    {
        class Program
        {
            public static Func<Predicate<IThing>, Func<IThing, Func<bool>>> whatEver = p => i => (() => p(i));
            static void Main(string[] args)
            {
                IThing it = null;
                IPerson you = null;
                IPerson I = null;
                if (whatEver(you.Believe)(it).OrNot())
                {
                    I.Believe(it);
                }
            }
        }
        public static class Extensions
        {
            public static bool OrNot(this Func<bool> func)
            {
                return func() || !func();
            }
        }
        public interface IPerson
        {
            bool Believe(IThing thing);
        }
        public interface IThing
        {
            void JustHappened();
        }
    }

    呵!这种方式又滥用了委托!

    小结

    使用一段代码结束本文(可跳过):

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    static void Main(string[] args)
    {
        IThing it = new OrNotExtensionIsTheBestOf2011();
        IPerson you = new Person { Name = "看本文的你" };
        IPerson I = new Person { Name = "鹤冲天" };
    
        if (((WhatEver)you.Believe).OrNot(it))
        {
            I.Believe(it);
        }
        //or 
        if (whatEver(you.Believe)(it).OrNot())
        {
            I.Believe(it);
        }
    
     }
    
    public class OrNotExtensionIsTheBestOf2011: IThing
    {
        public string Title
        { 
            get { return "OrNot 扩展方法是 2011 年最佳扩展"; }
        }
    
        public void JustHappened()
        {
            Console.WriteLine("OrNot 扩展方法被评为 2011 年最佳扩展");
        }
    }
    
    public class Person : IPerson
    {
        public string Name { get; set; }
    
        public bool Believe(IThing thing)
        {
            if(thing is OrNotExtensionIsTheBestOf2011)
            {
                var theBest = thing as OrNotExtensionIsTheBestOf2011;
                Console.WriteLine("支持 {0}", theBest.Title);
                Console.WriteLine("推荐 {0}", theBest.Title);
                return true;
            }
            throw new NotImplementedException();
        }
    }

    输出:

    image

    本文说笑,且莫当真。

    滥用扩展方法,切务模仿!

    c#扩展方法奇思妙用》系统文章从 2009 年 08 月开始写起,到现在一共有了 23 篇,欢迎阅读:
    基础篇: 中文处理string 常用扩展byte 常用扩展Random 扩展Dictionary<TKey, TValue> 扩展WhereIf 扩展IsBetween 通用扩展WhereIf 扩展Distinct 扩展
    高级篇: 改进 Scottgu 的 "In" 扩展Aggregate扩展其改进Enumerable.Cast<T>应用对扩展进行分组管理ToString(string format) 扩展WinForm 控件选择器树”通用遍历器Type类扩展
    变态篇: 由Fibonacci数列引出“委托扩展”及“递推递归委托”封装 if/else、swith/case及whileswitch/case 组扩展string 的翻身革命
    性能篇 扩展方法性能初测
    MVC篇: 巧用扩展方法优先级,美化所有页面TextBoxFor文本框

    -------------------

    思想火花,照亮世界

  • 相关阅读:
    技术人的思维修炼
    COMMIT在FORM中用法
    编译FORM 时出现错误 bad bind variable ’parameter.G_query_find‘
    描述性弹性域
    ORACLE EBS WebService推送报文例子 XML格式
    EBS值集,弹性域常用表
    ebs Form 表单个性化
    ebs界面颜色改变
    Oracle EBS数据定义移植工具:FNDLOAD
    在请求的参数中设置可选值列表为当前职责可访问的所有OU
  • 原文地址:https://www.cnblogs.com/Leo_wl/p/2132649.html
Copyright © 2020-2023  润新知