※1※数据库的存储过程(Stored Procedure):
参考链接:https://www.imooc.com/article/19811
※什么是数据库的存储过程:
是一组为了完成特定功能的SQL语句集,经编译后存储在数据库中,用户通过指定存储过程的名字并给出参数(如果该存储过程带有参数)来执行它。
※为什么要使用存储过程??(存储过程的好处/优点)
(1)实现了模块化编程。
(2)调用一次以后,相关信息就保存在内存中,下次调用时可以直接执行。
(3)存储过程可以接受输入参数并可以返回输出值。
(4)存储过程具有对数据库立即访问的功能。
(5)使用存储过程可以加快程序的运行速度。
(6)使用存储过程可以减少网络流量。
(7)使用存储过程可以提高数据库的安全性。
※存储过程的分类:
(1)数据库系统存储过程:数据库自带
(2)用户自定义数据库存储过程:由用户根据需要自定义
※2※谈谈你理解的事务:
参考链接:https://blog.csdn.net/JIESA/article/details/51317164
※事务的特性(ACID):对于一个事务必须遵守ISO/IEC所定制的ACID原则
Atomicity(原子性)、Consistency(一致性)、Isolation(隔离性)、Durability(持久性)
※事务产生的原因:
由事务的隔离特性,多事务同时并发执行对同一数据库的记录进行操作的时候,会产生如下问题:
♩:数据丢失:第一次提交的数据被第二次提交的数据覆盖,导致数据丢失
♩:不可重复读两个并发的事务,“事务A:singo消费”、“事务B:singo的老婆网上转账”,事务A事先读取了数据,事务B紧接了更新了数据,并提交了事务,而事务A再次读取该数据时,数据已经发生了改变。
♩:幻读:singo的老婆工作在银行部门,她时常通过银行内部系统查看singo的信用卡消费记录。有一天,她正在查询到singo当月信用卡的总消费金额 (select sum(amount) from transaction where month = 本月)为80元,而singo此时正好在外面胡吃海塞后在收银台买单,消费1000元,即新增了一条1000元的消费记录(insert transaction ... ),并提交了事务,随后singo的老婆将singo当月信用卡消费的明细打印到A4纸上,却发现消费总额为1080元,singo的老婆很诧异,以为出 现了幻觉,幻读就这样产生了。
♩:脏读:指当一个事务正在访问数据,并且对数据进行了修改,而这种修改还没有提交到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据。
※事务隔离级别(由低到高):

※3※如何实现对未登录网页的数据统计:
参考链接:https://blog.csdn.net/yongh701/article/details/43969143
※利用Application对象实现访问人数统计功能:
问题:当服务器重启时,由于Application会被销毁,重新建立,则人数统计会清零
解决:统计数据人数是添加数据库功能,这样无论服务器需要重启或者遭遇停机,不会造成访问人数的丢失
♫:统计一个页面的访问人数,而且这个访问人数,在服务器重启之后不会丢失

