jsp和jstl 事务&数据库连接池&DBUtils

Posted IT技术学习栈

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了jsp和jstl 事务&数据库连接池&DBUtils相关的知识,希望对你有一定的参考价值。

JSP & EL & JST&jsp

Java Server Page

  • 什么是jsp

从用户角度看待 ,就是是一个网页 , 从程序员角度看待 , 其实是一个java类, 它继承了servlet,所以可以直接说jsp 就是一个Servlet.

  • 为什么会有jsp?

html 多数情况下用来显示静态内容 , 一成不变的。但是有时候我们需要在网页上显示一些动态数据, 比如:查询所有的学生信息, 根据姓名去查询具体某个学生。这些动作都需要去查询数据库,然后在网页上显示。html是不支持写java代码  , jsp里面可以写java代码。

怎么用JSP

指令写法

<%@ 指令名字 %>

page指令

  • language

表明jsp页面中可以写java代码

  • contentType

其实即使说这个文件是什么类型,告诉浏览器我是什么内容类型,以及使用什么编码

 contentType="text/html; charset=UTF-8"

text/html MIMEType 这是一个文本,html网页
  • pageEncoding  jsp内容编码

  • extends 用于指定jsp翻译成java文件后,继承的父类是谁,一般不用改。

  • import 导包使用的,一般不用手写。

  • session

值可选的有true or false .

用于控制在这个jsp页面里面,能够直接使用session对象。

具体的区别是,请看翻译后的java文件   如果该值是true , 那么在代码里面会有getSession()的调用,如果是false :  那么就不会有该方法调用,也就是没有session对象了。在页面上自然也就不能使用session了。

  • errorPage

指的是错误的页面, 值需要给错误的页面路径

  • isErrorPage

上面的errorPage 用于指定错误的时候跑到哪一个页面去。那么这个isErroPage , 就是声明某一个页面到底是不是错误的页面。

include

包含另外一个jsp的内容进来。


<%@ include file="other02.jsp"%>
  • 背后细节:

把另外一个页面的所有内容拿过来一起输出。所有的标签元素都包含进来。

taglib

<%@ taglib prefix=""  uri=""%>  


uri: 标签库路径
prefix : 标签库的别名  

JSP 动作标签


<jsp:include page=""></jsp:include>
<jsp:param value="" name=""/>
<jsp:forward page=""></jsp:forward>
  • jsp:include

    <jsp:include page="other02.jsp">

包含指定的页面, 这里是动态包含。也就是不把包含的页面所有元素标签全部拿过来输出,而是把它的运行结果拿过来。

  • jsp:forward

    <jsp:forward page="">

前往哪一个页面。  


<%
//请求转发
request.getRequestDispatcher("other02.jsp").forward(request, response);
%>
  • jsp:param

意思是:在包含某个页面的时候,或者在跳转某个页面的时候,加入这个参数。

<jsp:forward page="other02.jsp"> <jsp:param value="beijing" name="address"/>


在other02.jsp中获取参数

<br>收到的参数是:<br>

<%= request.getParameter("address")%>


JSP内置对象

所谓内置对象,就是我们可以直接在jsp页面中使用这些对象。不用创建。

  • pageContext

  • request

  • session

  • application

以上4个是作用域对象 ,

  • 作用域

表示这些对象可以存值,他们的取值范围有限定。setAttribute   和  getAttribute


使用作用域来存储数据<br>

<%
pageContext.setAttribute("name", "page");
request.setAttribute("name", "request");
session.setAttribute("name", "session");
application.setAttribute("name", "application");
%>

取出四个作用域中的值<br>

<%=pageContext.getAttribute("name")%>
<%=request.getAttribute("name")%>
<%=session.getAttribute("name")%>
<%=application.getAttribute("name")%>

作用域范围大小:


pageContext -- request --- session -- application

四个作用域的区别

  • pageContext 【PageContext】

作用域仅限于当前的页面。  

还可以获取到其他八个内置对象。

  • request 【HttpServletRequest】

作用域仅限于一次请求, 只要服务器对该请求做出了响应。这个域中存的值就没有了。

  • session 【HttpSession】

作用域限于一次会话(多次请求与响应) 当中。

  • application 【ServletContext】

整个工程都可以访问, 服务器关闭后就不能访问了。


  • out 【JspWriter】

    • response  【HttpServletResponse】

  • exception  【Throwable】

    • config 【ServletConfig】

    • page 【Object】 ---就是这个jsp翻译成的java类的实例对象

EL表达式

