EF6CodeFirst+MVC5+Autofac泛型註冊 入門實例

来源:https://www.cnblogs.com/Yennefer716/archive/2018/03/09/CodeFirstAutofacMvc5.html
-Advertisement-
Play Games

貼一個EF6 CodeFirst模式結合MVC5和Autofac(泛型註冊)的一個入門實例 網上類似的例子實在太少,最近自己也有用到這一塊的知識,總結了一下,不要讓後人踩了自己踩過的坑。 1:新建三個項目,Web(MVC)、EntityFramework類庫(EF框架)、Core類庫(核心框架),n ...


貼一個EF6 CodeFirst模式結合MVC5和Autofac(泛型註冊)的一個入門實例

 

網上類似的例子實在太少,最近自己也有用到這一塊的知識,總結了一下,不要讓後人踩了自己踩過的坑。

 

1:新建三個項目,Web(MVC)、EntityFramework類庫(EF框架)、Core類庫(核心框架),nuget EntityFramework,Autofac,AutofacMvc5

 

2:建立簡單對象:書籍(Book)Model,繼承主鍵為Int的基類

①:介面

namespace:Core.Domain.Interface

 1 /// <summary>
 2     /// 實體對象介面
 3     /// </summary>
 4     public interface IEntity<PrimaryKeyType>
 5     {
 6         /// <summary>
 7         /// 主鍵
 8         /// </summary>
 9         PrimaryKeyType Id { get; }
10     }
IEntity

②:基類

namespace:Core.Domain.Base

 1     /// <summary>
 2     /// Int抽象基類
 3     /// </summary>
 4     /// <typeparam name="Type"></typeparam>
 5     public abstract class EntityBase : Interface.IEntity<int>
 6     {
 7         #region Properties
 8 
 9         /// <summary>
10         ///  數據狀態
11         /// </summary>
12         public Enum.EntityEnumType Status { get; set; }
13 
14         public DateTime CreateDate { get; set; }
15 
16         public DateTime UpdateDate { get; set; }
17 
18         /// <summary>
19         /// 主鍵
20         /// </summary>
21         [Key]
22         public int Id { get; set; }
23         #endregion
24 
25         #region Constructor
26         public EntityBase()     :this(Enum.EntityEnumType.Normal,DateTime.Now,DateTime.Now)
27         { }
28 
29         /// <summary>
30         /// 帶參初始化
31         /// </summary>
32         public EntityBase(Enum.EntityEnumType _status,DateTime _createDate,DateTime _updateDate)
33         {
34             Status = _status;
35             CreateDate = _createDate;
36             UpdateDate = _updateDate;
37         }
38         #endregion
39 
40         #region Methods
41 
42         #endregion
43 
44     }
EntityBase

③:書籍類

namespace:Web.Models

 1     [Table("Book")]//表名  
 2     /// <summary>
 3     /// 實體—書籍
 4     /// </summary>
 5     public class Book:EntityBase
 6     {
 7         #region Constructor
 8         public Book()
 9             : base()
10         { }
11         #endregion
12 
13         #region Fileds
14         /// <summary>
15         /// 書籍編號
16         /// </summary>
17         [DisplayName("書籍編號")]
18         public int BookCode { get; set; }
19 
20         /// <summary>
21         /// 書名
22         /// </summary>
23         [DisplayName("書名")]
24         [MaxLength(50,ErrorMessage = "書名過長"),MinLength(2,ErrorMessage = "書名過短")]
25         public string BookName { get; set; }
26 
27         /// <summary>
28         /// 作者
29         /// </summary>
30         [MaxLength(50,ErrorMessage = "作者姓名過長")]
31         [DisplayName("作者")]
32         public string Author { get; set; }
33 
34         #endregion
35 
36         #region 導航屬性
37 
38         #endregion
39 
40 
41         #region IEntity成員
42         #endregion
43     }
Book

解釋一下類的配置

CodeFirst 利用一種被稱為約定Conventions)優於配置Configuration)的編程模式允許你使用自己的 對象 來表示 EF 所依賴的模型去執行查詢、更改追蹤、以及更新功能

簡單的說,你創建的對象必須遵循EF給你指定的規則,比如最後兩位以Id為結尾就是主鍵啊什麼什麼什麼,反正我是沒遵循。

如果你不想遵循EF的規則,那麼——

Code First 提供了兩種方式來配置你的類:

  • DataAnnotations, 使用簡單屬性;
  • Fluent API, 以編程命令行式的方式來描述你的配置

有關於配置的文章,請參考

EF官方文檔:https://msdn.microsoft.com/en-us/library/ee712907(v=vs.113).aspx

 另外,有問題記得F1看文檔,不要一直百度

3:倉儲Repository設計

①:IRepository

