Linq

Enumerable Where Select

Linq 讀作 link 語言集成查詢(Language Integrated Query),讓我們可以像操作數據庫那樣操作內存數據。它有個特點,在使用時執行
如何在使用時獲取數據,就要用到一個關鍵字yield,使用時和return一起出現。

先看個例子

        static void Main(string[] args)
        {
            foreach (var item in GenerateStrings())
            {
                Console.WriteLine(item.ToString());
            }
        }
        static IEnumerable<string> GenerateStrings()
        {
            for (int i = 0; i < 10; i++)
            {
                yield return i.ToString();
            }
        }

在foreach的數據源中,調用這個方法。在輸出那設一個斷點,單步執行,可以看到,數據源并不是在方法執行完成后再返回到foreach中,而是每調用到下一個數據時,在方法中獲得下一個數據。這就是yield return.
這種做法有什么好處?在需要的時候調用,數據只在需要用到的時候調入內存,而不是把整個集合都調入內存,節省存儲空間,在這個例子里其實并不能很好的體現好嘛。

        static void Main(string[] args)
        {
            var sequence = GenerateStrings();//==①
            sequence = sequence.Where(x => x.Length < 2);//==②    x => x.Length < 2//==⑤
            foreach (var item in sequence)//==③
            {
                Console.WriteLine(item.ToString());//==⑥
            }
        }
        static IEnumerable<string> GenerateStrings()//==④
        {
            for (int i = 8; i < 100; i++)
            {
                yield return i.ToString();
            }
        }

在遇到第①,②句時,并沒有進到 GenerateStrings其中,而是遇到foreach時,依次進到GenerateStrings中,取到返回值后,再判斷⑤子句,成立后才會輸出。
但我在調試的時候明明沒有進到GenerateStrings,他就已經有結果了,不懂。

再看一個,我們自己寫一個擴展方法,功能同上一個的where子句

    public static class MyLinq
    {
        //一個擴展方法
        public static IEnumerable<string> MyWhere(this IEnumerable<string> source)
        {
            foreach (string item in source)
            {
                if (item.Length < 2)
                    yield return item;
            }
        }
    }
    class Program
    {

        static void Main(string[] args)
        {
            var sequence = GenerateStrings();
            sequence = sequence.Where(x => x.Length < 2);
            var sequence2 = GenerateStrings();
            sequence2 = sequence2.MyWhere();
            foreach (var item in sequence)
            {
                Console.WriteLine(item.ToString());
            }
            foreach (var item in sequence2)
            {
                Console.WriteLine(item.ToString());
            }
            Console.ReadKey();
        }
        static IEnumerable<string> GenerateStrings()
        {
            for (int i = 8; i < 100; i++)
            {
                yield return i.ToString();
            }
        }

    }

輸出都是一樣的
8
9
8
9

介紹一下第二個foreach的執行過程
遇到sequence2 = sequence2.MyWhere();時是先不執行的
在foreach到的時候才會執行MyWhere
到MyWhere中的foreach后,需要一個數據源
那么就去執行var sequence2 = GenerateStrings()
依次的獲取數據,拿到一個8后,判斷長度是否小于2,滿足,返回8,輸出。
9也是一樣
到10的時候,判斷不小于2
重復獲取數據,判斷,直到99,沒有了,退出。

下面改一下那個擴展方法

 public static IEnumerable<string> MyWhere(this IEnumerable<string> source,Func<string,bool> predicate)
        {
            foreach (string item in source)
            {
                if (predicate(item))
                    yield return item;
            }
        }

還有這里
sequence2 = sequence2.MyWhere(x=>x.Length<2);

這樣以后,我們的擴展方法就更加靈活,功能上和C#提供的Where就一樣了。后面是一個委托,用來判斷條件,返回一個bool值,在if里判斷。
再牛逼一點,把我們的MyWhere變成一個泛型方法

public static IEnumerable<T> MyWhere<T>(this IEnumerable<T> source, Func<T, bool> predicate)
        {
            foreach (T item in source)
            {
                if (predicate(item))
                    yield return item;
            }
        }

改完這個,主函數里是不用動的,泛型大法好。會玩的你知道下一步怎么玩了么?

        static IEnumerable<int> GenerateIntegers()
        {
            for (int i = 8; i < 100; i++)
            {
                yield return i;
            }
        }

寫個返回整數的方法咯。
var sequence3 = GenerateIntegers().MyWhere(x => x % 7 == 0);
這個就可以拿到所有7的倍數啦。
但是要注意到,我們剛才那個泛型方法沒有任何錯誤檢查機制,所以…

下面是select

select就比較簡單了
var sequence4 = GenerateIntegers().Select(x => x.ToString());
var sequence5 = GenerateIntegers().Select(x =>true);
自己感受一下select后面的lambda表達式

如果,我們來重寫一下這個select

        public static IEnumerable<string> MySelect(this IEnumerable<int> source,Func<int,string> selector)
        {
            foreach (int item in source)
            {
                yield return selector(item);
            }
        }

像這樣
var sequence4 = GenerateIntegers().MySelect(x => x.ToString());
感受一下,結果是一樣的。
那么我們又可以寫個泛型擴展方法了

        public static IEnumerable<TResult> MySelect<TSource,TResult>(this IEnumerable<TSource> source, Func<TSource, TResult> selector)
        {
            foreach (TSource item in source)
            {
                yield return selector(item);
            }
        }

其實select 有一個重載的版本是可以獲取索引的

            var sequence6 = GenerateIntegers().Select((x,index)=>new { index, str=x.ToString()+"str" });
            foreach (var s in sequence6)
            {
                Console.WriteLine("{0}=={1}", s.index, s.str);
            }

借助一個匿名類型,我們可以同時得到索引和內容。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,563評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,694評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,672評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,965評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,690評論 6 413
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,019評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,013評論 3 449
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,188評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,718評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,438評論 3 360
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,667評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,149評論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,845評論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,252評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,590評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,384評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,635評論 2 380

推薦閱讀更多精彩內容

  • 1. Java基礎部分 基礎部分的順序:基本語法,類相關的語法,內部類的語法,繼承相關的語法,異常的語法,線程的語...
    子非魚_t_閱讀 31,737評論 18 399
  • 《ijs》速成開發手冊3.0 官方用戶交流:iApp開發交流(1) 239547050iApp開發交流(2) 10...
    葉染柒丶閱讀 5,275評論 0 7
  • 首先申明這不是我寫的,只是我看到的一篇十分不錯的文章,為了防止自己找不到,就整理了下。 一:與LINQ有關的語言特...
    破冰前行閱讀 2,735評論 0 6
  • 應用程序還需要操作存儲在其他數據源(如SQL數據庫或XML文件)中的數據,甚至通過Web服務訪問它們。傳統上,查詢...
    CarlDonitz閱讀 601評論 0 0
  • 中午吃什么呢?看還有兩個紫皮洋蔥,那就炒洋蔥吃吧。扒皮洗凈開切,一如既往地我又被嗆得鼻子酸眼淚汪汪,有點睜不開眼喘...
    幸福_娟閱讀 641評論 6 4