什么是Linq表達(dá)式题诵?什么是Lambda表達(dá)式洁仗?
如圖:
由此可見Linq表達(dá)式和Lambda表達(dá)式并沒有什么可比性。
那與Lambda表達(dá)式相關(guān)的整條語(yǔ)句稱作什么呢性锭?在微軟并沒有給出官方的命名赠潦,在《深入理解C#》中稱為點(diǎn)標(biāo)記。
查詢表達(dá)式草冈、點(diǎn)標(biāo)記你更喜歡哪個(gè)祭椰?
所以臭家,我們的標(biāo)題的提問根本就不合適。應(yīng)該是“查詢表達(dá)式和點(diǎn)標(biāo)記你更喜歡哪個(gè)方淤?”钉赁。如:
//查詢表達(dá)式
var students1 = from t in db.Students
where t.Name == "張三"
select new { t.Id, t.Name, t.Age };
//點(diǎn)標(biāo)記
var students2 = db.Students
.Where(t => t.Name == "張三")
.Select(t => new { t.Id, t.Name, t.Age });
為什么選擇點(diǎn)標(biāo)記
我相信更多的人偏向選擇點(diǎn)標(biāo)記。具體什么原因我也說不清(可能是點(diǎn)標(biāo)記中的Lambda更加優(yōu)雅吧)携茂。對(duì)于我個(gè)人來說你踩,也是更加喜歡點(diǎn)標(biāo)記這種方式。
1讳苦、所有的查詢表達(dá)式都可以轉(zhuǎn)成對(duì)應(yīng)的點(diǎn)標(biāo)記带膜。反之,不是所有的點(diǎn)標(biāo)記都可以轉(zhuǎn)成查詢表達(dá)式鸳谜。
為什么膝藕?因?yàn)椴樵儽磉_(dá)式在編譯后就直接變成了點(diǎn)標(biāo)記:(以下是上面兩個(gè)語(yǔ)句對(duì)應(yīng)的編譯后的反編譯C#代碼)
生成了一模一樣的代碼。(由于是編譯后的咐扭,好多亂七八糟的代碼芭挽。我們只看Where和Select關(guān)鍵字就知道,使用的都是點(diǎn)標(biāo)記蝗肪。)
2袜爪、點(diǎn)標(biāo)記確實(shí)比查詢表達(dá)式更加優(yōu)雅
例一:
//查詢表達(dá)式
var students1 = from t in db.Students
where t.Name == "張三"
select t;
//點(diǎn)標(biāo)記
var students2 = db.Students
.Where(t => t.Name == "張三");
我為什么一定要 select t 啊,這句沒卵用的廢話就不能省嗎薛闪?是的辛馆,省了就報(bào)錯(cuò):
例二:
必須需要括號(hào)包裹起來才能取結(jié)果集?你還能更丑一點(diǎn)嗎豁延?
//查詢表達(dá)式
var students1 = (from t in db.Students
where t.Name == "張三"
select t).ToList();
//點(diǎn)標(biāo)記
var students2 = db.Students
.Where(t => t.Name == "張三")
.ToList();
例三:(為什么說:"不是所有的點(diǎn)標(biāo)記都可以轉(zhuǎn)成查詢表達(dá)式"【此例只適用于IEnumerator】)
此條點(diǎn)標(biāo)記你能轉(zhuǎn)成查詢表達(dá)式嗎昙篙?
var list = new List<string>() { "張三", "張三", "張三", "張三", "李四", "張三", "李四", "張三", "李四" };
var students2 = list
.Where((item, index) => item == "張三" && index % 2 == 0)
.Select((item, index) => new { item, index })
.ToList();
查詢表達(dá)式你能Reverse嗎?
var list = new List<string>() { "張三1", "張三2", "張三3", "張三0", "李四9", "張三3", "李四", "張三2", "李四" };
var students2 = list
.Where((item, index) => item.Contains("張三"))
.Select((item, index) => new { item, index })
.Reverse()//反序
.ToList();
ListA.Distinct().ToList();//去重
ListA.Except(ListB).ToList();//差集
ListA.Union(ListB).ToList(); //并集
ListA.Intersect(ListB).ToList();//交集
什么時(shí)候使用查詢表達(dá)式诱咏?
通過上面的對(duì)比瓢对,好像查詢表達(dá)式一文不值了。no胰苏,不是這樣的。
比如下面幾種情況我們就可以選擇使用查詢表達(dá)式:
例一:(本例適用于Linq to Object 和 沒有建主外鍵的EF查詢)
點(diǎn)標(biāo)記中的Join需要傳四個(gè)參數(shù)表達(dá)式,是不是有點(diǎn)暈了。而克。功炮。
var list1 = new Dictionary<string, string> { { "1", "張三" }, { "2", "李四" }, { "3", "張三" }, { "4", "張三" } };
var list2 = new Dictionary<string, string> { { "1", "張三" }, { "2", "李四" }, { "3", "李四" }, { "4", "張三" } };
//查詢表達(dá)式
var obj1 = from l1 in list1
join l2 in list2
on l1.Key equals l2.Key
select new { l1, l2 };
//點(diǎn)標(biāo)記
var obj = list1.Join(list2, l1 => l1.Key, l2 => l2.Key, (l1, l2) => new { l1, l2 });
例二:
點(diǎn)標(biāo)記需要區(qū)分OrderBy、ThenBy有沒有覺得麻煩
//查詢表達(dá)式
var obj1 = from l1 in list1
join l2 in list2
on l1.Key equals l2.Key
orderby l1.Key, l2.Key descending
select new { l1, l2 };
//點(diǎn)標(biāo)記
var obj = list1.Join(list2, l1 => l1.Key, l2 => l2.Key, (l1, l2) => new { l1, l2 })
.OrderBy(li => li.l1.Key)
.ThenByDescending(li => li.l2.Key)
.Select(t => new { t.l1, t.l2 });
總覺得查詢表達(dá)式更多的只是為了照顧那些寫慣了sql的程序員氏堤。
聯(lián)接查詢(內(nèi)聯(lián)、左聯(lián)、交叉聯(lián))
關(guān)于聯(lián)接查詢使用查詢表達(dá)式會(huì)更合適一些這個(gè)上面已經(jīng)說了陕赃。
接下來我們寫內(nèi)聯(lián)卵蛉、左聯(lián)、交叉聯(lián)的查詢表達(dá)式和對(duì)應(yīng)的點(diǎn)標(biāo)記代碼么库。(目的:可能有些人不會(huì)傻丝,同時(shí)在這里也給自己做個(gè)備忘)
內(nèi)聯(lián):
左聯(lián):
交叉聯(lián):
其實(shí)關(guān)于聯(lián)接查詢,如果EF建好了主外鍵我還是覺得點(diǎn)標(biāo)記用起來更爽爽的诉儒。
結(jié)束:
本文并不是要改變你的習(xí)慣葡缰,也不是否定你的觀點(diǎn)。僅僅只是表達(dá)個(gè)人對(duì)點(diǎn)標(biāo)記和查詢表達(dá)式的些許理解忱反。
關(guān)于是使用查詢表達(dá)式還是點(diǎn)標(biāo)記泛释,可能起著更大決定性的作用的是團(tuán)隊(duì)共同的習(xí)慣和規(guī)范。
然后還想說說温算,只要我們對(duì)比什么怜校,很可能就會(huì)有人跳出了,什么不要比注竿,用好了都一樣茄茁,什么什么才是最重要的,什么什么的蔓搞。胰丁。。
就像很多人會(huì)反感java和C#的對(duì)比喂分,其實(shí)我個(gè)人覺得對(duì)比下底層實(shí)現(xiàn)锦庸、對(duì)比下語(yǔ)法簡(jiǎn)易也不是不可以的,只要我們可以從中學(xué)到知識(shí)(個(gè)人也是不喜歡對(duì)比 誰誰誰學(xué)什么工資多少多少)蒲祈。
昨天的自己對(duì)比今天的自己甘萧,今天的自己對(duì)比明天的自己。只要可以進(jìn)步為什么不要對(duì)比呢梆掸?