無需多言,做爲開源屆首屈一指的典例,項目開發中無處不在;
核心IOC容器,用來裝載bean(java中的類)-用Spring的IOC容器來管理Bean的生命週期,有了這樣一種機制,咱們就能夠不用在代碼中去重複的作new操做。
aop,面向切面編程,spring中最主要的是用於事務方面的使用。css
做用於web層,至關於controller,與struts中的action同樣,都是用來處理用戶請求的。同時,相比於struts2來講,更加細粒度,它是基於方法層面的,而struts是基於類層面的。
html
MyBatis 是一款優秀的持久層框架,它支持定製化 SQL、存儲過程以及高級映射。MyBatis 避免了幾乎全部的 JDBC 代碼和手動設置參數以及獲取結果集。MyBatis 可使用簡單的 XML 或註解來配置和映射原生信息,將接口和 Java 的 POJOs(Plain Old Java Objects,普通的 Java對象)映射成數據庫中的記錄。[來自:www.mybatis.org/mybatis-3/z…]java
他人總結mysql
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的項目結構。pom.xml
爲了不學習小夥伴崇尚拿來主義【也就是去除了xmlns之類的東西】,這裏只放項目依賴的jar包的dependencies;本案例將本着「需則用」的原則,避免在網上看到的各類亂七八糟的依賴都丟進來的狀況,形成資源浪費和干擾閱讀。web
<dependencies>
<!-- 單元測試 -->
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.11</version>
</dependency>
<!-- 1.日誌 slf4j-->
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.1.1</version>
</dependency>
<!-- 2.數據庫鏈接驅動 -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.37</version>
<scope>runtime</scope>
</dependency>
<!-- 2.數據庫鏈接池 -->
<dependency>
<groupId>c3p0</groupId>
<artifactId>c3p0</artifactId>
<version>0.9.1.2</version>
</dependency>
<!-- 3.MyBatis 以及 spring-mybatis -->
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.3.0</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>1.2.3</version>
</dependency>
<!-- 4.Servlet 相關依賴 -->
<dependency>
<groupId>taglibs</groupId>
<artifactId>standard</artifactId>
<version>1.1.2</version>
</dependency>
<dependency>
<groupId>jstl</groupId>
<artifactId>jstl</artifactId>
<version>1.2</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.5.4</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
</dependency>
<!-- 5.Spring -->
<!-- 5.1 Spring核心 :core bean context -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-core</artifactId>
<version>4.1.7.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-beans</artifactId>
<version>4.1.7.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>4.1.7.RELEASE</version>
</dependency>
<!-- 5.2 Spring jdbc依賴,事務依賴 -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>4.1.7.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>4.1.7.RELEASE</version>
</dependency>
<!-- 5.3 Spring web依賴>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>4.1.7.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>4.1.7.RELEASE</version>
</dependency>
<!-- 5.4 Spring test -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>4.1.7.RELEASE</version>
</dependency>
<!-- 6.redis客戶端:Jedis【不使用的話能夠直接去除】 -->
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>2.7.3</version>
</dependency>
<dependency>
<groupId>com.dyuproject.protostuff</groupId>
<artifactId>protostuff-core</artifactId>
<version>1.0.8</version>
</dependency>
<dependency>
<groupId>com.dyuproject.protostuff</groupId>
<artifactId>protostuff-runtime</artifactId>
<version>1.0.8</version>
</dependency>
<!-- 7.工具類 -->
<dependency>
<groupId>commons-collections</groupId>
<artifactId>commons-collections</artifactId>
<version>3.2</version>
</dependency>
</dependencies>複製代碼
*項目結構圖redis
jdbc.propertiesspring
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://serverName:port/dbname?useUnicode=true&characterEncoding=utf8
jdbc.username=[填寫本身的數據庫用戶名]
jdbc.password=[填寫本身的數據庫登陸密碼]複製代碼
<?xml version="1.0" encoding="UTF-8"?>
<configuration debug="true">
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
ch.qos.logback.classic.encoder.PatternLayoutEncoder -->
<encoder>
<pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
</encoder>
</appender>
<root level="debug">
<appender-ref ref="STDOUT" />
</root>
</configuration>複製代碼
mybatis-config
這裏主要是MyBaties全局配置文件的配置,能夠將一些類的別名、主鍵自增配置、駝峯命名規則配置等。sql
<configuration>
<!-- 配置全局屬性 -->
<settings>
<!-- 使用jdbc的getGeneratedKeys獲取數據庫自增主鍵值 -->
<setting name="useGeneratedKeys" value="true" />
<!-- 使用列別名替換列名 默認:true -->
<setting name="useColumnLabel" value="true" />
<!-- 開啓駝峯命名轉換:Table{create_time} -> Entity{createTime} -->
<setting name="mapUnderscoreToCamelCase" value="true" />
</settings>
</configuration>複製代碼
spring 相關配置文件
爲了更加清晰的瞭解spring各個組件的做用,這裏將數據源的配置、事務配置和視圖解析器的配置分開來。
spring-dao.xml
這裏面主要就是spring配置整合mybatis的具體過程,具體包括:
1.引入數據庫配置文件
2.配置數據源【數據庫鏈接池】
3.配置SqlSessionFactory對象
4.配置掃描Dao接口包,動態實現Dao接口,注入到spring容器中數據庫
<!-- 1.配置數據庫相關參數properties的屬性:${url} -->
<context:property-placeholder location="classpath:jdbc.properties" />
<!-- 2.數據庫鏈接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<!-- 配置鏈接池屬性 -->
<property name="driverClass" value="${jdbc.driver}" />
<property name="jdbcUrl" value="${jdbc.url}" />
<property name="user" value="${jdbc.username}" />
<property name="password" value="${jdbc.password}" />
<!-- c3p0鏈接池的私有屬性 -->
<property name="maxPoolSize" value="30" />
<property name="minPoolSize" value="10" />
<!-- 關閉鏈接後不自動commit -->
<property name="autoCommitOnClose" value="false" />
<!-- 獲取鏈接超時時間 -->
<property name="checkoutTimeout" value="10000" />
<!-- 當獲取鏈接失敗重試次數 -->
<property name="acquireRetryAttempts" value="2" />
</bean>
<!-- 3.配置SqlSessionFactory對象 -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<!-- 注入數據庫鏈接池 -->
<property name="dataSource" ref="dataSource" />
<!-- 配置MyBaties全局配置文件:mybatis-config.xml -->
<property name="configLocation" value="classpath:mybatis-config.xml" />
<!-- 掃描entity包 使用別名 -->
<property name="typeAliasesPackage" value="com.glmapper.framerwork.entity" />
<!-- 掃描sql配置文件:mapper須要的xml文件 -->
<property name="mapperLocations" value="com.glmapper.framerwork.mapper/*.xml" />
</bean>
<!-- 4.配置掃描Dao接口包,動態實現Dao接口,注入到spring容器中 -->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<!-- 注入sqlSessionFactory -->
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory" />
<!-- 給出須要掃描Dao接口包 -->
<property name="basePackage" value="com.glmapper.framerwork.dao" />
</bean>複製代碼
<!-- 掃描service包下全部使用註解的類型 -->
<context:component-scan base-package="com.glmapper.framerwork.service" />
<!-- 配置事務管理器 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<!-- 注入數據庫鏈接池 -->
<property name="dataSource" ref="dataSource" />
</bean>
<!-- 配置基於註解的聲明式事務 -->
<tx:annotation-driven transaction-manager="transactionManager" />複製代碼
spring-web.xml
配置SpringMVC;須要說明一下,通常咱們在實際的開發過程當中,會配置json2map解析。這裏沒有用到就不貼出來,讀者能夠自行網上搜索一波。編程
<!-- 1.開啓SpringMVC註解模式 -->
<mvc:annotation-driven />
<!-- 2.靜態資源默認servlet配置
(1)加入對靜態資源的處理:js,css,圖片等
(2)容許使用"/"作總體映射
-->
<mvc:default-servlet-handler/>
<!-- 3.配置視圖解析器ViewResolver -->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="viewClass" value="org.springframework.web.servlet.view.JstlView" />
<property name="prefix" value="/WEB-INF/jsp/" />
<property name="suffix" value=".jsp" />
</bean>
<!-- 4.掃描web相關的bean -->
<context:component-scan base-package="com.glmapper.framerwork.web" />複製代碼
<!-- 編碼過濾器 -->
<filter>
<filter-name>encodingFilter</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<async-supported>true</async-supported>
<init-param>
<param-name>encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>encodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<!-- Spring監聽器 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<!-- 防止Spring內存溢出監聽器 -->
<listener>
<listener-class>org.springframework.web.util.IntrospectorCleanupListener</listener-class>
</listener>
<!-- 配置DispatcherServlet -->
<servlet>
<servlet-name>mvc-dispatcher</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 配置springMVC須要加載的配置文件
spring-dao.xml,spring-service.xml,spring-web.xml
Mybatis - > spring -> springmvc
-->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/spring-*.xml</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>mvc-dispatcher</servlet-name>
<!-- 默認匹配全部的請求 -->
<url-pattern>/</url-pattern>
</servlet-mapping>複製代碼
至此,全部的配置文件結束,下面將進行具體的代碼環節商品類
/**
* 商品信息類
* @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<Goods> queryAll(@Param("offset") int offset, @Param("limit") int limit);
/**
* 減小商品庫存
*
* @param bookId
* @return 若是影響行數等於>1,表示更新的記錄行數
*/
int reduceNumber(long goodsId);
}複製代碼
訂單dao
public interface OrderInfoDao {
/**
* 插入訂單記錄
*
* @param OrderInfo orderInfo
* @return 插入的行數
*/
int insertOrderInfo(OrderInfo orderInfo);
/**
* 經過主鍵查詢訂單記錄,返回訂單實體
* @param orderId
* @return
*/
OrderInfo queryByOrderId(String orderId);
}複製代碼
下單服務接口orderService
@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<Goods> 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等。當前本文的列子就是一個空殼子,以備參考吧】