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 }
如果域中所存的是数组
<% String [] a = {"aa","bb","cc","dd"}; pageContext.setAttribute("array", a); %>
使用EL表达式取出作用域中数组的值<br>
${array[0] } , ${array[1] },${array[2] },${array[3] }
如果域中锁存的是集合
使用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);
%>
取出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"] }
取值细节:
从域中取值。得先存值。<%
//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 }
取值方式
如果这份值是有下标的,那么直接使用[]
<%
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表达式配合
怎么使用
导入jar文件到工程的WebContent/Web-Inf/lib jstl.jar standard.jar
在jsp页面上,使用taglib 指令,来引入标签库
注意:如果想支持 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>
学生信息管理系统
需求分析
先写 login.jsp , 并且搭配一个LoginServlet 去获取登录信息。
创建用户表, 里面只要有id , username 和 password
创建UserDao, 定义登录的方法
/**
* 该dao定义了对用户表的访问规则
*/
public interface UserDao {
/**
* 这里简单就返回一个Boolean类型, 成功或者失败即可。
* •
* 但是开发的时候,登录的方法,一旦成功。这里应该返回该用户的个人信息
* @param userName
* @param password
* •
* @return true : 登录成功, false : 登录失败。
*/
boolean login(String userName , String password);
}
创建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;
}
}
在LoginServlet里面访问UserDao, 判断登录结果。以区分对待
创建stu_list.jsp , 让登录成功的时候跳转过去。
创建学生表 , 里面字段随意。
定义学生的Dao . StuDao
public interface StuDao {
/**
* 查询出来所有的学生信息
* @return List集合
*/
List<Student> findAll();
}
对上面定义的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; } }
在登录成功的时候,完成三件事情。
查询所有的学生
把这个所有的学生集合存储到作用域中。
跳转到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("用户名或者密码错误!");
}
}
在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
initParamJSTL
使用1.1的版本, 支持EL表达式, 1.0不支持EL表达式
拷贝jar包, 通过taglib 去引入标签库
<c:set>
<c:if>
<c:forEach>
事务&数据库连接池&DBUtils
事务
Transaction 其实指的一组操作,里面包含许多个单一的逻辑。只要有一个逻辑没有执行成功,那么都算失败。所有的数据都回归到最初的状态(回滚)
为什么要有事务?
为了确保逻辑的成功。例子:银行的转账。
使用命令行方式演示事务。
开启事务
start transaction;
提交或者回滚事务
commit; 提交事务, 数据将会写到磁盘上的数据库rollback ; 数据回滚,回到最初的状态。
关闭自动提交功能。
演示事务
使用代码方式演示事务
代码里面的事务,主要是针对连接来的。
通过conn.setAutoCommit(false )来关闭自动提交的设置。
提交事务 conn.commit();
回滚事务 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);
}
}
事务的特性
原子性
指的是 事务中包含的逻辑,不可分割。
一致性
指的是 事务执行前后。数据完整性
隔离性
指的是 事务在执行期间不应该受到其他事务的影响
持久性
指的是 事务执行成功,那么数据应该持久保存到磁盘上。
事务的安全隐患
不考虑隔离级别设置,那么会出现以下问题。
读
脏读 不可重读读 幻读.
* 脏读
> 一个事务读到另外一个事务还未提交的数据
* 不可重复读
> 一个事务读到了另外一个事务提交的数据 ,造成了前后两次查询结果不一致。
读未提交 演示
设置A窗口的隔离级别为 读未提交
两个窗口都分别开启事务
写
丢失更新
读已提交演示
设置A窗口的隔离级别为 读已提交
A B 两个窗口都开启事务, 在B窗口执行更新操作。
在A窗口执行的查询结果不一致。一次是在B窗口提交事务之前,一次是在B窗口提交事务之后。
这个隔离级别能够屏蔽 脏读的现象, 但是引发了另一个问题 ,不可重复读。
可串行化
如果有一个连接的隔离级别设置为了串行化 ,那么谁先打开了事务, 谁就有了先执行的权利, 谁后打开事务,谁就只能得着,等前面的那个事务,提交或者回滚后,才能执行。但是这种隔离级别一般比较少用。容易造成性能上的问题。效率比较低。
按效率划分,从高到低
读未提交 > 读已提交 > 可重复读 > 可串行化
按拦截程度 ,从高到底
可串行化 > 可重复读 > 读已提交 > 读未提交
事务总结
需要掌握的
在代码里面会使用事务
conn.setAutoCommit(false);
conn.commit();
conn.rollback();
事务只是针对连接连接对象,如果再开一个连接对象,那么那是默认的提交。
事务是会自动提交的。
需要了解的
安全隐患
读
脏读
一个事务读到了另一个事务未提交的数据
不可重复读
一个事务读到了另一个事务已提交的数据,造成前后两次查询结果不一致
幻读
一个事务读到了另一个事务insert的数据 ,造成前后查询结果不一致 。
写
丢失更新。
隔离级别
读未提交
引发问题:脏读
读已提交
解决:脏读 , 引发:不可重复读
可重复读
解决:脏读 、 不可重复读 , 未解决:幻读
可串行化
解决:脏读、 不可重复读 、 幻读。
mysql 默认的隔离级别是 可重复读
Oracle 默认的隔离级别是 读已提交
丢失更新
解决丢失更新
悲观锁
可以在查询的时候,加入 for update
乐观锁
要求程序员自己控制。
数据库连接池
数据库的连接对象创建工作,比较消耗性能。
2.一开始现在内存中开辟一块空间(集合) , 一开先往池子里面放置 多个连接对象。后面需要连接的话,直接从池子里面去。不要去自己创建连接了。使用完毕, 要记得归还连接。确保连接对象能循环利用。
自定义数据库连接池
代码实现
出现的问题:
单例。
无法面向接口编程。
UserDao dao = new UserDaoImpl();dao.insert();
需要额外记住 addBack方法
DataSource dataSource = new MyDataSource();
因为接口里面没有定义addBack方法。
怎么解决? 以addBack 为切入点。
解决自定义数据库连接池出现的问题。
由于多了一个addBack 方法,所以使用这个连接池的地方,需要额外记住这个方法,并且还不能面向接口编程。
我们打算修改接口中的那个close方法。原来的Connection对象的close方法,是真的关闭连接。打算修改这个close方法,以后在调用close, 并不是真的关闭,而是归还连接对象。
如何扩展某一个方法?
原有的方法逻辑,不是我们想要的。想修改自己的逻辑
直接改源码 无法实现。
继承, 必须得知道这个接口的具体实现是谁。
使用装饰者模式。
开源连接池
DBCP
导入jar文件
不使用配置文件:
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);
}
}
使用配置文件方式:
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);
查询
直接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());
直接使用框架已经写好的实现类。
* 查询单个对象
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的主要内容,如果未能解决你的问题,请参考以下文章