對EF開發來說,導航屬性肯定都用過,事實上,它是由VS IDE工具根據你的數據庫關系結構自動生成的外鍵屬性,在類視圖中可以看到相關屬性,它是以外鍵表名來標識的,如果是一對多的關系,那么,它會為屬性加上ICollection泛型集合用來標識,而今天我們要說的當然不是自動生成的,而是手動加的屬性,這樣屬性需要我們手動進行join,然后把它按需賦值,而使用include當然是無效的,呵呵。
一般地,我們習慣上把集合屬性定義為List<T>,但是,對於linq to entities來說,這個東西並不是很受歡迎,而標准結果集大家都知道是IEnumerable<T>,它是所有集合的基類,自身只提供了集合遍歷的方法,這也是我們不用它作為導航屬性的原因,因為我們一般需要為導航集合賦值的,而使用IEnumerable賦值就比較麻煩,需要借助List等集合。
引入IEnumerable的原因主要是List無法實現一個復雜的查詢,如圖:
這個查詢返回一個派生類型,下面的復雜查詢將會用到上面的結果,而這時,List類型的導航屬性將是不被允許的
我們通過監視器可以看到,查詢返回的默認是IEnumerable,所以,我們要屬性改為IEnumerable,結果當然是正常的
而我們之前不用IEnumerable的原因,就是因為它的方法太單調了,沒有添加,移除,查找等方法,而這時,不用它又不行,所以,只能把它進行擴展了,呵呵
擴展方法如下:
/// <summary> /// IEnumerable接口的擴展方法,支持它的實現類是List的情況 /// </summary> public static class IEnumerableExtensions { /// <summary> /// 向集合中添加元素 /// </summary> /// <typeparam name="T"></typeparam> /// <param name="collection"></param> /// <param name="value"></param> public static void Add<T>(this IEnumerable<T> collection, T value) { (collection as List<T>).Add(value); } /// <summary> /// 從集合中刪除元素 /// </summary> /// <typeparam name="T"></typeparam> /// <param name="collection"></param> /// <param name="value"></param> public static void Remove<T>(this IEnumerable<T> collection, T value) { (collection as List<T>).Remove(value); } /// <summary> /// 檢索集合中是否包含某個元素 /// </summary> /// <typeparam name="T"></typeparam> /// <param name="collection"></param> /// <param name="value"></param> /// <returns></returns> public static bool Contains<T>(this IEnumerable<T> collection, T value) { return (collection as List<T>).Contains(value); } }
調用也是十分簡單,和List類型的一樣樣,呵呵,舒服!