※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")取得