ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

平台管理后台与商家菜单资源管理:平台管理后台数据服务设计

2021-06-05 22:57:48  阅读:216  来源: 互联网

标签:数据服务 name Operators 平台 private public 后台 operators id


平台管理后台与商家菜单资源管理

平台管理后台是为电商平台的运营方提供服务的,它主要包含商家管理和一些公共配置管理的功能。在商家管理的设计中,包括商家的注册、审核、商家用户的权限管理和菜单资源管理等功能。除一些公共管理功能的设计外,平台管理后台本身也有安全控制管理的设计。

平台管理后台的项目工程为manage-microservice,完整的源代码可以从本书源代码中下载。

本章实例代码的分支为V2.1,检出请注意更新。平台管理后台的开发主要包含两大部分内容,一部分是管理后台本身的访问控制管理设计,另一部分是商家及其菜单资源管理。这两部分的功能都在模块manage-web 中实现。

平台管理后台与商家菜单资源管理:平台管理后台数据服务设计

 

平台管理后台数据服务设计

平台管理后台是一个独立的应用系统,它有自身的用户体系和独立的权限管理设计。平台管理后台的访问控制及其权限管理主要由操作员、角色和部门等实体组成,这几个实体的关联关系是:一个操作员只能从属于一个部门,同时一个操作员可以拥有多个角色。

实体建模

为实现访问控制及简单的权限管理设计,我们定义了三个实体对象,分别是操作员、角色和部门。

操作员的实体对象Operators主要由名称、邮箱、性别、密码和所属部门等属性组成,实现代码如下所示:

@Entity
@Table (name = "toperator")
public class Operators extends IdEntity{
private String name;
private string email;private Integer sex;
@DateTimeFormat (pattern = "yyyy-MM-dd HH:mm :ss")
ecolumn(name = "created",columnDefinition = "timestamp defaultCurrent timestamp ")
@Temporal (TemporalType.TIMESTAMP)private Date created;
private String password;
@ManyToOne
@JoinColumn (name = "did")@JsonBackReference
private Department department;
@ManyToMany(cascade ={},fetch = FetchType.EAGER)@JoinTable(name = "operator part",
joinColumns = {@JoinColumn(name - "operator_id")},inverseJoinColumns = {@JoinColumn(name = "part_id")})
private List<Part> parts = new ArrayList<>();
public Operators() {
}
...
}

在这一设计中,主要实现了操作员的关联关系。一方面以多对一的关系关联了部门实体,即一个操作员只能属于一个部门;另一方面以多对多的关系关联了角色实体,即一个操作员可以拥有多个角色。

角色的实体对象Part主要由名称和创建时间等属性组成,实现代码如下所示:

@Entity
@Table(name = "t part")
public class Part extends IdEntity {
private String name;
@DateTimeFormat (pattern = "yyyy-MM-dd HH:mm:ss")
@Column (name = "created",columnDefinition = "timestamp defaultcurrenttimestamp")
@Temporal(TemporalType.TIMESTAMP)private Date created;
public Part(){
}
...
}

这个角色设计并没有关联资源,有关它的访问权限设计将在本章后面使用一种更加简单的方式实现。

部门的实体对象Department主要由名称和创建时间等属性组成,实现代码如下所示:

@Entity
@Table(name = "t department")
public class Department extends IdEntity {
private String name;
@DateTimeFormat (pattern = "yyyy-MM-dd HH:mm :ss")
eColumn (name = "created",columnDefinition = "timestamp defaultcurrent timestamp ")
@Temporal (TemporalType.TIMESTAMP)private Date created;
public Department (){
}
...
}

在实体对象之间的关联关系设计中,使用了单向关联的设计原则,即在部门实体设计中,不再与操作员建立关联关系。如果需要从部门中查询操作员,则可以通过使用SQL查询语句实现

在上面的实体对象设计中,都继承了一个抽象类Baseld,用它实现了对象中ID主键的定义,代码如下所示:

@MappedSuperclass
public abstract class Baseld implements Serializable{
private static final long serialVersionUID =1L;
@Id
@Generatedvalue (strategy = GenerationType. IDENTITY)private Long id;
public Long getId() {
return id;
public void setId (Long id){
this.id= id;
}
}

为实体赋予行为

为实体定义一个存储库接口,通过绑定JPA存储库,就可以为之赋予一些基本行为,实现实体的持久化设计。在存储库接口的定义中,还可以通过声明方法增加一个实体的其他操作方法。

平台管理后台与商家菜单资源管理:平台管理后台数据服务设计

 

对于上节的操作员实体对象来说,可以通过如下方式设计它的操作方法:

@Repository
public interface OperatorRepository extends JpaRepository<Operators,Long>,JpaSpecificationExecutor<0perators> {
CQuery ("select t from Operators t where t.name =?1 and t.email =?2")Operators findByNameAndEmail(String name, String email);
GQuery ("select distinct u from Operators u where u.name= :name ")Operators findByName (RParam( "name") String name);
cQuery ("select distinct u from Operators u where u.id= :id")Operators findById(@Param("id") Long id);
@Query("select o from Operators o "+
"left join o.parts p"+
"where p.id= :id")
List<0perators> findByPartId(@Param ("id") Long id);
}

这里声明了几个方法,都是通过SQL查询语句扩展操作员实体对象操作行为的。其中findByPartld实现了通过部门ID查询操作员列表的功能。

另外,在角色实体和部门实体的持久化设计中,只要创建一个简单的存储库接口,通过绑定JPA接口,就可以实现基本操作,不再赘述。

数据访问服务设计

数据访问服务是对存储库接口调用的一个服务层封装设计,通过服务层的开发,可以为存储库接口的调用提供统一的事务管理,实现其他扩展设计。

下面以操作员实体数据访问服务开发为例进行说明。

对于一般的增删改查的操作,可以使用如下所示的设计:
@Service
@Transactional
public class OperatorService
CAutowired
private operatorRepository operatorRepository;
public String insert (Operators operators) {
try{
Operators old = findByName (operators .getName());
if(old == null) {
operatorRepository.save (operators) ;
return operators.getId() .toString();
}else{
return "用户名'"+ old.getName()+”已经存在! ";
}catch (Exception e) {
e.printStackTrace() ;
return e.getMessage() ;
}
}
public string update (Operators operators) {
try{
operatorRepository. save (operators) ; 
return operators.getId() . toString() ;
}catch (Exception e){
e.printStackTrace() ;
return e.getMessage() ;
}}
public String delete (Long id) {
try{
operatorRepository .deleteById(id) ;
return id. toString() ;
}catch (Exception e) {
e.printStackTrace() ;
return e.getMessage() ;
}
}
public List<0perators>findA11(){
return operatorRepository.findAl1();
}
public Operators find0ne (Long id){
return operatorRepository.findByOperatorId(id);
}
public Operators findByName (string name){
return operatorRepository.findByName (name) ;
public List<Operators> findByPartId (Long partId){
return operatorRepository.findByPartId(partId);
}
}

这些方法都是通过调用操作员实体的存储库接口实现数据存取操作的。其中,对于操作员来说,因为需要使用用户名进行登录验证,所以在新增数据时使用了去重检查。

平台管理后台与商家菜单资源管理:平台管理后台数据服务设计

 

对于分页查询来说,可以通过定义一个Specification实现复杂的查询,代码如下所示:

@service
@Transactional
public class OperatorService {
@Autowired
private OperatorRepository operatorRepository;
public Page<0perators> findAll (0peratorsVo operatorsVo){
Sort sort = Sort.by (Sort.Direction.DESC, "created");
Pageable pageable = PageRequest.of(operatorsVo.getPage(),
operatorsVo.getsize(), sort);
return operatorRepository.findAll (new Specification<Operators>(){
@override
public Predicate toPredicate (Root<Operators> root, CriteriaQuery<
query, CriteriaBuilder criteriaBuilder) {
List<Predicate> predicatesList = new ArrayList<Predicate>();
if(CommonUtils.isNotNull (operatorsVo.getName())){
predicatesList.add (criteriaBuilder.like (root.get ("name"),
"g"+operatorsVo.getName() +"o"));
}
if (CommonUtils.isNotNull(operatorsVo.getCreated())){
predicatesList.add (criteriaBuilder.greaterhan (root.get ("created"),operatorsVo. getCreated()));
query.where(predicatesList.toArray (new
Predicate[predicatesList.size()]));
return query.getRestriction();
},pageable);
}
}

其中,分页查询的参数可以根据需要进行设计,这里只提供了操作员名称和创建日期两个参数进行查询。

其中,分页查询的参数可以根据需要进行设计,这里只提供了操作员名称和创建日期两个参数进行查询。

单元测试

在完成数据访问服务设计之后,可以进行一个单元测试,以验证设计是否正确。

下面是一个插入数据的测试用例,通过这个测试用例我们可以创建一个登录系统的管理员用户:

@Runwith (SpringRunner.class)
@ContextConfiguration (classes ={JpaConfiguration.class,ManageRestApiApplication.class})
@SpringBootTest
public class BbServiceTest {
private static Logger logger =
LoggerFactory.getLogger (BbServiceTest.class);
@Autowired
private OperatorService operatorService;@Autowired
private PartService partService;@Autowired
private DepartmentService departmentService;
@Test
public void insertData(){
Partpart=new Part(;part.setName ( "admins");
partService.save(part);
Department department = new Department ();department.setName("技术部");
departmentService.save(department);
Operators operators =new Operators();operators.setName ("admin");
operators.setSex(1);
operators.set Department (department);
List<Part> partList = operators.getParts();partList.add(part);
operators.setParts(partList);
BCryptPasswordEncoder bc = new BCryptPasswordEncoder();operators.setPassword(bc.encode ( "123456"));
operatorService.insert(operators);
assert operators.getId() >0: "create error";
}
}

这个测试用例执行了以下操作:

(1)创建了一个角色,名称为admins(也可以把它当成一个用户组)。

(2)创建了一个部门,名称为技术部。

(3)创建了一个用户,名称为admin,并给用户设定密码为123456。

如果测试成功通过,则这些生成的数据可以为后面的开发所使用。我们可以使用admin这个用户作为系统管理员登录系统。

其他测试用例可以参照这个方法来设计。

本文给大家讲解的内容是平台管理后台与商家菜单资源管理:平台管理后台数据服务设计
  1. 下篇文章给大家讲解的是平台管理后台与商家菜单资源管理:平台管理后台的访问控制设计;
  2. 觉得文章不错的朋友可以转发此文关注小编;
  3. 感谢大家的支持!

标签:数据服务,name,Operators,平台,private,public,后台,operators,id
来源: https://blog.51cto.com/u_15254813/2870999

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有