JavaWeb学习笔记
前台和后台
前台是给普通用户使用的
后台是给系统管理员使用的
前台和后台访问地址不一样
例如:
前台地址:/client/bookServlet
后台地址:/manager/bookServlet
表单重复提交
表单重复提交有三种常见的情况:
情况一:提交完表单。如果采用请求转发,当用户提交完请求,浏览器会记录下最后一次请求的全部信息。此时若用户按下F5功能键,就会发起浏览器记录的最后一次请求,导致表单重复提交!
解决方案:请求重定向
情况二:用户正常提交服务器,但是由于网络延迟等原因,长时间未收到服务器响应,此时,用户重复点击提交,也会造成表单重复提交
情况三:用户正常提交服务器,且成功提交。但成功后用户回退浏览器,重新提交,也会造成表单重复提交

ThreadLocal
ThreadLocal的作用:解决多线程的数据安全问题
ThreadLocal可以给当前线程关联一个数据(可以是普通变量、对象、数组、集合等)
特点:
- ThreadLocal可以为当前线程关联一个数据(它可以像Map一样存取数据,key为当前线程)
- 每一个ThreadLocal对象,只能为当前线程关联一个数据,如果要为当前线程关联多个数据,就需要使用多个ThreadLocal对象实例
- 每个ThreadLocal对象实例定义的时候,一般都是static类型
- ThreadLocal中保存数据,在线程销毁后,会由JVM虚拟机自动释放
使用Map实现数据关联
package com.tang.web.servlet;
import java.util.Hashtable;
import java.util.Map;
import java.util.Random;
public class ThreadLocalTest {
public static Map<String, Object> data = new Hashtable<String, Object>();
private static Random random = new Random();
public static class Task implements Runnable{
@Override
public void run() {
//在run方法中,随机生成一个变量(线程要关联的数据),然后以当前线程名为key保存到map中
Integer integer = random.nextInt(1000);
//获取当前线程名
String name = Thread.currentThread().getName();
System.out.println("线程【"+name+"】生成的随机数是:"+integer);
data.put(name, integer);
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
new OrderService().createOrder();
//在run方法结束之前,以当前线程名取出数据并打印
System.out.println("线程【"+name+"】取出的数据为:"+data.get(name));
}
}
public static void main(String[] args) {
for (int i=0; i<3; i++){
new Thread(new Task()).start();
}
}
}
package com.tang.web.servlet;
public class OrderService {
public void createOrder(){
String name = Thread.currentThread().getName();
System.out.println("OrderService 线程【"+name+"】取出的数据:"+ThreadLocalTest.data.get(name));
}
}
使用ThreadLocal实现数据关联
package com.tang.web.servlet;
import java.util.Random;
public class ThreadLocalTest {
public static ThreadLocal<Object> threadLocal = new ThreadLocal<Object>();
private static Random random = new Random();
public static class Task implements Runnable{
@Override
public void run() {
//在run方法中,随机生成一个变量(线程要关联的数据)
Integer integer = random.nextInt(1000);
String name = Thread.currentThread().getName();
System.out.println("线程【"+name+"】随机生成的数据是:"+integer);
threadLocal.set(integer);
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
new OrderService().createOrder();
//在run方法结束之前,取出数据并打印
System.out.println("线程【"+name+"】取出的数据为:"+threadLocal.get());
}
}
public static void main(String[] args) {
for (int i=0; i<3; i++){
new Thread(new Task()).start();
}
}
}
package com.tang.web.servlet;
public class OrderService {
public void createOrder(){
String name = Thread.currentThread().getName();
System.out.println("OrderService 线程【"+name+"】取出的数据:"+ThreadLocalTest.threadLocal.get());
}
}
使用ThreadLocal确保所有操作使用同一个Connection对象来实现事务管理
要确保所有操作要么都成功,要么都失败,必须使用数据库的事务
要确保所有操作都在一个事务里,就必须要确保所有操作都使用同一个Connection连接对象
如何确保所有操作都使用同一个Connection连接对象?
我们可以将Connection连接对象作为ThreadLocal为当前线程关联的数据,这就要确保所有操作都在同一个线程中完成。这样,所有操作都使用同一个线程关联的同一个Conneion连接对象完成jdbc操作!
public class JdbcUtils{
private static DruidDataSource dataSource;
private static ThreadLocal<Connection> conns = new ThreadLocal<Connection>();
static {
try {
Properties properties = new Properties();
//读取 jdbc.properties 属性配置文件
InputStream inputStream = JdbcUtils.class.getClassLoader().getResourceAsStream("jdbc.properties");
//从流中加载数据
properties.load(inputStream);
//创建数据库连接池
dataSource = (DruidDataSource) DruidDataSourceFactory.createDataSource(properties);
} catch (Exception e) {
e.printStackTrace();
}
}
//获取数据库连接池中的连接
public static Connection getConnection(){
Connection conn = conns.get();
if(conn == null) {
try {
conn = dataSource.getConnection(); //从连接池中获取连接
conns.set(conn);
conn.setAutoCommit(false); //设置为手动管理事务
} catch (Exception e) {
e.printStackTrace();
}
}
return conn;
}
//提交事务,并关闭释放连接
public static void commitAndClose() {
Connection connection = conns.get();
if (connection != null) { //如果不等于null,说明之前使用过连接
try {
connection.commit(); //提交事务
} catch (SQLException e) {
e.printStackTrace();
} finally {
try {
connection.close(); //关闭连接,释放资源
} catch (SQLException e) {
e.printStackTrace();
}
}
}
//一定要执行remove操作,防止threadLocal太多了,导致内存泄漏
conns.remove();
}
//回滚事务,并关闭释放连接
public static void rollbackAndClose() {
Connection connection = conns.get();
if (connection != null) { //如果不等于null,说明之前使用过连接
try {
connection.rollback(); //回滚事务
} catch (SQLException e) {
e.printStackTrace();
} finally {
try {
connection.close(); //关闭连接,释放资源
} catch (SQLException e) {
e.printStackTrace();
}
}
}
//一定要执行remove操作,防止threadLocal太多了,导致内存泄漏
conns.remove();
}
传统交互方式和Ajax
传统的交互方式:
浏览器发送请求
服务器收到请求,处理请求经常要给页面携带数据。 req.setAttribute("map", map); 转发到页面
浏览器接收到页面数据,在页面使用EL表达式获取数据
导致页面整个刷新:造成很大的服务器负担
只让服务器返回我们需要的部分数据即可,不用返回整个页面。XMLHttpRequest代替浏览器发送请求,接收响应数据,再利用dom增删改的方式来改变页面效果;
浙公网安备 33010602011771号