2012-05-10 58 views
2

爲了儘快幹起我的代碼,我想要做以下事情。 (我不知道它是否是最好的方法,但是如果我不斷需要更新多個方法,如果我想在整個站點上更改某些內容,它似乎更好地擁有不斷增加的代碼庫)繼承/泛型基礎知識。如何實現代碼結構DRY

我知道什麼關於繼承是可怕的。因爲Iv'e從來沒有質疑我使用過的任何代碼/庫,而且我以前從未真正嘗試過寫過類似的東西,但我想學習......希望這將成爲我開悟的日子:P

我的問題:

說Iv'e有一個add方法(在我的所有控制器)是這樣的:

public ActionResult Add(VM_Down_Time_Capture viewModel) 
     { 
      using (Down_Time_CaptureRepository repository = new Down_Time_CaptureRepository()) 
      { 
       if (!ModelState.IsValid) 
        return ReturnValidationFailure(ViewData.ModelState.Values); 

       Down_Time_Capture model = new Down_Time_Capture(); 
       model.InjectFrom(viewModel); 

       string mserMsg = repository.Add(model, User.Identity.Name); 

       if (!string.IsNullOrEmpty(mserMsg)) 
        return ReturnCustomValidationFailure(Server.HtmlEncode(mserMsg)); 

       repository.Save(); 

       return Json("Added successfully.", JsonRequestBehavior.AllowGet); 
      } 
     } 

而此刻我有以下爲好。

由T4 Templates/EF生成。

ViewModels, Repositories, (Standard) EF Models 

我想我需要爲每個頁面(可使用T4來完成),從包含基本的CRUD功能的自定義ControllerBase類繼承的ModelSpecfic基本控制器。這樣我可以有每個控制器自定義代碼和我的代碼基地將清潔&小&是不會得到影響,我應該需要重新生成基本文件

我不太明白如何實現的東西線我需要的。到目前爲止,我的理解是生病需要我的存儲庫和視圖模型從基礎繼承,並以某種方式在[B]中指定我正在使用的那些......但至於如何做到這一點,我不需要知道

例如(這是我最好的嘗試,不是我的實際代碼,非常哈克因爲我驚人的困惑:S)

public class Down_Time_CaptureController : Down_Time_CaptureBase 
    { 
    //[A] 
    } 

    //Generated by T4 
    public class Down_Time_CaptureBase: ControllerBase 
    { 
     //[B] 
     public override EntityObject CreateNewModel() 
     { 
      return new Down_Time_Capture(); 
     } 

     public override Base_Repository CreateNewRepository() 
     { 
      return new Down_Time_CaptureRepository(); 
     } 

     public override Base_ViewModel CreateNewViewModel() 
     { 
      return new VM_Down_Time_Capture(); 
     } 

     //how would i go about specifying which repository & model & view model to use 
     //although i expect it to be something to what i did here above 
     //and how would i go about calling the new generic add method (but in context of this controller)? 

    } 

    //coded once 
    public abstract class ControllerBase: Controller 
    { 
     //[C] 
     //make abstract so i have to override it 
     public abstract Base_Controller CreateNewModel(); 
     public abstract Base_Controller CreateNewRepository(); 
     public abstract Base_Controller CreateNewViewModel(); 

     //I'm assuming my generified add method would go in here 
     public virtual ActionResult Add(Base_ViewModel viewModel) 
     { 
      using (Base_Repository repository = CreateRepository()) 
      { 
        if (!ModelState.IsValid) 
         return ReturnValidationFailure(ViewData.ModelState.Values); 

        EntityObject model = CreateNewModel(); 
        model.InjectFrom(viewModel); 

        string mserMsg = repository.Add(model, User.Identity.Name); 

        if (!string.IsNullOrEmpty(mserMsg)) 
         return ReturnCustomValidationFailure(Server.HtmlEncode(mserMsg)); 

        repository.Save(); 

        return Json("Added successfully.", JsonRequestBehavior.AllowGet); 
      }  
     } 
    } 
+0

+1使用ValueInjecter! – MattDavey

回答

3

這裏是你問一個簡單的通用解釋爲:

// concrete controller implementation 
public class Down_Time_CaptureController: ControllerBase<Down_Time_Capture, VM_Down_Time_Capture, Down_Time_CaptureRepository> 
{ 
} 

// generic controller base 
public abstract class ControllerBase<TModel, TViewModel, TRepository>: Controller 
     where TModel : Base_Model, new() 
     where TViewModel : Base_ViewModel, new() 
     where TRepository : Base_Repository, new() 
{ 
    protected virtual TModel CreateNewModel() 
    { 
      return (TModel)Activator.CreateInstance<TModel>(); 

    } 

    protected virtual TRepository CreateNewRepository() 
    { 
      return (TRepository)Activator.CreateInstance<TRepository>(); 
    } 

    protected virtual TViewModel CreateNewViewModel() 
    { 
      return (TViewModel)Activator.CreateInstance<TViewModel>(); 
    } 

    //I'm assuming my generified add method would go in here 
    public virtual ActionResult Add(TViewModel viewModel) 
    { 
     using (var repository = CreateRepository()) 
     { 
       if (!ModelState.IsValid) 
        return ReturnValidationFailure(ViewData.ModelState.Values); 

       var model = CreateNewModel(); 
       model.InjectFrom(viewModel); 

       string mserMsg = repository.Add(model, User.Identity.Name); 

       if (!string.IsNullOrEmpty(mserMsg)) 
        return ReturnCustomValidationFailure(Server.HtmlEncode(mserMsg)); 

       repository.Save(); 

       return Json("Added successfully.", JsonRequestBehavior.AllowGet); 
     }  
    } 
} 

的幾個注意事項:

  1. 您可能需要爲三種類型(Model,ViewModel,Repository)創建接口並將其用作通用約束。
  2. 您可能需要一個通用的Repository接口和基本實現(因此您不必獨立編碼每個存儲庫,並將相似的邏輯從一個存儲到另一個)。
  3. 考慮使用Inversion of Control container and dependency injection。例如,不是讓控制器處理創建存儲庫的實例,而是使其成爲屬性並從構造函數中進行設置。然後,您可以使用您選擇的IoC(如Ninject或Autofac)並註冊具體的實現,它將管理創建以及依賴和控制器本身的創建和生命週期。
+0

你我的朋友是紳士和學者。謝謝...現在讓編碼開始。 –