1.解剖Linq to object

  LINQ想必大家都不陌生了,它的出现使得我们的代码变得更短、更优雅了。至于LINQ是什么,Linq to object这类的扩展方法到底做了些什么。我们使用的EF是如何实现的(如何解析Expression)。我都将一一为大家分享下我的理解。如果有说得不对的地方,也请大家指出。下面进入正题

 

 

大家先将代码左上角的using System.Linq去掉

当我们想对一个集合的数据进行筛选的时候,我们以前会这样做

        static void Main(string[] args)
        {
            List<int> list = new List<int>()
            {
                2,3,6,4,7,8
            };

            List<int> result = new List<int>();
            foreach (var item in list)
            {
                if (item < 4)
                {
                    result.Add(item);
                }
            }
        }

  

但我们仔细想想,我们要筛选数据时,就一定要遍历它,所以循环这部份是必须的。还有什么是必须的呢?筛选条件!筛选数据时筛选的条件是不同的,但它们有个共同的特点,那就是条件的结果不是true就是false。既然有这么多的共同点,我们能不能把它抽象出来写成一个通用的扩展方法供我们调用呢?聪明的你估计已经想到这个方法大概要怎样构造了。这个方法的参数有2个,一个是我们要进行数据筛选的集合,另一个是我们的筛选条件。方法体里面要做的就是去遍历这个集合,当满足筛选条件时,我们就把这个元素存到结果集合中,最后返回结果集合。想必完成后的代码是这样的

 

        public static List<int> Where(List<int> list, Func<int, bool> func)
        {
            List<int> result = new List<int>();
            foreach (int item in list)
            {
                bool istrue = func(item);//执行我们的筛选函数
                if (istrue)//如果满足筛选条件,就添加进result集合中
                {
                    result.Add(item);
                }
            }
            return result;
        }

 

这时我们就可以这样进行数据筛选了,是不是清爽简洁了很多?

        static void Main(string[] args)
        {
            List<int> list = new List<int>()
            {
                2,3,6,4,7,8
            };
            var result = Where(list, x => x < 4);
            var result2 = Where(list, x => x / 2 == 0);
        }

  

但问题来了。我们的方法只支持List<int>啊,如果换成int[]就不行了。怎么解决呢?仔细想想我们只需要遍历它就可以了,不需要对它操作啊,想到遍历想到什么?foreach!想到foreach想到什么?IEnumerable!不多说。马上搞起。修改后应该变成这样

 

        static void Main(string[] args)
        {
            List<int> list = new List<int>()
            {
                2,3,6,4,7,8
            };
            int[] array = list.ToArray();
            var result = Where(array, x => x < 4);
            var result2 = Where(list, x => x / 2 == 0);
        }

        public static IEnumerable<int> Where(IEnumerable<int> list, Func<int, bool> func)
        {
            foreach (int item in list)
            {
                bool istrue = func(item);
                if (istrue)
                {
                    yield return item;
                }
            }
        }

  

看似已经完成了,但我们往细的想想,我们的方法只支持int类型,我们想把它变成通用的,任何类型都支持。这时我们想到什么?没错,泛型。。

还有,我们平时都是list.Where(条件)来调用的。相信很多同学都知道吧,这种叫做扩展方法。我们如何改呢?下面让我们来一口把这个Where方法完成吧!

 

    class Program
    {
        static void Main(string[] args)
        {
            List<int> list = new List<int>()
            {
                2,3,6,4,7,8
            };
            int[] array = list.ToArray();
            array.Where(x => x > 0);
            list.Where(x => x / 2 == 0);
        }
    }

    public static class MyLinq
    {
        public static IEnumerable<T> Where<T>(this IEnumerable<T> list, Func<T, bool> func)
        {
            foreach (T item in list)
            {
                bool istrue = func(item);
                if (istrue)
                {
                    yield return item;
                }
            }
        }
    }

  

小贴士:扩展方法就是静态类中的静态方法,参数有this xxx。这个静态类不能是内部类

 

 

想必大家现在对LINQ有了一定的认识了吧?看完后是不是觉得LINQ其实就那么回事呢?看完后也要做下练习来巩固下吧。下面请大家自行完成一个Select扩展方法。答案在文章末尾。

这也是我第一篇博文,写的时候手都在抖,怕表达不清或说错了什么误导了大家。很早就开始想写博文了,但都太忙了。现在总算有个开始了

 

 

 

 

 

 

 

 

 

 

 

练习题答案

        public static IEnumerable<TResult> Select<TInput, TResult>(this IEnumerable<TInput> list, Func<TInput, TResult> func)
        {
            foreach (TInput item in list)
            {
                TResult result = func(item);
                yield return result;
            }
        }

  

 

posted @ 2014-12-10 15:01  Poiuyt_cyc  阅读(1453)  评论(4编辑  收藏  举报