namespace:Core.Repository

 1  public interface IRepository<TEntity> where TEntity:class
 2     {
 3         /// <summary>
 4         /// 添加一個對象
 5         /// </summary>
 6         /// <param name="item"></param>
 7         void Insert(TEntity item);
 8 
 9         /// <summary>
10         /// 刪除一個對象
11         /// </summary>
12         void Delete(TEntity item);
13 
14         /// <summary>
15         /// 更新一個對象
16         /// </summary>
17         void Update(TEntity item);
18 
19         /// <summary>
20         /// 通過主鍵取對象
21         /// </summary>
22         /// <returns></returns>
23         TEntity Find(params object[] id);
24 
25         /// <summary>
26         /// 拿到可查詢結果集
27         /// </summary>
28         /// <returns></returns>
29         System.Linq.IQueryable<TEntity> GetModel();
30 
31         /// <summary>
32         /// 設置數據上下文
33         /// </summary>
34         /// <param name="db"></param>
35         void SetDataContextByResloveName(string dbContext);
36     }
IRepository

SetDataContextByResloveName這個方法,到下文的Autofac配置時再說

②:IExtentionRepository 擴展

namespace:Core.IExtentionRepository

 1     /// <summary>
 2     /// 擴展倉儲
 3     /// </summary>
 4     /// <typeparam name="TEntity"></typeparam>
 5     public interface IExtentionRepository<TEntity>:IRepository<TEntity> where TEntity:class
 6     {
 7         #region 行為
 8         /// <summary>
 9         /// 添加集合
10         /// </summary>
11         /// <param name="item"></param>
12         void Insert(IEnumerable<TEntity> item);
13         /// <summary>
14         /// 修改集合
15         /// </summary>
16         /// <param name="item"></param>
17         void Update(IEnumerable<TEntity> item);
18         /// <summary>
19         /// 刪除集合
20         /// </summary>
21         /// <param name="item"></param>
22         void Delete(IEnumerable<TEntity> item);
23         #endregion
24 
25         #region 查詢
26         /// <summary>
27         /// 根據指定lambda表達式,得到結果集
28         /// </summary>
29         /// <param name="predicate"></param>
30         /// <returns></returns>
31         IQueryable<TEntity> GetModel(Expression<Func<TEntity, bool>> predicate);
32 
33         /// <summary>
34         /// 根據指定lambda表達式,得到第一個實體
35         /// </summary>
36         /// <param name="predicate"></param>
37         /// <returns></returns>
38         TEntity Find(Expression<Func<TEntity, bool>> predicate);
39         #endregion
40     }
IExtentionRepository

