inqToDB框架最大的優(yōu)勢(shì)應(yīng)該是實(shí)現(xiàn)了對(duì)Linq的支持。如果少了這一個(gè)功能相信他在使用上的快感會(huì)少了一個(gè)層次。本來(lái)筆者想要直接講解LinqToDB框架是如何實(shí)現(xiàn)對(duì)Linq的支持。寫(xiě)到一半的時(shí)候卻發(fā)現(xiàn)本系列在內(nèi)容上的引導(dǎo)顯得格外的生硬。思考在三最后還是決定在講解LinqToDB框架之前來(lái)一章過(guò)度文。
Linq查詢的原理
我們?cè)趯W(xué)習(xí)Linq的時(shí)候會(huì)見(jiàn)到一些很常見(jiàn)的關(guān)鍵詞語(yǔ)。比如Linq To SQL、Linq To Objects、Linq To XML等。事實(shí)這些一般都是根據(jù)不同的數(shù)據(jù)源來(lái)進(jìn)行命名的。 說(shuō)實(shí)話筆者當(dāng)初學(xué)習(xí)的時(shí)候,看到這些命名險(xiǎn)些以為只有這幾種。事實(shí)不是這樣子的。Linq有倆個(gè)核心類(lèi)——Enumerable類(lèi)和Queryable類(lèi)。這倆個(gè)類(lèi)可以說(shuō)貫穿整個(gè)Linq知識(shí)體系。如果有心的朋友可以點(diǎn)開(kāi)對(duì)應(yīng)的dll包就是發(fā)現(xiàn)他們都在System.Linq命名空間下。同時(shí)他們都是用于擴(kuò)展相應(yīng)的靜態(tài)方法。而且方法名大至相同。然后他們卻在本質(zhì)上有著細(xì)微的差別。Enumerable類(lèi)是對(duì)IEnumerable<T>接口進(jìn)行擴(kuò)展并且傳入了Func類(lèi)型的參數(shù)。數(shù)據(jù)源是來(lái)自于內(nèi)存中的。而Queryable類(lèi)是對(duì)IQueryable<T>接口進(jìn)行擴(kuò)展,傳入?yún)?shù)是表達(dá)式(Expression類(lèi)型)。數(shù)據(jù)源是來(lái)自于第三方。比如SQL Server、MySql等。
Linq的思想就是提供一個(gè)統(tǒng)一模型操作來(lái)處理數(shù)據(jù)。所以本質(zhì)來(lái)講對(duì)數(shù)據(jù)源不是很講究。比如數(shù)據(jù)源是文件,或則說(shuō)數(shù)據(jù)源是Excel之類(lèi)的。相信可能有人已經(jīng)看到過(guò)Linq To Excel呢?主要辛苦還是這些開(kāi)發(fā)底層的人。對(duì)于使用者來(lái)講沒(méi)有什么多大的差別。Linq現(xiàn)在面對(duì)數(shù)據(jù)源而擴(kuò)展功能有很多。其中專(zhuān)對(duì)數(shù)據(jù)庫(kù)來(lái)講,最流行還是有Linq To SQL。而且擴(kuò)展數(shù)據(jù)庫(kù)的Linq功能大多數(shù)都用IQueryable<T>接口。當(dāng)然,這不是說(shuō)用IEnumerable<T>接口就不行了。只是這倆種接口在實(shí)現(xiàn)上有著很大的差別。IEnumerable<T>接口我們都知道他一般是專(zhuān)對(duì)于內(nèi)存的。這意味著我們必須把相應(yīng)的數(shù)據(jù)全部加載到內(nèi)存中才可以進(jìn)行查詢。這樣子的操作太傷性能了。而IQueryable<T>接口我們可以巧妙的用上表達(dá)式樹(shù)(Expression Tree)進(jìn)行轉(zhuǎn)化生成對(duì)應(yīng)的數(shù)據(jù)庫(kù)SQL語(yǔ)句,然后在執(zhí)行數(shù)據(jù)庫(kù)。這才是顯得合理。
ORM思想能流行大體上可以說(shuō)是因?yàn)樗乃枷敫淤N切于人類(lèi)的思維方式。在筆者看來(lái)如果把Linq技術(shù)說(shuō)成也是ORM思想的產(chǎn)物之一,這樣子的說(shuō)法也不為過(guò)。這也是筆者喜歡Linq的地方。LinqToDB框架只所以都能支持Linq。不可否認(rèn)也是依據(jù)這一種上面所講的原理來(lái)實(shí)現(xiàn)的。大體的想法如下。
實(shí)現(xiàn)Linq提供的IQueryable<T>接口和IQueryProvider接口。生成相關(guān)的表達(dá)式樹(shù)。
把對(duì)應(yīng)的表達(dá)式樹(shù)轉(zhuǎn)化生成對(duì)應(yīng)數(shù)據(jù)庫(kù)的SQL語(yǔ)句。并執(zhí)行。
根據(jù)映射的信息,生成對(duì)應(yīng)的集合類(lèi)。(這里的集合類(lèi)是指SQL語(yǔ)句執(zhí)行結(jié)果轉(zhuǎn)成類(lèi)放入的集合)
實(shí)現(xiàn)自定義的Linq查詢一定離不開(kāi)倆類(lèi)——IQueryable<T>接口和IQueryProvider接口。上面的工作可以說(shuō)都在這倆類(lèi)上面。IQueryable<T>接口一般用于生成對(duì)應(yīng)的表達(dá)式樹(shù)。而IQueryProvider接口用于執(zhí)行表達(dá)式樹(shù),轉(zhuǎn)化成對(duì)應(yīng)的SQL語(yǔ)句,執(zhí)行數(shù)據(jù)庫(kù)并生成映射的模型對(duì)象。
注意:IQueryable<T>接口是什么樣子生成相關(guān)的表達(dá)式樹(shù)呢?讓筆者來(lái)講的話,筆者覺(jué)得有一點(diǎn)浪費(fèi)時(shí)間。但是不要當(dāng)心博客園里面有一位大神寫(xiě)的博文一定能滿足你——王清培的《.NET深入解析LINQ框架》。
實(shí)現(xiàn)Linq查詢
支持Linq查詢本意上就是實(shí)現(xiàn)上面所講的倆個(gè)接口。當(dāng)你實(shí)現(xiàn)IQuer