Spring 框架是 Java 应用最广的框架,它的成功来源于理念,而不是技术本身,它的理念包括 IoC (控制反转) 和 A面向切面编程)。Spring框架是个轻量级的Java EE框架,所谓轻量级,是指不依赖于容器就能运行的。简单来说,Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架。
作用于web层,相当于controller,与struts中的action一样,都是用来处理用户请求的。同时,相比于struts2来说,更加细粒度,它是基于方法层面的,而struts是基于类层面的。Spring MVC 分离了控制器、模型对象、分派器以及处理程序对象的角色,这种分离让它们更容易进行定制。
Struts2 与 SpringMVC 的对比:
MyBatis 本是apache的一个开源项目iBatis, 2010年这个项目由apache software foundation 迁移到了google code,并且改名为MyBatis 。MyBatis 是一款优秀的持久层框架,它支持定制化 SQL、存储过程以及高级映射。MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集。MyBatis 可以使用简单的 XML 或注解来配置和映射原生信息,将接口和 Java 的 POJOs(Plain Old Java Objects,普通的 Java对象)映射成数据库中的记录。
Hibernate 和 MyBatis 的对比:
本项目将以购物为背景,主要包括商品信息及库存【因为想顺便学习一下事务的处理】、订单信息。下面将从数据库创建、项目结构说明、配置文件、业务代码等方面进行一步步说明。
1. 商品表
CREATE TABLE `goods` (
`goods_id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '商品ID',
`goodsname` varchar(100) NOT NULL COMMENT '商品名称',
`number` int(11) NOT NULL COMMENT '商品库存',
PRIMARY KEY (`goods_id`)
) ENGINE=InnoDB AUTO_INCREMENT=1000 DEFAULT CHARSET=utf8 COMMENT='商品表'
初始化表数据
INSERT INTO `goods` (`goods_id`, `goodsname`, `number`)
VALUES (1001, 'SN卫衣', 15)
2. 订单表
CREATE TABLE `orderinfo` (
`order_id` varchar(20) NOT NULL COMMENT '订单编号',
`goods_id` bigint(18) NOT NULL COMMENT '商品ID',
`user_id` bigint(10) NOT NULL COMMENT '用户ID',
`order_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '下单时间' ,
PRIMARY KEY (`order_id`),
INDEX `idx_order_id` (`order_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='订单表'
OK,至此表结构及初始化数据构建完成,下面说下基于Mavan的项目结构。
因为项目是使用maven来管理jar包的,先来贴一下,pom.xml的配置
pom.xml
为了避免学习小伙伴崇尚拿来主义【也就是去除了xmlns之类的东西】,这里只放项目依赖的jar包的dependencies;本案例将本着“需则用”的原则,避免在网上看到的各种乱七八糟的依赖都丢进来的情况,造成资源浪费和干扰阅读。
junit
junit
4.11
ch.qos.logback
logback-classic
1.1.1
mysql
mysql-connector-java
5.1.37
runtime
c3p0
c3p0
0.9.1.2
org.mybatis
mybatis
3.3.0
org.mybatis
mybatis-spring
1.2.3
taglibs
standard
1.1.2
jstl
jstl
1.2
com.fasterxml.jackson.core
jackson-databind
2.5.4
javax.servlet
javax.servlet-api
3.1.0
org.springframework
spring-core
4.1.7.RELEASE
org.springframework
spring-beans
4.1.7.RELEASE
org.springframework
spring-context
4.1.7.RELEASE
org.springframework
spring-jdbc
4.1.7.RELEASE
org.springframework
spring-tx
4.1.7.RELEASE
org.springframework
spring-test
4.1.7.RELEASE
redis.clients
jedis
2.7.3
com.dyuproject.protostuff
protostuff-core
1.0.8
com.dyuproject.protostuff
protostuff-runtime
1.0.8
commons-collections
commons-collections
3.2
*项目结构图
src/test/java:用于junit的测试类 src/main/java:
dao:数据库处理
service:业务处理
enums:项目枚举
mapper:dao中方法对应mybatis映射文件,Sql就在这里面
web:控制器,controller
entity:项目中的实体类,如:商品类和订单类
jdbc.properties
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://serverName:port/dbname?useUnicode=true&characterEncoding=utf8
jdbc.username=[填写自己的数据库用户名]
jdbc.password=[填写自己的数据库登录密码]
logback.xml
这里直接用的是控制台输出,如果是生产环境,可以根据具体的需求进行配置。
ch.qos.logback.classic.encoder.PatternLayoutEncoder -->
%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n
mybatis-config
这里主要是MyBaties全局配置文件的配置,可以将一些类的别名、主键自增配置、驼峰命名规则配置等。
spring 相关配置文件
为了更加清晰的了解spring各个组件的作用,这里将数据源的配置、事务配置和视图解析器的配置分开来。
spring-dao.xml
这里面主要就是spring配置整合mybatis的具体过程,具体包括:
spring-service
实际的开发过程中事务一般都是在service层进行操作。因此用一个单独的spring-service.xml来进行事务的相关的配置
spring-web.xml
配置SpringMVC;需要说明一下,一般我们在实际的开发过程中,会配置json2map解析。这里没有用到就不贴出来,读者可以自行网上搜索一波。
web.xml
encodingFilter
org.springframework.web.filter.CharacterEncodingFilter
true
encoding
UTF-8
encodingFilter
/*
org.springframework.web.context.ContextLoaderListener
org.springframework.web.util.IntrospectorCleanupListener
mvc-dispatcher
org.springframework.web.servlet.DispatcherServlet
contextConfigLocation
classpath:spring/spring-*.xml
mvc-dispatcher
/
至此,所有的配置文件结束,下面将进行具体的代码环节
这里mapper中的xml文件就不贴了,自行脑补。。。。
实体类:包括商品和订单
商品类
/**
* 商品信息类
* @author glmapper
*
*/
public class Goods {
private long goodsId;// 商品ID
private String goodsName;// 商品名称
private int number;// 商品库存
public long getGoodsId() {
return goodsId;
}
public void setGoodsId(long goodsId) {
this.goodsId = goodsId;
}
public String getGoodsName() {
return goodsName;
}
public void setGoodsName(String goodsName) {
this.goodsName = goodsName;
}
public int getNumber() {
return number;
}
public void setNumber(int number) {
this.number = number;
}
}
订单类
/**
* 订单信息类
* @author glmapper
*
*/
public class OrderInfo {
private String orderId;//订单ID
private long goodsId;//商品ID
private long userId;//用户ID
private Date orderTime;//下单时间
public String getOrderId() {
return orderId;
}
public void setOrderId(String orderId) {
this.orderId = orderId;
}
public long getGoodsId() {
return goodsId;
}
public void setGoodsId(long goodsId) {
this.goodsId = goodsId;
}
public long getUserId() {
return userId;
}
public void setUserId(long userId) {
this.userId = userId;
}
public Date getOrderTime() {
return orderTime;
}
public void setOrderTime(Date orderTime) {
this.orderTime = orderTime;
}
}
public interface GoodsDao {
/**
* 通过ID查询单件商品信息
*
* @param id
* @return
*/
Goods queryById(long id);
/**
* 查询所有商品信息
*
* @param offset 查询起始位置
* @param limit 查询条数
* @return
*/
List queryAll(@Param("offset") int offset, @Param("limit") int limit);
/**
* 减少商品库存
*
* @param bookId
* @return 如果影响行数等于>1,表示更新的记录行数
*/
int reduceNumber(long goodsId);
}
public interface OrderInfoDao {
/**
* 插入订单记录
*
* @param OrderInfo orderInfo
* @return 插入的行数
*/
int insertOrderInfo(OrderInfo orderInfo);
/**
* 通过主键查询订单记录,返回订单实体
* @param orderId
* @return
*/
OrderInfo queryByOrderId(String orderId);
}
@Service("orderService")
public class OrderServiceImpl implements OrderService {
//log生成器
private Logger logger = LoggerFactory.getLogger(OrderServiceImpl.class);
// 注入dao依赖【商品dao,订单dao】
@Autowired
private GoodsDao goodsDao;
@Autowired
private OrderInfoDao orderInfoDao;
@Override
public Goods getById(long goodsId) {
// TODO Auto-generated method stub
return goodsDao.queryById(goodsId);
}
@Override
public List getList(int offset,int limit) {
// TODO Auto-generated method stub
return goodsDao.queryAll(offset, limit);
}
@Override
@Transactional
public OrderInfo buyGoods(long goodsId, long userId) {
//扣减库存,插入订单 =一个事务 如果失败则执行回滚
try {
// 减库存
int update = goodsDao.reduceNumber(goodsId);
if (update <= 0) {// 库存不足
throw new NoNumberException("no number");
} else {
// 执行预约操作
OrderInfo orderInfo=new OrderInfo();
orderInfo.setGoodsId(goodsId);
orderInfo.setUserId(userId);
orderInfo.setOrderTime(new Date());
String orderId=getRandomOrderId(goodsId);
orderInfo.setOrderId(orderId);
int insert = orderInfoDao.insertOrderInfo(orderInfo);
if (insert <= 0) {// 重复预约
throw new RepeatAppointException("repeat appoint");
} else {// 预约成功
return orderInfo;
}
}
} catch (Exception e) {
//这里可以丰富下具体的返回信息
logger.error("下单失败");
}
return null;
}
private String getRandomOrderId(long goodsId) {
SimpleDateFormat dateFormater = new SimpleDateFormat("yyyyMMddhhmmss");
String prefix=dateFormater.format(new Date());
String goodsIdStr=goodsId+"";
String temp="";
for (int i = 0; i < 6; i++) {
Random random=new Random(goodsIdStr.length()-1);
temp+=goodsIdStr.charAt(random.nextInt());
}
return prefix+temp;
}
}
OK,至此所有核心代码及配置文件罗列完毕;【mapper中的xml和具体的controller就不贴了,相信大家对这个也不陌生。本文主要意图在于梳理下自己学习中的一些点,SSM框架在实际的应用开发中还会有很多其他的开源技术结合进来,如:quartz,redis等。当前本文的列子就是一个空壳子,以备参考吧】
我有一个微信公众号,经常会分享一些Java技术相关的干货;如果你喜欢我的分享,可以用微信搜索“Java团长”或者“javatuanzhang”关注。
参考: