【asp.net core 系列】- 11 Service層的實現樣板

0.前言

在《asp.net core 系列》之實戰系列中,咱們在以前的篇幅中對項目有了一個大概的認知,也搭建了一個基礎的項目骨架。那麼就讓咱們繼續完善這個骨架,讓它更加豐滿。這一篇,我將帶領小夥伴們一塊兒實現用戶管理功能。數據庫

圖片

1. 數據表

通常狀況下,咱們會把用戶表和登陸信息表放在兩個表裏。爲何會這樣設計呢?出於如下幾種考慮:安全

  • 使功能分割,用戶信息管理是用戶管理,登陸是登陸asp.net

  • 增長安全,下降無關信息的查詢,例如訪問登陸接口不會連帶檢索用戶的普通訊息,當進行用戶信息管理的時候,不會把登陸信息也帶過來ide

等等工具

廢話很少說,直接上代碼:測試

namespace Data.Enums
{
   /// <summary>
   /// 登陸類型
   /// </summary>
   public enum LoginType : byte
   {
       /// token登陸
       Token,
       /// 用戶名密碼
       Password
   }
   /// <summary>
   /// 性別
   /// </summary>
   public enum SexEnum
   {
       /// 男
       Male,
       /// 女
       Female,
       /// 隱私
       None
   }
}

SysUserAuthEntity.cs加密

using Data.Enums;
using Data.Infrastructure;

namespace Data.Models
{
   public class SysUserAuthEntity : BaseEntity<int>
   {
       public string UserName { get; set; }
       public string Password { get; set; }

       public LoginType LoginType { get; set;}
   }
}

SysUserInfoEntity.csspa

using System;
using Data.Enums;
using Data.Infrastructure;

namespace Data.Models
{

   public class SysUserInfoEntity : BaseEntity<int>
   {
       public string NickName { get; set; }
       public string ImageUrl { get; set; }
       public SexEnum Sex { get; set; }
       public DateTime? BirthDay { get; set; }

       public int SysUserAuthId { get; set; }

       public virtual SysUserAuthEntity SysUserAuth { get; set; }
   }
}

這裏並無使用數據庫Sql語句做爲數據庫描述,而是使用了Entity類做爲描述,這是由於數據庫到實體類之間仍是有一層轉換,對於開發而言接觸更多的是實體類,而不是數據表。.net

2. 生成 Repository相關

使用工具代碼的方式有不少,我在這裏推薦一種, Test項目中,添加一個測試類,具體代碼以下:命令行

using NUnit.Framework;
using System;
using System.Collections.Generic;
using System.Text;
using Utils.Develop;

namespace Test
{
   public class DevelopTest
   {
       [Test]
       public void TetDevelop()
       {
           var d = Develop.CurrentDirect;
           Console.WriteLine(d);
           Assert.IsTrue(d.Contains("template"));
           var entities = Develop.LoadEntities();
           foreach (var item in entities)
           {
               Console.WriteLine(item.FullName);
           }
       }
       [Test]
       public void TestCreateDevelop()
       {
           var entities = Develop.LoadEntities();
           foreach (var item in entities)
           {
               Develop.CreateRepositoryInterface(item);
               Develop.CreateRepositoryImplement(item);
               Develop.CreateEntityTypeConfig(item);
           }
           Assert.Pass();
       }
   }
}

具體的命令行執行比較麻煩,會執行全部的測試單元:

cd Test/
dotnet test

固然了,IDE支持單個測試單元的執行,具體操做這裏就不作過多的介紹了。

3. Service 接口和實現類

一般Service接口會提供一個簡單Crud的Service接口,而後其餘業務接口繼承該接口。這樣能夠減小代碼的重複,由於重複的代碼在開發過程當中是很是討厭的一種狀況,由於一旦一處發生變動,其餘的也有可能發生變動。因此遇到重複代碼通常都會進行必定程度的封裝:

using System;
using System.Collections.Generic;
using System.Linq.Expressions;
using Data.Infrastructure;

namespace Service.Insfrastructure
{
   public interface BaseService<T>
   {
       T Get(object key);
       T Get(Expression<Func<T, bool>> predicate);

       PageModel<T> SearchPage(PageCondition<T> condition);

       void Delete(Expression<Func<T, bool>> predicate);

       void Update(T entity);

       List<T> Search(Expression<Func<T, bool>> predicate);

   }
}

暫時就提供了這些最多見的請求方法。

在Service.Implements項目中:

using System;
using System.Collections.Generic;
using System.Linq.Expressions;
using Data.Infrastructure;
using Domain.Insfrastructure;
using Service.Insfrastructure;

namespace Service.Implements.Insfrastructure
{
   public abstract class BaseServiceImpl<T> : BaseService<T>
   {
       private IRepository<T> LocalRepository { get; }

       protected BaseServiceImpl(IRepository<T> repository)
       {
           LocalRepository = repository;
       }


       public T Get(object key)
       {
           return LocalRepository.Get(key);
       }

       public T Get(Expression<Func<T, bool>> predicate)
       {
           return LocalRepository.Get(predicate);
       }

       public PageModel<T> SearchPage(PageCondition<T> condition)
       {
           return LocalRepository.Search(condition);
       }

       public void Delete(Expression<Func<T, bool>> predicate)
       {
           LocalRepository.Delete(predicate);
       }

       public void Update(T entity)
       {
           LocalRepository.Update(entity);
       }

       public List<T> Search(Expression<Func<T, bool>> predicate)
       {
           return LocalRepository.Search(predicate);
       }
   }
}

這個類設置爲抽象類。

4. 用戶管理的接口

先建立了兩個簡單的示範接口:

using Data.Models;
using Service.Insfrastructure;

namespace Service
{
   public interface ISysUserService : BaseService<SysUserInfoEntity>
   {
       void Register(SysUserAuthEntity auth, SysUserInfoEntity info);

       void ChangePassword(int userId, string oldPwd, string newPwd);
   }
}

實現類:

using System;
using Data.Models;
using Domain.Repository;
using Service.Implements.Insfrastructure;

namespace Service.Implements
{
   public class SysUserServiceImpl : BaseServiceImpl<SysUserInfoEntity>, ISysUserService
   {
       protected ISysUserAuthRepository AuthRepository { get; }
       protected ISysUserInfoRepository InfoRepository { get; }

       public SysUserServiceImpl(ISysUserAuthRepository authRepository, ISysUserInfoRepository infoRepository) : base(
           infoRepository)
       {
           AuthRepository = authRepository;
           InfoRepository = infoRepository;
       }

       public void Register(SysUserAuthEntity auth, SysUserInfoEntity info)
       {
           var authItem = AuthRepository.Get(p => p.LoginType == auth.LoginType && p.UserName == auth.UserName);
           if (authItem != null)
           {
               throw new Exception("用戶信息已經存在");
           }

           info.SysUserAuth = auth;
           InfoRepository.Insert(info);
       }

       public void ChangePassword(int userId, string oldPwd, string newPwd)
       {
           var info = InfoRepository.Get(userId);
           var auth = AuthRepository.Get(info.SysUserAuthId);
           if (oldPwd == null || oldPwd != auth?.Password)
           {
               throw new Exception("原密碼錯誤");
           }

           auth.Password = newPwd;

       }
   }
}

這裏沒對密碼進行加密處理,直接使用明文。這在正式開發中是不容許的,密碼不能使用明文保存。固然,這也不是最終代碼,下一篇咱們將介紹一下.net core中常見的加密實現。

5. 總結

這一篇經過幾個簡單的示例爲你們介紹了一下Service層的開發邏輯以及理念。下一篇將爲你們介紹一下.net core中幾種簡單的加密實現。

相關文章
相關標籤/搜索