是为了简化咱们的jsp代码,具体一点就是为了简化在jsp里面写的那些java代码。

  • 写法格式

${表达式 }

如果从作用域中取值,会先从小的作用域开始取,如果没有,就往下一个作用域取。一直把四个作用域取完都没有, 就没有显示。


如何使用

1. 取出4个作用域中存放的值。


<%
pageContext.setAttribute("name", "page");
request.setAttribute("name", "request");
session.setAttribute("name", "session");
application.setAttribute("name", "application");
%>

按普通手段取值<br>

<%= pageContext.getAttribute("name")%>
<%= request.getAttribute("name")%>
<%= session.getAttribute("name")%>
<%= application.getAttribute("name")%>

<br>使用EL表达式取出作用域中的值<br>

${ pageScope.name }
${ requestScope.name }
${ sessionScope.name }
${ applicationScope.name }
  1. 如果域中所存的是数组

    <% String [] a = {"aa","bb","cc","dd"}; pageContext.setAttribute("array", a); %>


    使用EL表达式取出作用域中数组的值<br>

    ${array[0] } , ${array[1] },${array[2] },${array[3] }
  1. 如果域中锁存的是集合


使用EL表达式取出作用域中集合的值<br>

${li[0] } , ${li[1] },${li[2] },${li[3] }

<br>-------------Map数据----------------<br>
<%
Map map = new HashMap();
map.put("name", "zhangsna");
map.put("age",18);
map.put("address","北京..");

map.put("address.aa","深圳..");

pageContext.setAttribute("map", map);
%>
  1. 取出Map集合的值

    <% Map map = new HashMap(); map.put("name", "zhangsna"); map.put("age",18); map.put("address","北京..");


    map.put("address.aa","深圳..");

  pageContext.setAttribute("map", map);   %>   使用EL表达式取出作用域中Map的值<br>     {map.age } , {map["address.aa"] }

取值细节:

  1. 从域中取值。得先存值。<%


    //pageContext.setAttribute("name", "zhangsan");
    session.setAttribute("name", "lisi...");

    %>

    <br>直接指定说了,到这个作用域里面去找这个name<br>${ pageScope.name }

  <br>//先从page里面找,没有去request找,去session,去application <br>   ${ name }

  <br>指定从session中取值<br>   ${ sessionScope.name }  

  1. 取值方式

如果这份值是有下标的,那么直接使用[]


<%
String [] array = {"aa","bb","cc"}
session.setAttribute("array",array);
%>

${ array[1] } --> 这里array说的是attribute的name

如果没有下标, 直接使用 .的方式去取


<%
User user = new User("zhangsan",18);

session.setAttribute("u", user);
%>

${ u.name } , ${ u.age }

一般使用EL表达式,用的比较多的,都是从一个对象中取出它的属性值,比如取出某一个学生的姓名。

EL表达式 的11个内置对象。

${ 对象名.成员 }

  • pageContext

作用域相关对象

  • pageScope

  • requestScope

  • sessionScope

  • applicationScope

头信息相关对象

  • header

  • headerValues

参数信息相关对象

  • param

  • paramValues

  • cookie全局初始化参数

  • initParam

JSTL

全称 :JSP Standard Tag Library  jsp标准标签库

简化jsp的代码编写。替换 <%%> 写法。一般与EL表达式配合

怎么使用

  1. 导入jar文件到工程的WebContent/Web-Inf/lib  jstl.jar standard.jar

  2. 在jsp页面上,使用taglib 指令,来引入标签库

  3. 注意:如果想支持 EL表达式,那么引入的标签库必须选择1.1的版本,1.0的版本不支持EL表达式


<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
  • c:set


<!-- 声明一个对象name, 对象的值 zhangsan , 存储到了page(默认) , 指定是session -->
<c:set var="name" value="zhangsan" scope="session"></c:set>

${sessionScope.name }
  • c:if

判断test里面的表达式是否满足,如果满足,就执行c:if标签中的输出 , c:if 是没有else的。


<c:set var="age" value="18" ></c:set>
<c:if test="${ age > 26 }">
年龄大于了26岁...
</c:if>

<c:if test="${ age <= 26 }">
年龄小于了26岁...
</c:if>

------------------------------

定义一个变量名 flag 去接收前面表达式的值,然后存在session域中

<c:if test="${ age > 26 }" var="flag" scope="session">
年龄大于了26岁...
</c:if>
  • c:forEach