基本准备:
♩:在数据库中添加字段用于记录访问人数,无主键,不自增,默认为零
♩:添加MySql依赖
♩:在web.xml中添加如下字段:
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<context-param> <--. 启动一个WEB项目的时候,容器(如:Tomcat)会去读它的配置文件web.xml.读两个节点: <listener></listener> 和 <context-param></context-param>-->
<param-name>url</param-name>
<param-value>jdbc:mysql://localhost:3306/test?useUnicode=true&amp;characterEncoding=utf8&amp;useOldAliasMetadataBehavior=true</param-value>
</context-param>
<context-param>
<param-name>user</param-name>
<param-value>root</param-value>
</context-param>
<context-param>
<param-name>pwd</param-name>
<param-value>root</param-value>
</context-param>
</web-app>
这样url,user,root就是我们连接数据库的全局变量
♩:连接数据库
♩:注意:整个网站工程的所有页面,利用application.getInitParameter("")方法获取这些全局变量
这里值得注意的是,原本的&号,必须写成转义字符&amp;否则web.xml会把&当作一个编译符号,整个网络工程无法在tomcat服务器中启动。
♩:编写JSP页面:
1 <%@ page language="java" contentType="text/html; charset=utf-8"
2 pageEncoding="utf-8"%>
3 <%@ page import="java.sql.*"%>
4 <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
5 <html>
6 <head>
7 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
8 <title>index</title>
9 </head>
10 <body>
11 <%
12 //取出web.xml中的全局变量
13 String url=application.getInitParameter("url");
14 String user=application.getInitParameter("user");
15 String pwd=application.getInitParameter("pwd");
16 //连接数据库,把访问人数+1
17 String sql = null;
18 Class.forName("com.mysql.jdbc.Driver");
19 Connection con=DriverManager.getConnection(url,user,pwd);
20 sql = "update visitcount set visitCount=visitCount+1";
21 con.createStatement().execute(sql);
22 sql = "select visitCount from visitcount";
23 ResultSet rs=con.prepareStatement(sql).executeQuery();
24 //这里由于整个数据库表只有一行,所以无需用到while循环来取
25 //只需要把指向第0行的游标向下移一个位置就可以了
26 rs.next();
27 //把这个变量写到application
28 application.setAttribute("counter",String.valueOf(rs.getInt("visitCount")));
29 //再把这个变量取出来,让访问网站的所有人看到
30 out.print("你是第"+application.getAttribute("counter")+"个访问本站的人!");
31 %>
32 </body>
33 </html>
※4※JSP的内置对象有哪些:
参考链接:https://blog.csdn.net/bighuan/article/details/68925746
1 内置对象名 类型 四大域对象 域作用范围 2 request HttpServletRequest request域 3 response HttpServletResponse 4 config ServletConfig 5 application ServletContext context域 同一个web应用中使用 6 session HttpSession session域 只能在同一个会话使用 7 exception Throwable 8 page Object(this) 9 out JspWriter 10 pageContext PageContext page域 只能在当前jsp页面使用
※5※Filter的生效时机:
过滤器(Filter)、拦截器(Interceptor)、监听器(Listenter)
参考链接:https://www.cnblogs.com/CloverSH/p/4531492.html
※web 应用程序启动时,web 服务器将创建Filter 的实例对象,并调用其init方法,读取web.xml配置,完成对象的初始化功能,从而为后续的用户请求作好拦截的准备工作(filter对象只会创建一次,init方法也只会执行一次)。
※6※<context-param>与<init-param>的区别与作用
参考链接:https://www.cnblogs.com/hzj-/articles/1689836.html
※在启动一个WEB项目的时候,
♩:容器(如:Tomcat)会去读取它的配置文件web.xml中的两个节点:<listener></listener>和<context-param></context-param>
♩:容器创建一个ServletContext(上下文),这个WEB项目所有的部分都将共享这个上下文
♩:容器将<context-param></context-param>转化为键值对,并交给ServletContext
♩:容器创建<listener></listener>中的类实例,即创建监听
♩:在监听中会有contextInitalized(ServletContextEvent args)初始化方法,在这个方法中获得ServletContext = ServletContextEvent.getServletContext();
context-param的值 = ServletContext.getInitParameter("context-param的键");
♩:得到这个context-param的值后,你就可以做一些操作,注意这个时候你的WEB项目还没有完全启动,这个动作会比所有的Servlet都要早。
♫:在启动项目之前就对数据库进行连接
在<context-param>中设置数据库的连接方式,在监听类中初始化数据库的连接。
这个监听是自己写的一个类,除了初始化方法,还有销毁方法,用于关闭应用前释放资源,比如对数据库的关闭
1 <!-- 加载spring的配置文件 -->
2 <context-param>
3 <param-name>contextConfigLocation</param-name>
4 <param-value>/WEB-INF/applicationContext.xml,/WEB-INF/action-servlet.xml,/WEB-
5 INF/jason-servlet.xml</param-value>
6 </context-param>
7 <listener>
8 <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
9 </listener>
10 <--又如: --->自定义context-param,且自定义listener来获取这些信息-->
11 <context-param>
12 <param-name>urlrewrite</param-name>
13 <param-value>false</param-value>
14 </context-param>
15 <context-param>
16 <param-name>cluster</param-name>
17 <param-value>false</param-value>
18 </context-param>
19 <context-param>
20 <param-name>servletmapping</param-name>
21 <param-value>*.bbscs</param-value>
22 </context-param>
23 <context-param>
24 <param-name>poststoragemode</param-name>
25 <param-value>1</param-value>
26 </context-param>
27 <listener>
28 <listener-class>com.laoer.bbscs.web.servlet.SysListener</listener-class>
29 </listener>
30 public class SysListener extends HttpServlet implements ServletContextListener {
31 private static final Log logger = LogFactory.getLog(SysListener.class);
32 public void contextDestroyed(ServletContextEvent sce) {
33 //用于在容器关闭时,操作
34 }
35 //用于在容器开启时,操作
36 public void contextInitialized(ServletContextEvent sce) {
37 String rootpath = sce.getServletContext().getRealPath("/");
38 System.out.println("-------------rootPath:"+rootpath);
39 if (rootpath != null) {
40 rootpath = rootpath.replaceAll("\\\\", "/");
41 } else {
42 rootpath = "/";
43 }
44 if (!rootpath.endsWith("/")) {
45 rootpath = rootpath + "/";
46 }
47 Constant.ROOTPATH = rootpath;
48 logger.info("Application Run Path:" + rootpath);
49 String urlrewrtie = sce.getServletContext().getInitParameter("urlrewrite");
50 boolean burlrewrtie = false;
51 if (urlrewrtie != null) {
52 burlrewrtie = Boolean.parseBoolean(urlrewrtie);
53 }
54 Constant.USE_URL_REWRITE = burlrewrtie;
55 logger.info("Use Urlrewrite:" + burlrewrtie);
56 其它略之....
57 }
58 }
59 /*最终输出
60 -------------rootPath:D:\tomcat_bbs\webapps\BBSCS_8_0_3\
61 2009-06-09 21:51:46,526 [com.laoer.bbscs.web.servlet.SysListener]-[INFO]
62 Application Run Path:D:/tomcat_bbs/webapps/BBSCS_8_0_3/
63 2009-06-09 21:51:46,526 [com.laoer.bbscs.web.servlet.SysListener]-[INFO]
64 Use Urlrewrite:true
65 2009-06-09 21:51:46,526 [com.laoer.bbscs.web.servlet.SysListener]-[INFO]
66 Use Cluster:false
67 2009-06-09 21:51:46,526 [com.laoer.bbscs.web.servlet.SysListener]-[INFO]
68 SERVLET MAPPING:*.bbscs
69 2009-06-09 21:51:46,573 [com.laoer.bbscs.web.servlet.SysListener]-[INFO]
70 Post Storage Mode:1
71 */
※WEB.xml里面可以定义两种参数:
♩:application范围的参数,存放在servletContext中,在web.xml中配置如下:
1 <context-param> 2 <param-name>context/param</param-name> 3 <param-value>avalible during application</param-value> 4 </context-param>
♩:servlet范围内的参数,只能在servlet的init()方法中取得,在web.xml中配置如下:
1 <servlet> 2 <servlet-name>MainServlet</servlet-name> 3 <servlet-class>com.wes.controller.MainServlet</servlet-class> 4 <init-param> 5 <param-name>param1</param-name> 6 <param-value>avalible in servlet init()</param-value> 7 </init-param> 8 <load-on-startup>0</load-on-startup> 9 </servlet>
在servlet中可以通过代码分别取用:
1 package com.wes.controller;
2 import javax.servlet.ServletException;
3 import javax.servlet.http.HttpServlet;
4 public class MainServlet extends HttpServlet ...{
5 public MainServlet() ...{
6 super();
7 }
8 public void init() throws ServletException ...{
9 System.out.println("下面的两个参数param1是在servlet中存放的");
10 System.out.println(this.getInitParameter("param1"));
11 System.out.println("下面的参数是存放在servletcontext中的");
12 System.out.println(getServletContext().getInitParameter("context/param"));
13 }
14 }
第一种参数在servlet里面可以通过getServletContext().getInitParameter("context/param")得到
第二种参数只能在servlet的init()方法中通过this.getInitParameter("param1")取得