技巧:LINQ組合查詢?nèi)绾魏喕幊?/h1>
如何通過linq組合查詢來簡化代碼編寫?很多朋友可能都想過問題。下面,就給大家介紹一下我對于linq組合查詢的一些使用心得。
自從學(xué)習(xí)linq以來,我發(fā)現(xiàn)了很多使用linq組合查詢來改善代碼的方式。每一個技巧都讓代碼寫起來更簡單,可讀性更強。
這里總結(jié)了這些技巧。我會介紹如何使用linq組合查詢來:
◆初始化數(shù)組
◆在一個循環(huán)中遍歷多個數(shù)組
◆生成隨機序列
◆生成字符串
◆轉(zhuǎn)換序列或集合
◆把值轉(zhuǎn)換為長度為1的序列
◆遍歷序列的所有子集
如果你在linq組合查詢方面有心得也歡迎在評論中一起分享。
1. 初始化數(shù)組
通常,我們需要把數(shù)組的值初始化為相同的值或遞增的序列值,或者可能是一個步進不為1的遞增/遞減序列。有了linq組合查詢,我們可以在數(shù)組的初始化器中完成所有工作,不再需要循環(huán)!
在如下的示例代碼中,***行代碼初始化了一個長度為10的數(shù)組,所有元素都是-1,第二行代碼初始化b為0、1、2到9,第三行代碼初始化c為100、110、120到190。
- int[] a = enumerable.repeat(-1, 10).toarray();
- int[] b = enumerable.range(0, 10).toarray();
- int[] c = enumerable.range(0, 10).select(i => 100 + 10 * i).toarray();
要提醒一下:如果你初始化一個很大的數(shù)組,***不考慮這種優(yōu)雅的方式而是使用傳統(tǒng)的方式來替代。linq組合查詢的這種解決方案會動態(tài)產(chǎn)生數(shù)組,因此垃圾數(shù)組需要在運行時被回收。也就是說,我總是會在小數(shù)組或測試調(diào)試代碼的情況下使用這種技巧。
2. 在一個循環(huán)中遍歷多個數(shù)組
有個朋友問我一個C#的問題:有沒有辦法在一個循環(huán)中遍歷多個集合?他的代碼差不多是這樣:
- foreach (var x in array1) {
- dosomething(x);
- }
- foreach (var x in array2) {
- dosomething(x);
- }
這樣的話,循環(huán)主體會很大,而且他也不希望這樣重復(fù)的代碼。但是,他又不希望創(chuàng)建一個數(shù)組來保存array1和array2的所有元素。
linq組合查詢提供了一種優(yōu)雅的解決方案:concat操作。我們可以使用單個循環(huán)來重寫上面的代碼,如下:
- foreach (var x in array1.concat(array2)) {
- dosomething(x);
- }
注意,由于linq組合查詢在枚舉器級別進行操作,他不會產(chǎn)生新的數(shù)組來保存array1和array2的元素。因此,除了優(yōu)雅之外,這個方案還很高效。
3. 生成隨機序列
這是一個生成n長度隨機序列的簡單技巧:
- random rand = new random();
- var randomseq = enumerable.repeat(0, n).select(i => rand.next());
有了linq組合查詢的延遲特性,序列不會實現(xiàn)進行計算并保存到數(shù)組中,而是在迭代randomseq的時候按需生成隨機數(shù)。
4. 生成字符串
linq組合查詢同樣也是生成各種類型字符串的好工具。對于測試或調(diào)試,生成字符串時很有用的。假設(shè)我們需要生成一個n長度的字符串,按照“abcabcabc”的方式。使用linq組合查詢,解決方案非常優(yōu)雅:
- string str = new string(
- enumerable.range(0, n)
- .select(i => (char)(‘a(chǎn)’ + i % 3))
- .toarray());
petar petrov給出了另外一種有趣的方式使用linq組合查詢來生成字符串:
- string values = string.join(string.empty,
- enumerable.repeat(pattern, n).toarray());
#p#
5. 轉(zhuǎn)換序列或集合
在c#或vb中我們不能實現(xiàn)把序列從t類型轉(zhuǎn)換為u類型,即使t從u類繼承。因此,即使把list轉(zhuǎn)換為list,如果我們需要轉(zhuǎn)換list為list,linq組合查詢也提供了解決方案,但是它會進行列表的復(fù)制:
- list strlist = …;
- listobjlist = new list(strlist.cast());
chris cavanagh建議另外一種解決方式:
- var objlist = strlist.cast().tolist();
6. 把值轉(zhuǎn)換為長度為1的序列
當(dāng)我們需要把單個值轉(zhuǎn)化為一個長度為1的序列時,會怎么做?我們可以創(chuàng)建一個長度為1的數(shù)組,但是我還是喜歡linq組合查詢的repeat操作:
- ienumerable seq = enumerable.repeat(myvalue, 1);
7. 遍歷序列的所有子集
有的時候,遍歷數(shù)組的所有子集很有用。子集和問題、布爾可滿足性問題以及背包問題都可以通過遍歷某個序列的所有子集來簡單解決。
有了linq組合查詢,我們可以如下聲場所有arr數(shù)組的子集:
- t[] arr = ...;
- var subsets = from m in enumerable.range(0, 1 << arr.length)
- select
- from i in enumerable.range(0, arr.length)
- where (m & (1 << i)) != 0
- select arr[i];
注意,如果子集的個數(shù)超過了int,上面的代碼就不能工作。因此,僅當(dāng)你知道arr的長度不超過30的時候才去使用這個方式。如果arr長度超過30,你應(yīng)該不會是想去遍歷所有的子集,因為可能這會耗費幾分鐘或更長的時間。
評論和總結(jié)
希望這些技巧對你有用,這些示例代碼都使用c#實現(xiàn),但是你可以很容易得改變?yōu)槠渌?net語言。然而,linq組合查詢對于支持?jǐn)U展方法、lambda表達式和類型推斷的語言更方便,比如c#和vb。這里的每一段代碼都可行,但是我不能保證什么,請在使用前仔細(xì)檢查。
【編輯推薦】