从1 开始遍历到10 ,得到的结果 ,赋值给 i ,并且会存储到page域中, step , 增幅为2,
<c:forEach begin="1" end="10" var="i" step="2">
${i }
</c:forEach>

-----------------------------------------------

<!-- items : 表示遍历哪一个对象,注意,这里必须写EL表达式。
var: 遍历出来的每一个元素用user 去接收。-->
<c:forEach var="user" items="${list }">
${user.name } ----${user.age }
</c:forEach>

学生信息管理系统

  • 需求分析

  1. 先写 login.jsp , 并且搭配一个LoginServlet 去获取登录信息。

  2. 创建用户表, 里面只要有id , username  和 password

  3. 创建UserDao, 定义登录的方法

    /**


*   该dao定义了对用户表的访问规则
    */
    public interface UserDao {

    /**
* 这里简单就返回一个Boolean类型, 成功或者失败即可。
      *
* 但是开发的时候,登录的方法,一旦成功。这里应该返回该用户的个人信息
      * @param userName
      * @param password
      *
* @return true : 登录成功, false : 登录失败。
          */
        boolean login(String userName , String password);
        }
  1. 创建UserDaoImpl , 实现刚才定义的登录方法。


public class UserDaoImpl implements UserDao {

@Override
public boolean login(String userName , String password) {

Connection conn = null;
PreparedStatement ps = null;
ResultSet rs   = null;
try {
//1. 得到连接对象
conn = JDBCUtil.getConn();

String sql = "select * from t_user where username=? and password=?";

//2. 创建ps对象
ps = conn.prepareStatement(sql);
ps.setString(1, userName);
ps.setString(2, password);

//3. 开始执行。
rs = ps.executeQuery();

//如果能够成功移到下一条记录,那么表明有这个用户。
return rs.next();

} catch (SQLException e) {
e.printStackTrace();
}finally {
JDBCUtil.release(conn, ps, rs);
}
return false;
}

}
  1. 在LoginServlet里面访问UserDao, 判断登录结果。以区分对待

  2. 创建stu_list.jsp , 让登录成功的时候跳转过去。

  3. 创建学生表 , 里面字段随意。

  4. 定义学生的Dao . StuDao  