③:EFRepository EF倉儲

 namespace:EntityFramework

  1 public class EFRepository<TEntity> : 
  2         IRepository<TEntity>,
  3         IExtentionRepository<TEntity>
  4         where TEntity:class
  5     {
  6         #region Properties
  7         /// <summary>
  8         /// EF上下文
  9         /// </summary>
 10         private DbContext EFDbContext;
 11         #endregion
 12 
 13         #region Constructors
 14         public EFRepository()
 15             : this(null)
 16         { }
 17 
 18         /// <summary>
 19         /// 帶參構造函數
 20         /// </summary>
 21         public EFRepository(DbContext db)
 22         {            
 23             EFDbContext = db;
 24         }
 25         #endregion
 26 
 27         #region Methods
 28         /// <summary>
 29         /// 提交到資料庫
 30         /// </summary>
 31         public void SaveChanges()
 32         {
 33             try
 34             {
 35                 EFDbContext.SaveChanges();
 36             }
 37             catch (Exception ex)
 38             {
 39                 //保存日誌
 40                 //拋出異常
 41                 throw new MithrilCommonException(ex.Message, ex);
 42             }
 43         }
 44 
 45 
 46         #endregion
 47 
 48         #region IReposiotry<TEntity>
 49 
 50         /// <summary>
 51         /// 少量數據插入
 52         /// </summary>
 53         /// <param name="item"></param>
 54         public void Insert(TEntity item)
 55         {
 56             if (item != null)
 57             {
 58                 EFDbContext.Entry<TEntity>(item as TEntity);
 59                 EFDbContext.Set<TEntity>().Add(item as TEntity);
 60                 this.SaveChanges();
 61             }
 62         }
 63 
 64         /// <summary>
 65         /// 刪除一個實體
 66         /// </summary>
 67         /// <param name="id"></param>
 68         public void Delete(TEntity item)
 69         {
 70             if (item != null)
 71             {
 72                 //將實體以"UnChanged"的狀態放置到上下文中
 73                 EFDbContext.Set<TEntity>().Attach(item as TEntity);
 74                 //給定實體標記為“已刪除”
 75                 EFDbContext.Entry(item).State = EntityState.Deleted;
 76                 //EFDbContext.Set<TEntity>().Remove(item as TEntity);
 77                 this.SaveChanges();
 78             }
 79         }
 80 
 81         /// <summary>
 82         /// 更新一個實體
 83         /// </summary>
 84         /// <param name="item"></param>
 85         public void Update(TEntity item)
 86         {
 87             if(item != null)
 88             {
 89                 //將實體以"UnChanged"的狀態放置到上下文中
 90                 EFDbContext.Set<TEntity>().Attach(item as TEntity);
 91                 //更新
 92                 EFDbContext.Entry(item).State = EntityState.Modified;
 93                 this.SaveChanges();
 94             }
 95         }
 96 
 97         /// <summary>
 98         /// 根據id獲取實體
 99         /// </summary>
100         /// <param name="id"></param>
101         /// <returns></returns>
102         public TEntity Find(params object[] id)
103         {
104             return EFDbContext.Set<TEntity>().Find(id);
105         }
106 
107         /// <summary>
108         /// 拿到可查詢結果集
109         /// </summary>
110         /// <returns></returns>
111         System.Linq.IQueryable<TEntity> GetModel()
112         {
113             return null;
114         }
115 
116         //設置數據上下文
117         public void SetDataContextByResloveName(string contextName)
118         {
119             try
120             {
121                 EFDbContext = ServiceLocator.Instance.GetService<DbContext>(contextName);
122             }
123             catch (Exception)
124             {
125                 throw new ArgumentException("設置EFContext出錯");
126             }
127 
128         }
129 
130         /// <summary>
131         /// 拿到可查詢結果集
132         /// </summary>
133         /// <returns></returns>
134         IQueryable<TEntity> IRepository<TEntity>.GetModel()
135         {
136             throw new NotImplementedException();
137         }
138 
139 
140         #endregion
141 
142         #region IExtentionRepository
143         public void Insert(IEnumerable<TEntity> item)
144         {
145             foreach (var entity in item)
146             {
147                 EFDbContext.Entry<TEntity>(entity as TEntity);
148                 EFDbContext.Set<TEntity>().Add(entity as TEntity);
149             }
150             this.SaveChanges();
151         }
152 
153         public void Update(IEnumerable<TEntity> item)
154         {
155             #region 給每個對象改變狀態
156             foreach (var model in item)
157             {
158                 EFDbContext.Set<TEntity>().Attach(model as TEntity);
159                 EFDbContext.Entry(model).State = EntityState.Modified;
160                 this.SaveChanges();
161             }
162             #endregion
163         }
164 
165         public void Delete(IEnumerable<TEntity> item)
166         {
167             throw new NotImplementedException();
168         }
169 
170         public IQueryable<TEntity> GetModel(Expression<Func<TEntity, bool>> predicate)
171         {
172             throw new NotImplementedException();
173         }
174 
175         public TEntity Find(Expression<Func<TEntity, bool>> predicate)
176         {
177             throw new NotImplementedException();
178         }
179         #endregion
180     }
EFRepository

 其中拋出的Exception為自定義的異常類

4:CodeFirst遷移

vs17中:視圖 > 其他視窗 > 程式包管理控制台

預設項目欄中選擇相應的項目,

①:Enable-Migrations -ContextTypeName -Force

為你制定的上下文開始遷移工作,-ContextTypeName 為你的上下文名稱,-Force為可選項,是否覆蓋你所選的上下文的遷移

運行後,會生成Migration文件夾

②:Add-Migration -MigrationName -Force

當你新建資料庫,添加欄位、刪除、更新欄位時,輸入此命令,根據你DbContext實現類的策略以及之前對象的約束,CodeFirst會生成資料庫以及相應的表

③:Update-DataBase,將改動對應到實體庫上

貼一個測試DbContext類

 1 public class EFTestContext : DbContext
 2     {
 3 
 4         public EFTestContext()
 5             : base("name=EFTestContext")
 6         {
 7             //模型變更時更新資料庫
 8             Database.SetInitializer<EFTestContext>(new CreateDatabaseIfNotExists<EFTestContext>());
 9         }
10 
11         public DbSet<Book> Book { get; set; }
12 
13         protected override void OnModelCreating(DbModelBuilder modelBuilder)
14         {
15             modelBuilder.Entity<Book>().MapToStoredProcedures();
16             base.OnModelCreating(modelBuilder);
17         }
18     }
EFTestContext

5:MVC5 Autofac配置

 網上有很多例子,並不符合實際開發,這裡重點說一下一個介面有很多實現類的情況,應該如何配置Autofac

①:Global.asax配置

 1  protected void Application_Start()
 2         {
 3             AreaRegistration.RegisterAllAreas();
 4             FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
 5             RouteConfig.RegisterRoutes(RouteTable.Routes);
 6             BundleConfig.RegisterBundles(BundleTable.Bundles);
 7 
 8             IoCConfig();
 9         }
