In practice, Repository Pattern is generally used to separate the logic relating to the DB interaction (or any data source for that matter), and perform CRUD (Create, Read, Update, Delete) operations for the business objects or domain object or business entities, whatever you call them.
As per MSDN - The repository mediates between the data source layer and the business layers of the application. It queries the data source for the data, maps the data from the data source to a business entity, and persists changes in the business entity to the data source. A repository separates the business logic from the interactions with the underlying data source or Web service.
In early days, entity objects were heavier. They not only carries the properties but also the functions to perform operations on these properties.
So for example, the employee entity will have properties like Name, Department etc and then functions like, SaveEmployee, DeleteEmployee, FindEmployee etc.
As programming evolved, entities become lighter, now they just contain attributes and can be easily referenced throughout different layers. And operations on these entities moved to repositories!
Benefits:-In early days, entity objects were heavier. They not only carries the properties but also the functions to perform operations on these properties.
So for example, the employee entity will have properties like Name, Department etc and then functions like, SaveEmployee, DeleteEmployee, FindEmployee etc.
As programming evolved, entities become lighter, now they just contain attributes and can be easily referenced throughout different layers. And operations on these entities moved to repositories!
- A repository separates the business logic from the interactions with the underlying data source. This is especially important wherein the the data source may vary from dbms, webservice, sharepoint, feeds, files etc. With, Repository Pattern implemented, the business layer can easily be freed from the responsibility of knowing how the interactions to the data source are made. The respective business object repository will take care of it, making your business layer tidy.
- Repository Pattern makes it easy to unit test your code, with clearly defined responsibilities of different layers. You can easily test you application both with and without database (using mock/fake objects).
- It encourages good coding practice by its nature. With Repository Pattern implemented within the framework, it just becomes harder for the coder to break the rules and play around with the code. It is highly significant with the larger teams, big projects.
Ok...now let’s do some code works and see how this pattern works.
In repository pattern we have a repository class for every domain object, which will be responsible for data interactions with datasources. For example, if we have say, Person and Item domain entities, then there would be respective repositories for these entities.
A person like me with background of the good old 3-Tier architecture, when told to implement Repository Pattern, wrote something like this :(
public class PersonRepositoryA person like me with background of the good old 3-Tier architecture, when told to implement Repository Pattern, wrote something like this :(
{
public List<Person> GetAllPersons() { /*Implement method*/}
public Person GetPersonById(int personId) { /*Implement method*/}
public void CreatePerson(Person person) { /*Implement method*/}
public void UpdatePerson(Person person) { /*Implement method*/}
public void DeletePerson(Person person) { /*Implement method*/}
}
public class ItemRepository
{
public List<Item> GetAllItems() { /*Implement method*/}
public Item GetItemById(int itemId) { /*Implement method*/}
public void CreateItem(Item item) { /*Implement method*/}
public void UpdateItem(Item item) { /*Implement method*/}
public void DeleteItem(Item item) { /*Implement method*/}
}
Uuummnnn…., not bad, lets improvise this, and introduce Interfaces over the top of the repository classes above to add more flexibility to the design
Now the implementation would be like:
public interface IPersonRepository
{
public List<Person> GetAllPersons();
public Person GetPersonById(int personId);
public void CreatePerson(Person person);
public void UpdatePerson(Person person);
public void DeletePerson(Person person);
}
public interface IItemRepository
{
public List<Item> GetAllItems();
public Item GetItemById(int itemId);
public void CreateItem(Item item);
public void UpdateItem(Item item);
public void DeleteItem(Item item);
}
public class PersonRepository : IPersonRepository
{
public List<Person> GetAllPersons() { /*Implement method*/}
public Person GetPersonById(int personId) { /*Implement method*/}
public void CreatePerson(Person person) { /*Implement method*/}
public void UpdatePerson(Person person) { /*Implement method*/}
public void DeletePerson(Person person) { /*Implement method*/}
}
public class ItemRepository : IItemRepository
{
public List<Item> GetAllItems() { /*Implement method*/}
public Item GetItemById(int itemId) { /*Implement method*/}
public void CreateItem(Item item) { /*Implement method*/}
public void UpdateItem(Item item) { /*Implement method*/}
public void DeleteItem(Item item) { /*Implement method*/}
}
Doing this –
- Our code is no more dependent on the concrete class.
- We can now use patterns like dependency injection and provide different implementation as required.
- Automated testing will be very graceful; we can provide mock objects to the testing application, instead of messing up the actual database.
So far so good, but there is one problem, as the object increases, there will be more interfaces to manage. If we look closely, what we are doing are some generic operations for the business objects, and if we replace the interfaces with generic one, we can fix this up as well. So, the revised code will look something like:
public interface IRepository<T>
{
public List GetAll();
public T GetById(int id);
public void Create(T entity);
public void Update(T entity);
public void Delete(T entity);
}
public class PersonRepository : IRepository<Person>
{
public List<Person> GetAll() { /*Implement method*/}
public Person GetById(int id) { /*Implement method*/}
public void Create(Person entity) { /*Implement method*/}
public void Update(Person entity) { /*Implement method*/}
public void Delete(Person entity) { /*Implement method*/}
}
public class ItemRepository : IRepository<Item>
{
public List<Item> GetAll() { /*Implement method*/}
public Item GetById(int id) { /*Implement method*/}
public void Create(Item entity) { /*Implement method*/}
public void Update(Item entity) { /*Implement method*/}
public void Delete(Item entity) { /*Implement method*/}
}
With this in place, we have also standardized our repositories. Even for newer business objects, the coder will have to define the repository as per the generic interface only. You are of course free to add more generic operations to it like finding an object against lambda expression etc.
A really beautiful abstraction of CRUD operations!!
A really beautiful abstraction of CRUD operations!!
No comments:
Post a Comment