public interface StuDao {

/**
* 查询出来所有的学生信息
* @return List集合
*/
List<Student> findAll();
}
  1. 对上面定义的StuDao 做出实现 StuDaoImpl

    public class StuDaoImpl implements StuDao {


    @Override
    public List<Student> findAll() {
    List<Student> list = new ArrayList<Student>();
    Connection conn = null;
    PreparedStatement ps = null;
    ResultSet rs   = null;
    try {
    //1. 得到连接对象
    conn = JDBCUtil.getConn();

    String sql = "select * from t_stu";

    ps = conn.prepareStatement(sql);

    rs = ps.executeQuery();

    //数据多了,用对象装, 对象也多了呢?用集合装。   while(rs.next()){ //10 次 ,10个学生     Student stu = new Student();     stu.setId(rs.getInt("id"));   stu.setAge(rs.getInt("age"));   stu.setName(rs.getString("name"));   stu.setGender(rs.getString("gender"));   stu.setAddress(rs.getString("address"));     list.add(stu);     }   } catch (SQLException e) {   e.printStackTrace();   }finally {   JDBCUtil.release(conn, ps, rs);   }     return list;   }     }

  1. 在登录成功的时候,完成三件事情。

  2. 查询所有的学生

    1. 把这个所有的学生集合存储到作用域中。

    2. 跳转到stu_list.jsp


protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

//提交的数据有可能有中文, 怎么处理。
request.setCharacterEncoding("UTF-8");
response.setContentType("text/html;charset=utf-8");

//1. 获取客户端提交的信息
String userName = request.getParameter("username");
String password = request.getParameter("password");

//2. 去访问dao , 看看是否满足登录。
UserDao dao = new UserDaoImpl();
boolean isSuccess = dao.login(userName, password);

//3. 针对dao的返回结果,做出响应
if(isSuccess){
//response.getWriter().write("登录成功.");

//1. 查询出来所有的学生信息。
StuDao stuDao = new StuDaoImpl();
List<Student> list = stuDao.findAll();

//2. 先把这个集合存到作用域中。
request.getSession().setAttribute("list", list);

//2. 重定向
response.sendRedirect("stu_list.jsp");

}else{
response.getWriter().write("用户名或者密码错误!");
}
}
  1. 在stu_list.jsp中,取出域中的集合,然后使用c标签 去遍历集合。

	<table border="1" width="700">
<tr align="center">
<td>编号</td>
<td>姓名</td>
<td>年龄</td>
<td>性别</td>
<td>住址</td>
<td>操作</td>
</tr>

<c:forEach items="${list }" var="stu">
<tr align="center">
<td>${stu.id }</td>
<td>${stu.name }</td>
<td>${stu.age }</td>
<td>${stu.gender }</td>
<td>${stu.address }</td>
<td><a href="#">更新</a> <a href="#">删除</a></td>
</tr>
</c:forEach>
</table>



总结:

  • JSP

    三大指令

    	page
    include
    taglib

    三个动作标签
    <jsp:include>
    <jsp:forward>
    <jsp:param>


  九个内置对象

  四个作用域   pageContext   request   session   application     out   exception   response   page   config

  • EL

    ${ 表达式 }

    取4个作用域中的值

    ${ name }


    有11个内置对象。

    pageContext

    pageScope
    requestScope
    sessionScope
    applicationScope

    header
    headerValues

    param
    paramValues

    cookie
    initParam


  • JSTL

使用1.1的版本, 支持EL表达式, 1.0不支持EL表达式

拷贝jar包, 通过taglib 去引入标签库

<c:set>
<c:if>
<c:forEach>








事务&数据库连接池&DBUtils

事务

Transaction  其实指的一组操作,里面包含许多个单一的逻辑。只要有一个逻辑没有执行成功,那么都算失败。所有的数据都回归到最初的状态(回滚)

  • 为什么要有事务?

为了确保逻辑的成功。例子:银行的转账。

使用命令行方式演示事务。

  • 开启事务

    start transaction;

  • 提交或者回滚事务

    commit; 提交事务, 数据将会写到磁盘上的数据库rollback ;  数据回滚,回到最初的状态。

  1. 关闭自动提交功能。

  1. 演示事务

使用代码方式演示事务

代码里面的事务,主要是针对连接来的。

  1. 通过conn.setAutoCommit(false )来关闭自动提交的设置。

  1. 提交事务  conn.commit();

  1. 回滚事务 conn.rollback();

@Test
public void testTransaction(){

Connection conn = null;
PreparedStatement ps = null;
ResultSet rs = null;
try {
conn = JDBCUtil.getConn();

//连接,事务默认就是自动提交的。关闭自动提交。
conn.setAutoCommit(false);

String sql = "update account set money = money - ? where id = ?";
ps = conn.prepareStatement(sql);

//扣钱, 扣ID为1 的100块钱
ps.setInt(1, 100);
ps.setInt(2, 1);
ps.executeUpdate();


int a = 10 /0 ;

//加钱, 给ID为2 加100块钱
ps.setInt(1, -100);
ps.setInt(2, 2);
ps.executeUpdate();

//成功:提交事务。
conn.commit();

} catch (SQLException e) {
try {
//事变:回滚事务
conn.rollback();
} catch (SQLException e1) {
e1.printStackTrace();
}
e.printStackTrace();

}finally {
JDBCUtil.release(conn, ps, rs);
}
}

事务的特性

  • 原子性

指的是 事务中包含的逻辑,不可分割。

  • 一致性

指的是 事务执行前后。数据完整性

  • 隔离性

指的是 事务在执行期间不应该受到其他事务的影响

  • 持久性

指的是 事务执行成功,那么数据应该持久保存到磁盘上。

事务的安全隐患

不考虑隔离级别设置,那么会出现以下问题。

脏读 不可重读读  幻读.

* 脏读

> 一个事务读到另外一个事务还未提交的数据

* 不可重复读

> 一个事务读到了另外一个事务提交的数据 ,造成了前后两次查询结果不一致。

读未提交 演示

  1. 设置A窗口的隔离级别为 读未提交

  2. 两个窗口都分别开启事务

丢失更新


读已提交演示

  1. 设置A窗口的隔离级别为 读已提交

  1. A B 两个窗口都开启事务, 在B窗口执行更新操作。

  1. 在A窗口执行的查询结果不一致。一次是在B窗口提交事务之前,一次是在B窗口提交事务之后。

这个隔离级别能够屏蔽 脏读的现象, 但是引发了另一个问题  ,不可重复读。

可串行化

如果有一个连接的隔离级别设置为了串行化 ,那么谁先打开了事务, 谁就有了先执行的权利, 谁后打开事务,谁就只能得着,等前面的那个事务,提交或者回滚后,才能执行。但是这种隔离级别一般比较少用。容易造成性能上的问题。效率比较低。

  • 按效率划分,从高到低

读未提交  > 读已提交  > 可重复读  > 可串行化

  • 按拦截程度 ,从高到底

可串行化 > 可重复读 > 读已提交 >  读未提交

事务总结

需要掌握的

  1. 在代码里面会使用事务

    conn.setAutoCommit(false);


conn.commit();

conn.rollback();
  1. 事务只是针对连接连接对象,如果再开一个连接对象,那么那是默认的提交。

  2. 事务是会自动提交的。

需要了解的

安全隐患



脏读
一个事务读到了另一个事务未提交的数据
不可重复读
一个事务读到了另一个事务已提交的数据,造成前后两次查询结果不一致
幻读
一个事务读到了另一个事务insert的数据 ,造成前后查询结果不一致 。



丢失更新。

隔离级别

读未提交

引发问题:脏读

读已提交

解决:脏读 , 引发:不可重复读

可重复读

解决:脏读 、 不可重复读 , 未解决:幻读

可串行化

解决:脏读、 不可重复读 、 幻读。

mysql 默认的隔离级别是 可重复读

Oracle 默认的隔离级别是  读已提交

丢失更新

解决丢失更新

  • 悲观锁

可以在查询的时候,加入 for update

  • 乐观锁

要求程序员自己控制。


数据库连接池

  1. 数据库的连接对象创建工作,比较消耗性能。

2.一开始现在内存中开辟一块空间(集合) , 一开先往池子里面放置 多个连接对象。后面需要连接的话,直接从池子里面去。不要去自己创建连接了。使用完毕, 要记得归还连接。确保连接对象能循环利用。

自定义数据库连接池

  • 代码实现

  • 出现的问题:

    1. 单例。

    2. 无法面向接口编程。

      UserDao dao = new UserDaoImpl();dao.insert();

    1. 需要额外记住 addBack方法

DataSource dataSource = new MyDataSource();

因为接口里面没有定义addBack方法。

  1. 怎么解决?   以addBack 为切入点。

解决自定义数据库连接池出现的问题。

由于多了一个addBack 方法,所以使用这个连接池的地方,需要额外记住这个方法,并且还不能面向接口编程。

我们打算修改接口中的那个close方法。原来的Connection对象的close方法,是真的关闭连接。打算修改这个close方法,以后在调用close, 并不是真的关闭,而是归还连接对象。

如何扩展某一个方法?

原有的方法逻辑,不是我们想要的。想修改自己的逻辑

  1. 直接改源码  无法实现。

  2. 继承, 必须得知道这个接口的具体实现是谁。

  3. 使用装饰者模式。


开源连接池

DBCP

  1. 导入jar文件

  2. 不使用配置文件:


public void testDBCP01(){


Connection conn = null;
PreparedStatement ps = null;
try {

//1. 构建数据源对象
BasicDataSource dataSource = new BasicDataSource();
//连的是什么类型的数据库, 访问的是哪个数据库 , 用户名, 密码。。
//jdbc:mysql://localhost/bank 主协议:子协议 ://本地/数据库
dataSource.setDriverClassName("com.mysql.jdbc.Driver");
dataSource.setUrl("jdbc:mysql://localhost/bank");
dataSource.setUsername("root");
dataSource.setPassword("root");


//2. 得到连接对象
conn = dataSource.getConnection();
String sql = "insert into account values(null , ? , ?)";
ps = conn.prepareStatement(sql);
ps.setString(1, "admin");
ps.setInt(2, 1000);

ps.executeUpdate();

} catch (SQLException e) {
e.printStackTrace();
}finally {
JDBCUtil.release(conn, ps);
}

}
  1. 使用配置文件方式:


Connection conn = null;
PreparedStatement ps = null;
try {
BasicDataSourceFactory factory = new BasicDataSourceFactory();
Properties properties = new Properties();
InputStream is = new FileInputStream("src//dbcpconfig.properties");
properties.load(is);
DataSource dataSource = factory.createDataSource(properties);

//2. 得到连接对象
conn = dataSource.getConnection();
String sql = "insert into account values(null , ? , ?)";
ps = conn.prepareStatement(sql);
ps.setString(1, "liangchaowei");
ps.setInt(2, 100);

ps.executeUpdate();

} catch (Exception e) {
e.printStackTrace();
}finally {
JDBCUtil.release(conn, ps);
}

  • C3P0

拷贝jar文件 到 lib目录

不使用配置文件方式


Connection conn = null;
PreparedStatement ps = null;
try {
//1. 创建datasource
ComboPooledDataSource dataSource = new ComboPooledDataSource();
//2. 设置连接数据的信息
dataSource.setDriverClass("com.mysql.jdbc.Driver");

//忘记了---> 去以前的代码 ---> jdbc的文档
dataSource.setJdbcUrl("jdbc:mysql://localhost/bank");
dataSource.setUser("root");
dataSource.setPassword("root");

//2. 得到连接对象
conn = dataSource.getConnection();
String sql = "insert into account values(null , ? , ?)";
ps = conn.prepareStatement(sql);
ps.setString(1, "admi234n");
ps.setInt(2, 103200);

ps.executeUpdate();

} catch (Exception e) {
e.printStackTrace();
}finally {
JDBCUtil.release(conn, ps);
}

使用配置文件方式


//默认会找 xml 中的 default-config 分支。
ComboPooledDataSource dataSource = new ComboPooledDataSource();
//2. 设置连接数据的信息
dataSource.setDriverClass("com.mysql.jdbc.Driver");

//忘记了---> 去以前的代码 ---> jdbc的文档
dataSource.setJdbcUrl("jdbc:mysql://localhost/bank");
dataSource.setUser("root");
dataSource.setPassword("root");

//2. 得到连接对象
conn = dataSource.getConnection();
String sql = "insert into account values(null , ? , ?)";
ps = conn.prepareStatement(sql);
ps.setString(1, "admi234n");
ps.setInt(2, 103200);


DBUtils

增删改


//dbutils 只是帮我们简化了CRUD 的代码, 但是连接的创建以及获取工作。不在他的考虑范围
QueryRunner queryRunner = new QueryRunner(new ComboPooledDataSource());


//增加
//queryRunner.update("insert into account values (null , ? , ? )", "aa" ,1000);

//删除
//queryRunner.update("delete from account where id = ?", 5);

//更新
//queryRunner.update("update account set money = ? where id = ?", 10000000 , 6);

查询

  1. 直接new接口的匿名实现类


QueryRunner queryRunner = new QueryRunner(new ComboPooledDataSource());

Account account = queryRunner.query("select * from account where id = ?", new ResultSetHandler<Account>(){

@Override
public Account handle(ResultSet rs) throws SQLException {
Account account = new Account();
while(rs.next()){
String name = rs.getString("name");
int money = rs.getInt("money");

account.setName(name);
account.setMoney(money);
}
return account;
}

}, 6);

System.out.println(account.toString());
  1. 直接使用框架已经写好的实现类。


* 查询单个对象

QueryRunner queryRunner = new QueryRunner(new ComboPooledDataSource());
//查询单个对象
Account account = queryRunner.query("select * from account where id = ?",
new BeanHandler<Account>(Account.class), 8);


* 查询多个对象

QueryRunner queryRunner = new QueryRunner(new ComboPooledDataSource());
List<Account> list = queryRunner.query("select * from account ",
new BeanListHandler<Account>(Account.class));

ResultSetHandler 常用的实现类

以下两个是使用频率最高的


BeanHandler, 查询到的单个数据封装成一个对象
BeanListHandler, 查询到的多个数据封装 成一个List<对象>


ArrayHandler, 查询到的单个数据封装成一个数组
ArrayListHandler, 查询到的多个数据封装成一个集合 ,集合里面的元素是数组。


MapHandler, 查询到的单个数据封装成一个map
MapListHandler,查询到的多个数据封装成一个集合 ,集合里面的元素是map。


ColumnListHandlerKeyedHandlerScalarHandler

总结

事务


使用命令行演示

使用代码演示

脏读、

不可重复读、

幻读丢失更新


悲观锁
乐观锁

4个隔离级别
读未提交
读已提交
可重复读
可串行化

数据连接池

  • DBCP

    不使用配置

    使用配置

  • C3P0

    不使用配置

    使用配置 (必须掌握)

  • 自定义连接池

    装饰者模式

DBUtils

简化了我们的CRUD , 里面定义了通用的CRUD方法。


queryRunner.update();
queryRunner.query



以上是关于jsp和jstl 事务&数据库连接池&DBUtils的主要内容,如果未能解决你的问题,请参考以下文章

第77节:Java中的事务和数据库连接池和DBUtiles

09- 事务&数据库连接池&DBUtils

13-事务&数据库连接池&DBUtiles

Java Web -- 事务 & 数据库连接池 & DBUtiles

JavaWeb之事务&数据库连接池

jsp&el&jstl-01