2.1.开发环境检查
- 开发工具:SpringToolSuite(STS)、MySql6.0、MySQL Workbench
- 检查STS的jdk配置:jdk8
- 检查STS的tomcat配置:tomcat8.5
-
2.2.搭建javaWeb工程总体架构
2.2.1.工程类型
2.2.2.导入jar包
mysql-connector-java-bin.jar
- jackson-core.jar
- jackson-annotations.jar
-
2.2.3.工程目录结构
2.2.4.MVC架构解决方案
本工程采用:基于Servlet的简易MVC架构。
- 本工程采用:约定优于配置的原则来搭建简易MVC框架。
- 本工程中规定:
请求url与Controller方法映射示例: http://localhost:8080/elm/Controller类名/Controller方法名 - 本工程中不需要任何配置文件。
附录:DispatcherServlet代码:
package com.neusoft.elm.framework;
import com.fasterxml.jackson.databind.ObjectMapper;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.lang.reflect.Method;
/**
* 自定义前端控制器 拦截url格式要求: /控制器类名/控制器方法名
*/
@WebServlet("/")
public class DispatcherServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//中文编码处理
request.setCharacterEncoding("utf-8");
response.setCharacterEncoding("utf-8");
response.setContentType("application/json;charset=utf-8");
//获取客户端请求路径(/HelloController/say)
String path = request.getServletPath();
//根据请求路径,将Controller的类名和方法名解析出来
String className = path.substring(1,path.lastIndexOf("/"));
String methodName = path.substring(path.lastIndexOf("/")+1);
PrintWriter out = null;
//判断请求路径,根据不同的请求,分发给不同的业务处理器
try{
//通过Controller类全名获取此类的所有信息
Class clazz = Class.forName("com.neusoft.elm.controller."+className);
//创建Controller类的对象
Object controller = clazz.newInstance();
//获取Controller类对象中的方法
Method method = clazz.getMethod(methodName,new Class[]{HttpServletRequest.class});
//调用上面获取的方法
Object result = method.invoke(controller,new Object[]{request});
//获取向客户端响应的输出流
out = response.getWriter();
ObjectMapper om = new ObjectMapper();
//向客户端响应json数据
out.print(om.writeValueAsString(result));
}catch(Exception e){
e.printStackTrace();
System.out.println("DispatcherServlet信息:请求url:"+path);
System.out.println("DispatcherServlet信息:类名:"+className+"\t方法名:"+methodName);
}finally {
out.close();
}
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request,response);
}
}
2.2.5.跨域问题解决方案
本工程使用CORS解决前后端分离开发模式时的跨域问题。 CORS是一个W3C标准,全称是”跨域资源共享”(Cross-origin resource sharing)。 它允许浏览器向跨源服务器,发出XMLHttpRequest请求,从而克服了AJAX只能同源使用的限制。
CORS需要浏览器和服务器同时支持。目前,所有浏览器都支持该功能,IE浏览器不能低于IE10。 整个CORS通信过程,都是浏览器自动完成,不需要用户参与。对于开发者来说,CORS通信与同源的AJAX通信没有差别,代码完全一样。浏览器一旦发现AJAX请求跨源,就会自动添加一些附加的头信息,有时还会多出一次附加的请求,但用户不会有感觉。 因此,实现CORS通信的关键是服务器。只要服务器实现了CORS接口,就可以跨源通信。
附录:CORS过滤器代码:
package com.neusoft.elm.filter;
import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletResponse;
@WebFilter("/*")
public class CorsFilter implements Filter{
@Override
public void init(FilterConfig filterConfig) throws ServletException { }
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
HttpServletResponse response = (HttpServletResponse) servletResponse;
//注意:这里设置只允许http://localhost:8081进行跨域访问
response.setHeader("Access-Control-Allow-Origin", "http://localhost:8081");
response.setHeader("Access-Control-Allow-Credentials", "true");
response.setHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS, DELETE, PUT");
response.setHeader("Access-Control-Max-Age", "3628800");
response.setHeader("Access-Control-Allow-Headers", "x-requested-with,Authorization");
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() { }
}
2.2.6.事务处理解决方案
首先要明确一点:事务处理放在service层。所以:
- Connection的创建与销毁要放在service层。
- 为了保证在同一次请求处理的线程中,service层和dao层都共用同一个Connection对象,需要将Connection对象放入ThreadLocal中。service层和dao层使用的Connection对象一律从ThreadLocal获取。
- dao层不再处理异常,dao层产生的异常将直接抛给service层进行处理。
- dao层负责关闭PreparedStatement和ResultSet,service层负责关闭Connection。
附录: ThreadLocal为解决多线程程序的并发问题提供了一种新的思路。 当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本。 Synchronized用于线程间的数据共享,而ThreadLocal则用于线程间的数据隔离。
DBUtil类代码:
package com.neusoft.elm.util;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
public class DBUtil {
private static final ThreadLocal<Connection> TL = new ThreadLocal<Connection>();
private static final String URL="jdbc:mysql://localhost:3306/elm?characterEncoding=utf-8";
private static final String DRIVER="com.mysql.jdbc.Driver";
private static final String USERNAME="root";
private static final String PASSWORD="123";
// 获取Connection
public static Connection getConnection() {
Connection con = null;
con = TL.get();
if (con==null) {
con = createConnection();
TL.set(con);
}
return con;
}
// 开启一个事务
public static void beginTransaction() throws Exception {
Connection con = null;
con = TL.get();
if (con == null) {
con = createConnection();
TL.set(con);
}
con.setAutoCommit(false);
}
// 提交一个事务
public static void commitTransaction() throws Exception {
Connection con = TL.get();
con.commit();
}
// 回滚一个事务
public static void rollbackTransaction() throws Exception {
Connection con = TL.get();
con.rollback();
}
// 关闭各种资源
public static void close(ResultSet rs, PreparedStatement pst) {
try {
if (rs != null) {
rs.close();
}
if (pst != null) {
pst.close();
}
} catch (SQLException e) {
e.printStackTrace();
}
}
// 关闭各种资源
public static void close(PreparedStatement pst) {
try {
if (pst != null) {
pst.close();
}
} catch (SQLException e) {
e.printStackTrace();
}
}
// 关闭各种资源
public static void close() {
Connection con = TL.get();
try {
if (con != null) {
con.close();
}
//至关重要,否则容易造成内存溢出等问题。
TL.remove();
} catch (SQLException e) {
e.printStackTrace();
}
}
private static Connection createConnection() {
Connection con = null;
if (con == null) {
try {
Class.forName(DRIVER);
con = DriverManager.getConnection(URL, USERNAME, PASSWORD);
} catch (Exception e) {
e.printStackTrace();
}
}
return con;
}
}
dao层代码示例:
@Override
//必须要有: throws Exception
public int saveCart(Cart cart) throws Exception{
int result = 0;
String sql = "insert into cart values(null,?,?,?,1)";
try {
//Connection从ThreadLocal中获取
con = DBUtil.getConnection();
pst = con.prepareStatement(sql);
pst.setInt(1, cart.getFoodId());
pst.setInt(2, cart.getBusinessId());
pst.setString(3, cart.getUserId());
result = pst.executeUpdate();
}finally { //这里不能处理异常,也就是没有catch,只有finally
DBUtil.close(pst); //这里负责关闭PreparedStatement和ResultSet
}
return result;
}
service层代码示例:
@Override
public int createOrders(String userId,Integer businessId,Integer daId) {
try {
DBUtil.beginTransaction(); //开启一个事务
//调用多个Dao进行数据处理
//... ...
//... ...
DBUtil.commitTransaction(); //提交一个事务
} catch (Exception e) {
try {
DBUtil.rollbackTransaction(); //回滚一个事务
} catch (Exception e1) {
e1.printStackTrace();
}
e.printStackTrace();
} finally {
DBUtil.close(); // 关闭Connection
}
return orderId;
}
2.2.7.Servlet返回JSON解决方案
Jackson 是当前用的比较广泛的,用来序列化和反序列化 json 的 Java 的开源框架。 Jackson 社 区相对比较活跃,更新速度也比较快, 从 Github 中的统计来看,Jackson 是最流行的 json 解析器之一 。 Spring MVC 的默认 json 解析器便是 Jackson。
本项目中,Servlet中使用Jackson将java对象或集合转换为json对象或数组后,返回前端。
ObjectMapper om = new ObjectMapper();
out.print(om.writeValueAsString(result)); //result就是响应数据
2.2.8.AJAX解决方案
由于本项目前端使用Vue框架,所有AJAX采用axios框架(这是尤雨溪推荐使用的)。
this.$axios.post('DeliveryAddressController/saveDeliveryAddress', this.$qs.stringify(
this.deliveryAddress //josn对象类型的提交参数
)).then(response => {
//做响应处理
}).catch(error => {
console.error(error);
});
2.2.9.图片解决方案
对于购物系统来说,商家信息、食品信息中必须要使用大量的图片。
在本工程中,由于使用的图片都比较小,所以采用Base64编码方式来存储图片信息。
Base64是一种用64个字符来表示任意二进制数据的方法。 我们知道,图片就是二进制数据。而Base64就可以将图片的二进制数据转换成字符串形式。这样能让我们方便的在数据库中,对图片进行存储和读取。
<img src="">
[