前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >02 整合IDEA+Maven+SSM框架的高并发的商品秒杀项目之Service层

02 整合IDEA+Maven+SSM框架的高并发的商品秒杀项目之Service层

作者头像
nnngu
发布2018-03-15 16:04:38
1.2K0
发布2018-03-15 16:04:38
举报
文章被收录于专栏:nnngunnngunnngu

项目源代码:https://github.com/nnngu/nguSeckill


首先在编写Service层代码前,我们应该首先要知道这一层到底是干什么的。

Service层主要负责业务模块的逻辑应用设计。同样是首先设计接口,再设计其实现的类,接着在Spring的配置文件中配置其实现的关联。这样我们就可以在应用中调用Service接口来进行业务处理。Service层的业务实现,具体要调用到已定义的dao层的接口,封装Service层的业务逻辑有利于通用的业务逻辑的独立性和重复利用性,程序显得非常简洁。

在项目中要降低耦合的话,分层是一种很好的概念,就是各层各司其职,尽量不做不相干的事,所以Service层的话顾名思义就是业务逻辑,处理程序中的一些业务逻辑,以及调用dao层的代码,这里我们的dao层就是连接数据库的那一层,调用关系可以这样表达:

View(页面) > Controller(控制层) > Service(业务逻辑) > Dao(数据访问) > Database(数据库)

首先还是接口的设计,设计秒杀商品的接口,在com.nnngu.service.interfaces包下建立SeckillService.java接口文件,如下图:

SeckillService.java文件里面的内容请参照项目的源代码。

建立好接口之后我们要写实现类了,在写实现类的时候我们肯定会碰到一个这样的问题,你要向前端返回json数据的话,你是返回什么样的数据好?直接返回一个数字状态码或者文字?这样设计肯定是不好的,所以我们应该向前端返回一个实体信息json,里面包含了一系列的信息,无论是哪种状态都应该可以应对,既然是与数据库字段无关的类,那就不是PO了,所以我们建立一个DTO数据传输类。关于常见的几种对象我的解释如下:

  • PO:也就是我们为每一张数据库表写一个实体类
  • VO:对某个页面或者展现层所需要的数据,封装成一个实体类
  • BO:业务对象
  • DTO:跟VO的概念有点混淆,也是相当于页面需要的数据封装成一个实体类
  • POJO:简单的无规则java对象

在com.nnngu下建立dto包,然后建立Exposer类,这个类是秒杀时数据库那边处理的结果的对象

Exposer.java文件里面的内容请参照项目的源代码。

定义秒杀中可能会出现的异常

定义一个基础的异常,所有的子异常继承这个异常SeckillException

package com.nnngu.exception;

/**
 *  秒杀基础的异常
 * Created by nnngu
 */
public class SeckillException extends RuntimeException {
    // 代码省略,请参照项目的源代码
    ... ...
}

可能会出现秒杀关闭后被秒杀情况,所以建立秒杀关闭异常SeckillCloseException,需要继承我们前面写的基础异常

package com.nnngu.exception;

/**
 * 秒杀已经关闭异常,当秒杀结束就会出现这个异常
 * Created by nnngu
 */
public class SeckillCloseException extends SeckillException{
    // 代码省略,请参照项目的源代码
    ... ...
}

定义重复秒杀异常RepeatKillException

package com.nnngu.exception;

/**
 * 重复秒杀异常,不需要我们手动去try catch
 * Created by nnngu
 */
public class RepeatKillException extends SeckillException{
    // 代码省略,请参照项目的源代码
    ... ...
}

实现Service接口

com.nnngu.service包下创建SeckillServiceImpl.java类,具体代码请参照项目的源代码。

在这里我们捕获了运行时异常,这样做的原因就是Spring的事务默认发生了RuntimeException才会回滚,可以检测出来的异常是不会导致事务的回滚的,这样的目的就是你明知道这里会发生异常,所以你一定要进行处理。如果只是为了让编译通过的话,那捕获异常也没意思,所以这里要注意事务的回滚。

然后我们还发现这里存在硬编码的现象,就是返回各种字符常量,例如秒杀成功,秒杀失败等等,这些字符串是可以被重复使用的,而且这样维护起来也不方便,要到处去类里面寻找这样的字符串,所有我们使用枚举类来管理这样状态,在con.nnngu包下建立enums包,专门放置枚举类,然后再建立SeckillStatEnum枚举类。

枚举类SeckillStatEnum.java的代码请参照项目的源代码。

注入Service

resources/spring下建立applicationContext-service.xml文件,用来配置Service

applicationContext-service.xml的代码请参照项目的源代码。

在这里开启了基于注解的事务,常见的事务操作有以下几种方法:

  • 在Spring早期版本中是使用ProxyFactoryBean+XMl方式来配置事务。
  • 在Spring配置文件使用tx:advice+aop命名空间,好处就是一次配置永久生效,你无须去关心中间出的问题,不过出错了你很难找出在哪里出了问题。
  • 注解@Transactional的方式,注解可以在方法定义,接口定义,类定义。可以在public方法上,但是不能注解在private、final、static等方法上,因为Spring的事务管理默认是使用cglib动态代理的:
  • private方法因为访问权限限制,无法被子类覆盖
  • final方法无法被子类覆盖
  • static时类级别的方法,无法被子类覆盖
  • protected方法可以被子类覆盖,因此可以被动态字节码增强

不能被Spring AOP事务增强的方法

序号

动态代理策略

不能被事务增强的方法

1

基于JDK的动态代理

除了public以外的所有方法,并且 public static 的方法也不能被增强

2

基于cglib的动态代理

private,static,final 的方法

Service层的测试

添加测试类SeckillServiceImplTest.java,如下图:

SeckillServiceImplTest.java的代码请参照项目的源代码。

测试结果:

测试的方法:public void getSeckillList()

测试结果如下图:

到此,我们成功完成了Service层开发及测试。

下一篇:03 Java高并发秒杀项目之web层

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 定义秒杀中可能会出现的异常
  • 实现Service接口
  • 注入Service
    • 不能被Spring AOP事务增强的方法
    • Service层的测试
      • 测试结果:
        • 下一篇:03 Java高并发秒杀项目之web层
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档