ASP.NET MVC Patterns

2020-03-27 00:22发布

问题:

I am fairly new to MVC, but after playing with it (MVC 3/Razor), I am hooked.

I have a few questions:

1) What is the best, or most widely used pattern to develop MVC apps in? Repository, DDD, UOW?

2) I am using the Entity Framework 4, so could some please explain to me or point me to a good source that will explain the Repository Pattern w/EF4? Doesn't EF4 take place as the business layer and the data access layer? Does the Repository Pattern even provide a benefit?

3) Also, one last question, could someone explain the whole relationship between the Controller, the Model and the View? I get the basics, but maybe a little more in depth of the correct way to use it. View Models - Say I have a view that displays customer info, and one that edits it, should I have a view model and an edit model, or can the be passed around?

4) Examples??

Thanks for the help up front,
$("Sam")

** EDIT **

Am I on the right track here:

Public Class HomeController
    Inherits System.Web.Mvc.Controller

    Function Index(ByVal id As Integer) As ActionResult
        Return View(New HomeModel)
    End Function

    <HttpPost()> _
    Function Index(ByVal Model As HomeModel) As ActionResult
        Return View(Model)
    End Function

End Class

Public Class HomeModel
    Private _Repository As IRepository(Of Customer)

    Public Property Customer As Customer

    Public Sub New()

    End Sub

    Public Sub New(ByVal ID As Integer)
        _Repository = New CustomerRepository
        Customer = _Repository.GetByID(ID)
    End Sub

End Class

Public Interface IRepository(Of T)
    Function GetByID(ByVal ID As Integer) As T
    Sub Add(ByVal Entity As T)
    Sub Delete(ByVal Entity As T)

End Interface

Public Class CustomerRepository
    Implements IRepository(Of Customer)

    Public Sub Add(ByVal Entity As Customer) Implements IRepository(Of Customer).Add

    End Sub

    Public Sub Delete(ByVal Entity As Customer) Implements IRepository(Of Customer).Delete

    End Sub

    Public Function GetByID(ByVal ID As Integer) As Customer Implements IRepository(Of    Customer).GetByID
        Return New Customer With {.ID = ID, .FirstName = "Sam", .LastName = "Striano"}
    End Function
End Class

Public Class Customer
    Public Property ID As Integer
    Public Property FirstName As String
    Public Property LastName As String
End Class

回答1:

I use generic repositories that get instantiated in a service class (using Dependency Injection with Ninject).

The service class essentially performs two functions:

  1. It provides all the methods that the controller will consume.

  2. It has a property called ViewModel, that essentially maps the data that the views need into a MyViewModel class.

The Controller consumes the service class. With this "pattern", your controllers look like:

namespace ES.eLearningFE.Areas.Courses.Controllers
{
    public partial class CourseController : Controller
    {
        ICourseDisplayService service;
        public CourseController(ICourseDisplayService service)
        {
            this.service = service;
        }

        public virtual ActionResult Display(int CourseId, int StepOrder, string PupilName, string TutorName)
        {
            service.CourseId = CourseId;
            service.StepOrder = StepOrder;
            service.PupilName = PupilName;
            service.TutorName = TutorName;
            if (Request.IsAjaxRequest())
            {
                return PartialView(service.ViewModel);
            }
            else
            {
                return View(service.ViewModel);
            }
        }

    }
}

The ViewModel class only hold display data and no methods (except the odd really simple method to retrieve data from another property that is, for example a List<> object).

Works really well. An example of a service class:

namespace ES.eLearning.Domain.Services.Courses
{
    public class SqlCourseDisplayService : ICourseDisplayService
    {
        DataContext db;

        public SqlCourseDisplayService(DbDataContextFactory contextFactory)
        {
            db = contextFactory.Make();
            CoursesRepository      = new SqlRepository<Course>(db);
            StepsRepository        = new SqlRepository<CourseStep>(db);
            StepLinksRepository    = new SqlRepository<StepLink>(db);
            UserCoursesRepository  = new SqlRepository<UserCourse>(db);
            CourseTutorsRepository = new SqlRepository<CourseTutor>(db);
            UsersRepository        = new SqlRepository<User>(db);
        }
        #region ICourseDisplayService Members

        public ViewModels.CourseDisplayVM ViewModel
        {
            get
            {
                return new ViewModels.CourseDisplayVM
                {
                    CourseId = this.CourseId,
                    CourseName = this.Course.Name,
                    Steps = this.Steps,
                    ActiveStepIndex = this.ActiveStepIndex,
                    CurrentStepIndex = this.CurrentStepIndex,
                    Pupil = new UserDto { UserId = this.PupilId, UserName = this.PupilName },
                    Tutors = this.GetTutors(this.CourseId),
                    Tutor = tutorName == null ? null : new UserDto { UserName = this.TutorName, UserId = this.TutorId}
                };
            }
        }

        #region Entities

        int courseId;
        public int CourseId
        {
            get
            {
                if (courseId == 0) throw new ApplicationException("Invalid Course Id!");
                return courseId;
            }
            set
            {
                if (value == 0) throw new ApplicationException("Invalid Course Id!");
                try
                {
                    Course   = (from c in CoursesRepository.Query where c.CourseId == value select c).First();
                    Steps    = Course.CourseSteps.ToList();
                    courseId = value;
                }
                catch {throw new ApplicationException("No Course found for Course Id: " + value);}
            }
        }