10 
11         /// <summary>
12         /// IoC註入
13         /// </summary>
14         private void IoCConfig()
15         {
16             var builder = new ContainerBuilder();
17             #region Repository註冊
18             builder.RegisterGeneric(typeof(EFRepository<>)).Named("EFRepository",typeof(IRepository<>)).InstancePerLifetimeScope();
19             #endregion
20 
21             #region Service註冊
22             builder.RegisterType(typeof(EFTestContext)).Named("EF", typeof(DbContext)).InstancePerLifetimeScope();
23             #endregion
24 
25             //註冊 Controller 
26             builder.RegisterControllers(typeof(MvcApplication).Assembly);
27             var container = builder.Build();
28             //MVC擴展
29             DependencyResolver.SetResolver(new AutofacDependencyResolver(container));
30         }
泛型配置

泛型配置看官方文檔

http://docs.autofac.org/en/latest/advanced/adapters-decorators.html

②:ServiceLocator,用於實現同一介面不同實現的Autofac Reslove

 1 public sealed class ServiceLocator
 2     {
 3         #region Constructor
 4         public ServiceLocator()
 5         {
 6             
 7         }
 8         #endregion
 9 
10         #region Singleton
11         private static readonly ServiceLocator _instance = new ServiceLocator();
12 
13         /// <summary>
14         /// 實例
15         /// </summary>
16         public static ServiceLocator Instance
17         {
18             get
19             {
20                 return _instance;
21             }
22         }
23         #endregion
24 
25         #region Public Methods
26         public TEntity GetService<TEntity>(string resloverName)
27         {
28             return AutofacDependencyResolver.Current.RequestLifetimeScope.ResolveNamed<TEntity>(resloverName);
29         }
30         #endregion
31     }
ServiceLocator

6:MVC Web測試

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 下麵還有幾句心得大家可以看一下:第一點:【盲目下註】全拼運氣,一把輸了下把倍投,如果連續5把不開,基本腦袋就一片空白了關於PK10下註技巧以及心態沒了清醒的頭腦,你看任何計劃,就算連中20期的計劃你都會慫了,這個時候你就是送財童子了。第二點:【不懂分配金額】沒耐心,心急,一兩把不中,一股腦全壓了,這 ...
  • 前面Insus.NET有在Angularjs實現DropDownList的下拉列表的功能。但是沒有實現怎樣獲取下拉列表的value和text功能。 下麵分別使用ng-click和ng-change來實現。先參考這篇《ASP.NET MVC下使用AngularJs語言(三):ng-options》ht ...
  • 該分頁控制項的顯示邏輯: 1 當前頁面反色突出顯示,鏈接不可點擊 2 第一頁時首頁鏈接不可點擊 3 最後一頁時尾頁鏈接不可點擊 4 當前頁面左右各顯示頁碼可以設置調節,如果左右一樣則居中 5 當左邊頁碼不足時,右側補充 6 當右側頁面不足時左側補充 7 總顯示頁碼數為左側+右側+1(當前) 組成部分: ...
  • 1. 安裝CentOs,可使用最小安裝包鏡像: 2. 跟隨安裝步驟進行,中途要選擇磁碟,如下圖: 3. 設置Root用戶密碼: 4. 安裝完畢後重啟,進入系統,CentOs最小版缺少一些必要組件,首先要打開網卡設置,用 編輯如下文件: 編輯最後一行,將 改為 保存並退出 5. 重啟網卡,然後Ping ...
  • 有沒有想過在.NET中已經有了事件機制,為什麼在WPF中不直接使用.NET事件要加入路由事件來取代事件呢?最直觀的原因就是典型的WPF應用程式使用很多元素關聯和組合起來,是否還記得在WPF自學入門(一)XAM基本知識中提到過兩棵樹,邏輯樹LogicalTree 和可視化樹 VisualTree,那麼 ...
  • 一、業務背景 微信自動退款串接基於酷客多小程式商城系統,為方便財務人員進行訂單退款而開發,將酷客多小程式系統財務退款流程和微信退款系統打通。實現一個系統管理運營。 二、業務流程設計 1.退款單狀態:待退款、退款中、退款完成、自動退款失敗等 2.由於微信申請退款介面接受請求後不會立即進行退款處理,微信 ...
  • ·你為什麼選擇軟體工程專業?你認為你的條件如何?軟工是你喜歡的領域嗎?是你擅長的領域嗎? 當初學軟體是想要做網路游戲的,雖然到了大二也沒能力做。但不得不說軟體工程是一門很好的學科,它不僅僅是大眾普遍認為的編程那麼簡單。 比較嚴謹的說法,軟體工程是研究或應用工程化方法來設計、創造、構建和維護有效、實用 ...
  • 原文: [Comparing AWS Lambda performance of Node.js, Python, Java, C# and Go](https://read.acloud.guru/comparing-aws-lambda-performance-of-node-js-python... ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...