        public Data.Course Course { get; private set; }

        public int StepOrder { get; set; }

        public List<Data.CourseStep> Steps { get; private set; }

        public int ActiveStepIndex
        {
            get
            {
                if (PupilName == null)
                {
                    throw new ApplicationException("Pupil not set!");
                }
                if (CourseId == 0)
                {
                    throw new ApplicationException("Course not set!");
                }
                try
                {
                    var x = (from uc in UserCoursesRepository.Query where (uc.IdCourse == CourseId) && (uc.UserName == PupilName) select uc).First();
                    return x.ActiveStepIndex;
                }
                catch { throw new ApplicationException("Could not get Active Step!"); }
            }
        }

        #endregion

        #region Users

        string tutorName;
        public string TutorName
        {
            get 
            {
                if (tutorName == null) throw new ApplicationException("Invalid call to get Tutor Name [Null Tutor Name]!");
                return tutorName; 
            }
            set
            {
                tutorName = value;
                TutorId = (Guid)Membership.GetUser(tutorName).ProviderUserKey;
            }
        }

        public Guid TutorId { get; set; }

        string pupilName;
        public string PupilName
        {
            get { return pupilName; }
            set
            {
                pupilName = value;
                PupilId = (Guid)Membership.GetUser(pupilName).ProviderUserKey;
            }
        }

        public Guid PupilId { get; set; }

        #endregion

        #region Utility Properties

        public int CurrentStepIndex { get; set; }
        public int StepCount
        {
            get
            {
                return Steps == null ? 0 : Steps.Count();
            }
        }

        #endregion

        #region Private Utilities

        private List<UserDto> GetTutors(int CourseId)
        {
            return (from ct in CourseTutorsRepository.Query join u in UsersRepository.Query
                    on ct.TutorName equals u.UserName
                    where (ct.CourseId == courseId) 
                    select new UserDto { UserName = ct.TutorName, UserId = u.UserId }).ToList();
        }

        #endregion

        #region Repositories

        private IRepository<Course> CoursesRepository
        {
            get;
            set;
        }

        private IRepository<CourseStep> StepsRepository
        {
            get;
            set;
        }

        private IRepository<StepLink> StepLinksRepository
        {
            get;
            set;
        }

        private IRepository<UserCourse> UserCoursesRepository
        {
            get;
            set;
        }

        private IRepository<CourseTutor> CourseTutorsRepository
        {
            get;
            set;
        }

        private IRepository<User> UsersRepository
        {
            get;
            set;
        }

        #endregion

        #endregion
    }
}

May not be everyone's choice, but hey, it works for me... AND (more importantly) my clients and their users.

Edit

As requested in the comment below, the Repository that I use:

namespace ES.eLearning.Domain
{
    public class SqlRepository<T> : IRepository<T> where T : class
    {
        DataContext db;
        public SqlRepository(DataContext db)
        {
            this.db = db;
        }

        #region IRepository<T> Members

        public IQueryable<T> Query
        {
            get { return db.GetTable<T>(); }
        }

        public List<T> FetchAll()
        {
            return Query.ToList();
        }

        public void Add(T entity)
        {
            db.GetTable<T>().InsertOnSubmit(entity);
        }

        public void Delete(T entity)
        {
            db.GetTable<T>().DeleteOnSubmit(entity);
        }

        public void Attach(T entity)
        {
            db.GetTable<T>().Attach(entity);
        }

        public void Save()
        {
            db.SubmitChanges();
        }

        #endregion
    }
}

And the IRepository Interface:

namespace Wingspan.Web.Mvc
{
    public interface IRepository<TEntity> where TEntity : class
    {
        List<TEntity> FetchAll();
        IQueryable<TEntity> Query {get;}
        void Add(TEntity entity);
        void Delete(TEntity entity);
        void Attach(TEntity entity);
        void Save();
    }
}


回答2:

This should help you getting started. There are a lot of tutorials and videos available; for example:

  • Understanding Models, Views and Controllers
  • The ASP.NET MVC 2.0 basics and excellent introduction by Scott Hanselman. Personally one of my favorite speakers.

And also at www.asp.net; there are a few tutorials/examples to help you getting started. For example the Music Store sample

Unfortunately, I'm not so familiar with EF4/Repository pattern. But here's a blogpost about this pattern.



回答3:

1) I would say that the repository pattern is the most widely used, then there is inversion of controll too.

2) I can't really point out the benefits with using a repository for entity framework other than that the controller should not know about how to acces data other then asking a repository. This makes it easy to switch it out sometime. You can also eager load the data to make sure that the view don't call the database in every iteration of a foreach, for example a collection of users to display data from a child entity. You can probly do this anyway, but I feel that the repository is the right place to do it.

3) I can't tell you about the concept in a more in depth way, but I can tell some about viewmodels. In my opinion you should only use viewmodels if there is anything more then one entity you want to send to the view, for example a list of countries. You can alo use a viewmodel to "flatten" out very complex objects.



回答4:

I would defiantly say the repository pattern is used a lot. This pattern can be used with Dependency Injection. Using Dependency Injection makes Unit Testing a breeze because you can snap different repositories to an abstract repoistory. Check out http://ninject.org/ for a simple to use Dependecy injector for .NET.

View Models should hold display data and transfer that data from the controller to the view. If you want to edit and display customer info, take a look at this