会话:用户打开一个浏览器,点击了很多超链接,访问多个web资源,关闭浏览器,这个过程可称为会话。
有状态会话:(一个同学来过教室,下次再来教室,大家会知道这个同学曾经来过,称之有状态会话)
一个网站,怎么证明你访问过?
客户端——————服务端
1、服务端给客户端一个信件,客户端下次访问服务端带上信件就可;(cookie)
2、服务器登记客户端访问过,下次访问时来匹配就可;(session)
cookie
session
常见场景:网站登录(如B站),第一次登录,第二次访问就直接登录
简单地说,cookie 就是浏览器储存在用户电脑上的一小段文本文件。cookie 是纯文本格式,不包含任何可执行的代码。一个 Web 页面或服务器告知浏览器按照一定规范来储存这些信息,并在随后的请求中将这些信息发送至服务器,Web 服务器就可以使用这些信息来识别不同的用户。
1、从请求中拿到Cookie信息
2、服务器响应给客户端Cookie
Cookie[] cookies = req.getCookies(); //获得Cookie
cookie.getName()//获得Cookie中的key
cookie.getValue()//获得Cookie中的value
new Cookie("lastLoginTime", System.currentTimeMillis()+"")//新建一个Cookie
cookie.setMaxAge(24*60*60);//设置Cookie的有效期
resp.addCookie(cookie);//响应给客户端一个cookie
cookie:一般会保存在本地的用户目录下appdata
细节问题:
一个网站cookie是否存在上限
删除cookie方式:
测试:查看访问时间
编写Servlet类:
public class CookieDemo01 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//服务器,告诉你,你来的时间,将这个时间封装成一个信件,你下次带来,我就知道你来了
//解决中文乱码
req.setCharacterEncoding("UTF-8");
resp.setCharacterEncoding("UTF-8");
resp.setContentType("text/html;charset=utf-8");
PrintWriter out = resp.getWriter();
//cookie,服务器端从客户端获取
Cookie[] cookies = req.getCookies(); //返回值为数组,说明会返回多个
//判断cookie是否存在
if(cookies!=null){
//如果存在(遍历)
out.write("您第一次访问:");
for (int i = 0; i < cookies.length; i++) {
Cookie cookie = cookies[i];
//获取cookie的名字
if(cookie.getName().equals("lastLoginTime")){
//获取cookie中的值
long lastLoginTime = Long.parseLong(cookie.getValue());
Date date = new Date(lastLoginTime);
out.write(date.toLocaleString());
}
}
}else {
out.write("这是您第一次访问本站");
}
//服务器给客户端发送(响应)一个cookie
Cookie cookie = new Cookie("lastLoginTime", System.currentTimeMillis()+"");
//cookie有效期为1天
cookie.setMaxAge(24*60*60);
resp.addCookie(cookie);
}
在web.xml注册后运行:
测试2:将Cookie立刻过期:
//创建一个Cookie,名字必须和要删除的名字一致
Cookie cookie = new Cookie("lastLoginTime", System.currentTimeMillis()+"");
//将cookie有效期设置为0,立刻过期
cookie.setMaxAge(0);
测试3:将cookie中放入中文
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
resp.setContentType("text/html;charset=utf-8");
Cookie[] cookies = req.getCookies();
PrintWriter out = resp.getWriter();
//判断cookie是否存在
if(cookies!=null){
out.write("您第一次访问:");
for (int i = 0; i < cookies.length; i++) {
Cookie cookie = cookies[i];
//获取cookie的名字
if(cookie.getName().equals("name")){
//System.out.println(cookie.getValue());
//URLDecoder.decode(cookie.getValue(),"utf-8");//解码,以防在工作中遇到乱码
//out.write(cookie.getValue());
out.write(URLDecoder.decode(cookie.getValue(),"utf-8"));
}
}
}else {
out.write("这是您第一次访问本站");
}
Cookie cookie = new Cookie("name", URLEncoder.encode("风在吼","utf-8"));//编码
resp.addCookie(cookie);
}
什么是session?
Session和Cookie的区别:
使Session失效的两种方法:
session.invalidate();//注销之后会出现新的Session
在web.xml中
<session-config>
<session-timeout>15session-timeout>
session-config>
测试:读取Session的ID,并向Session中存放数据:
//得到Session
HttpSession session = req.getSession();
//获取Session的ID
String sessionid = session.getId();
//判断Session是不是新创建的
if (session.isNew()){
resp.getWriter().write("Session创建成功,ID为"+sessionid);
}else{
resp.getWriter().write("Session已经在服务器中,ID为"+sessionid);
}
在web.xml中注册后,运行:
Session存取数据:
新建一个person类
public class Person {
private String name;
private int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
在session中放入数据
//得到Session
HttpSession session = req.getSession();
//给Session中存放数据
session.setAttribute("name",new Person("黄河",5464));
将数据取出
//Session中获取数据
Person person = (Person) session.getAttribute("name");
System.out.println(person.toString());
将session手动注销:
public class SessionDemo03 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
HttpSession session = req.getSession();
session.removeAttribute("name");
//手动注销Session
session.invalidate();//注销之后会出现新的Session
}
}
1、先取为null
2、先存入,在取
3、还可以取对象
使用场景:
JSP全称Java Server Pages,是一种动态网页开发技术。它使用JSP标签在HTML网页中插入Java代码。标签通常以<%开头以%>结束。
JSP是一种Java servlet,主要用于实现Java web应用程序的用户界面部分。网页开发者们通过结合HTML代码、XHTML代码、XML元素以及嵌入JSP操作和命令来编写JSP。
JSP通过网页表单获取用户输入数据、访问数据库及其他数据源,然后动态地创建网页。
JSP标签有多种功能,比如访问数据库、记录用户选择信息、访问JavaBeans组件等,还可以在不同的网页中传递控制信息和共享信息。
最大的特点:
JSP和HTML区别:
学习思路:JSP到底是怎么执行?
JSP处理
以下步骤表明了Web服务器是如何使用JSP来创建网页的:
本机上所体现为:
JSP最终也会被转换成为一个Java类
JSP 本质上就是一个Servlet
//初始化
public void _jspInit() {
}
//销毁
public void _jspDestroy() {
}
//JSPService
public void _jspService(.HttpServletRequest request,HttpServletResponse response)
源码中所做一些功能:
1、判断请求
if (!javax.servlet.DispatcherType.ERROR.equals(request.getDispatcherType())) {
final java.lang.String _jspx_method = request.getMethod();
if ("OPTIONS".equals(_jspx_method)) {
response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
return;
}
if (!"GET".equals(_jspx_method) && !"POST".equals(_jspx_method) && !"HEAD".equals(_jspx_method)) {
response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
response.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "JSP 只允许 GET、POST 或 HEAD。Jasper 还允许 OPTIONS");
return;
}
}
2、内置一些对象
final javax.servlet.jsp.PageContext pageContext; //页面上下文
javax.servlet.http.HttpSession session = null; //session
final javax.servlet.ServletContext application; //applicationContext
final javax.servlet.ServletConfig config; //config
javax.servlet.jsp.JspWriter out = null; //out
final java.lang.Object page = this; //page:当前页
//外部两个对象
HttpServletRequest request //请求
HttpServletResponse response //响应
3、输出页面前增加的代码
response.setContentType("text/html"); //设置响应的页面类型
pageContext = _jspxFactory.getPageContext(this, request, response,
null, true, 8192, true);
_jspx_page_context = pageContext;
application = pageContext.getServletContext();
config = pageContext.getServletConfig();
session = pageContext.getSession();
out = pageContext.getOut();
_jspx_out = out;
4、以上的这些个对象我们可以在JSP页面中直接使用!
例如上面所示的out对象:
因为jsp源码中(如3所示的那些对象)已经存在,我们拿来就可用
用户访问服务器中.jsp页面流程:
用户——使用客户端访问——》服务器(服务器中存在web容器)——》JSP页面——转换(将.JSP页面转换为Java文件)——》xxx_jsp.java文件——编译——》xxx_jsp.class文件——再返回给服务器——》客户端(用户真正拿到的,就是服务器处理完毕的class对象,就是Servlet)
在JSP页面中;
只要是 JAVA代码就会原封不动的输出;
如果是HTML代码,就会被转换为:
out.write("\r\n");
以这样的格式输出到前端
JSP表达式
<%--JSP表达式
作用:用来将程序的输出,输出到客户端
<%= 变量或者表达式%>
--%>
<%= new java.util.Date()%>
测试:
JSP脚本片段
<%--jsp脚本片段--%>
<%
int sum = 0;
for (int i = 1; i <=100 ; i++) {
sum+=i;
}
out.println("Sum="+sum+"
");
%>
脚本片段的再实现
<%
int x = 10;
out.println(x);
%>
这是一个JSP文档
<%
int y = 2;
out.println(y);
%>
<%--换行--%>
<%--在代码嵌入HTML元素--%>
<%
for (int i = 0; i < 5; i++) {
%>
Hello,World <%=i%>
<%
}
%>
JSP声明
<%!
static {
System.out.println("Loading Servlet!");
}
private int globalVar = 0;
public void high(){
System.out.println("进入了方法high!");
}
%>
JSP声明:会被编译到JSP生成Java的类中!其他的,就会被生成到_jspService方法中!
在JSP,嵌入Java代码即可!
<%%> 片段
<%=%> 表达式输出一个值
<%!%> 定义全局
<%--EL表达式:${}--%>
<%--JSP注释--%>
JSP的注释,不会在客户端显示,HTML就会!
JSP指令用来设置整个JSP页面相关的属性,如网页的编码方式和脚本语言。
JSP中的三种指令标签:
指令 | 描述 |
---|---|
<%@ page … %> | 定义网页依赖属性,比如脚本语言、error页面、缓存需求等等 |
<%@ include … %> | 包含其他文件 |
<%@ taglib … %> | 引入标签库的定义 |
<%@ page … %>测试程序:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%--定制错误页面--%>
<%@page errorPage="error/500.jsp" %>
<%--显示的声明这是一个错误页面--%>
<%@ page isErrorPage="true" %>
Title
<%
int i=1/0;
%>
<%--错误页面放入一个图片--%>
<%@ include … %>进行测试:
<%--@include 会将两个页面合二为一--%>
<%@include file="common/header.jsp"%>
网页主体
<%@include file="common/footer.jsp"%>
<%--JSP标签
jsp:include :拼接页面,本质还是三个
--%>
<%--要把路径写对--%>
网页主体
下面是源码:
jsp标签所放的东西,会在jspService方法外声明
@include file=“common/header.jsp”,会直接out.write()输出
<%--内置对象--%>
<%
pageContext.setAttribute("type1","风1");//保存的数据只在一个页面中有效
request.setAttribute("type2","风2"); //保存的数据只在一次请求中有效,请求转发会携带这个数据
session.setAttribute("type3","风3"); //保存的数据只在一次会话中有效(打开浏览器到关闭浏览器)
application.setAttribute("type4","风4");//保存的数据在服务器中有效(打开服务器到关闭)
%>
测试:
<%--通过pageContext取出保存的值--%>
<%--注意:脚本片段中的代码,会被原封不动的生成到.jsp.java
所以要求:这里面的代码必须保证Java语法的正确性
--%>
<%
//通过寻找方式来
//从底层到高层(作用域):page->request->session->application
String type1 = (String) pageContext.findAttribute("type1");
String type2 = (String) pageContext.findAttribute("type2");
String type3 = (String) pageContext.findAttribute("type3");
String type4 = (String) pageContext.findAttribute("type4");
String type5 = (String) pageContext.findAttribute("type5");//不存在
%>
<%--使用EL表达式输出 ${}--%>
取出的值为:
${type1}
${type2}
${type3}
${type4}
${type5}
<%--EL表达式与普通查找语句对比--%>
<%=type5%>
<%--null--%>
从另一个页面测试取出上述的值:(取值代码与前面一致)
所需导的依赖:
<dependency>
<groupId>javax.servlet.jsp.jstlgroupId>
<artifactId>jstl-apiartifactId>
<version>1.2version>
dependency>
<dependency>
<groupId>taglibsgroupId>
<artifactId>standardartifactId>
<version>1.1.2version>
dependency>
EL表达式: ${ }
测试:
<%--
在转发的时候携带参数:http://localhost:8080/jsptag.jsp?name=xiaohong&age=11
--%>
尝试取出参数:
<%=request.getParameter("name")%>
<%=request.getParameter("age")%>
JSTL标签库的使用就是为了弥补HTML标签的不足;它自定义许多标签,可以供我们使用,标签的功能和Java代码一样!
根据JSTL标签所提供的功能,可以将其分为5个类别:
格式化标签
SQL标签
XML 标签
核心标签 (掌握部分)
JSTL 函数
JSTL标签库使用步骤
解决报错方法:
将jstl和standard的jar包,从maven环境中放入tomcat服务器的lib中
栗子:if测试
if测试
<%--判断如果提交的用户名是admin,则登录成功--%>
when测试栗子:
<%--定义一个变量score,值为70--%>
你的成绩为优秀
你的成绩为一般
你的成绩为良好
你的成绩为不及格
测试foreach循环:
<%
ArrayList people = new ArrayList<>();
people.add(0,"张三");
people.add(1,"李四");
people.add(2,"王五");
people.add(3,"赵六");
people.add(4,"田六");
request.setAttribute("list",people);
%>
<%--
var , 每一次遍历出来的变量
items, 要遍历的对象
begin, 哪里开始
end, 到哪里
step, 步长
--%>
设置begin end step后
通常称之为:实体类
JavaBean有特定的写法:
一般用来和数据库的字段做映射 ORM;
ORM :对象关系映射
一个JavaBean对象的属性应该是可访问的。这个属性可以是任意合法的Java数据类型,包括自定义Java类。
一个JavaBean对象的属性可以是可读写,或只读,或只写。 getPropertyName() 和 setPropertyName() 两个方法
进行测试:
先写一个Student.java文件
public class Student {
private String name=null;
private int age = 0;
public Student() {
}
public Student(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
Student Name:
Student Age:
得到结果:
MVC概念:Model view Controller 模型、视图、控制器
早些年,架构为:二层
用户直接访问控制层,控制层就可以直接操作数据库;
servlet--CRUD-->数据库
//将增删改茶的代码要放入Servlet里
//弊端:程序十分臃肿,不利于维护
//于是:
//Servlet的代码中会有:处理请求、响应、视图跳转、处理JDBC、处理业务代码、处理逻辑代码
Model
View
Controller (Servlet)
在JSP/Servlet开发的软件系统中,这三个部分的描述如下所示:
1、Web浏览器发送HTTP请求到服务端,然后被Controller(Servlet)获取并进行处理(例如参数解析、请求转发)
2、Controller(Servlet)调用核心业务逻辑——Model部分,获得结果
3、Controller(Servlet)将逻辑处理结果交给View(JSP),动态输出HTML内容
4、动态生成的HTML内容返回到浏览器显示
MVC模式在Web开发中有很大的优势,它完美规避了JSP与Servlet各自的缺点,让Servlet只负责业务逻辑部分,而不会生成HTML代码;同时JSP中也不会充斥着大量的业务代码,这样能大提高了代码的可读性和可维护性。
Filter:过滤器 ,用来过滤网站的数据;
Filter开发步骤:
1、导包:在pom.xml添加依赖
2、编写过滤器
实现Filter接口,Filter所在的包为javax.servlet
重写对应的方法
public class CharacterEncodingFilter implements Filter {
//初始化:web服务器启动,就已经初始化了,随实等待过滤对象出现
public void init(FilterConfig filterConfig) throws ServletException {
System.out.println("CharacterEncodingFilter初始化");
}
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
request.setCharacterEncoding("utf-8");
response.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=UTF-8");
System.out.println("CharacterEncodingFilter执行前");
chain.doFilter(request,response);//让请求执行,如果不写,程序到这里就停止
System.out.println("CharacterEncodingFilter执行后");
}
//销毁:服务器关闭的时候才会销毁
public void destroy() {
System.out.println("CharacterEncodingFilter销毁");
}
}
编写一个显示类
public class ShowServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.getWriter().write("开心");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>ShowServletservlet-name>
<servlet-class>com.XX.servlet.ShowServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>ShowServletservlet-name>
<url-pattern>/servlet/showurl-pattern>
servlet-mapping>
<filter>
<filter-name>CharacterEncodingFilterfilter-name>
<filter-class>com.XX.filter.CharacterEncodingFilterfilter-class>
filter>
<filter-mapping>
<filter-name>CharacterEncodingFilterfilter-name>
<url-pattern>/servlet/*url-pattern>
filter-mapping>
没有配置Filter 的时候,ShowServlet页面上的字为乱码
配置之后:
进入几次这个页面就会在IDEA控制台会输出几次:
当关闭服务器时,Filter才会注销
实现一个监听器的接口;
对监听器进行使用,测试:统计当前网上在线人数:
1、实现监听接口
2、进行代码实现(主要思想是统计session个数)
public class OnlineCountListener implements HttpSessionListener {
//创建session监听
//一旦创建Session就会触发一次这个
public void sessionCreated(HttpSessionEvent se) {
ServletContext ctx = se.getSession().getServletContext();
System.out.println(se.getSession().getId());
Integer onlineCount = (Integer) ctx.getAttribute("OnlineCount");
if (onlineCount==null){
onlineCount=new Integer(1);
}else{
int count = onlineCount.intValue();
onlineCount = new Integer(count+1);//count+1为count++
}
ctx.setAttribute("OnlineCount",onlineCount);
}
//销毁session监听
public void sessionDestroyed(HttpSessionEvent se) {
ServletContext ctx = se.getSession().getServletContext();
Integer onlineCount = (Integer) ctx.getAttribute("OnlineCount");
if (onlineCount==null){
onlineCount=new Integer(0);
}else{
int count = onlineCount.intValue();
onlineCount = new Integer(count-1);//count+1为count++
}
ctx.setAttribute("OnlineCount",onlineCount);
}
}
在首页index.jsp上编写显示程序
当前有<%=this.getServletConfig().getServletContext().getAttribute("OnlineCount")%> 人在线
在web.xml中进行注册
<listener>
<listener-class>com.feng.listener.OnlineCountListenerlistener-class>
listener>
点击运行
会出现页面上所显示人数与实际不一致,由于服务器原因,例如我这里出现3个SessionId:
只有最后一个才是浏览器中的。
再点击运行按钮,进行重新部署redeploy
控制台上就出现一个sessionid
浏览器处:
测试:用户登录之后才能进入主页,用户注销之后就不能进入主页
1、编写一个登录页面(Login.jsp),错误页面(error.jsp),登陆成功后的页面(success.jsp)
2、提交按钮转到LoginServlet类中做判断,用户名是否和程序中所写一致,进入主页,否则进入编写的错误页面
public class LoginServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获取前端请求的参数
String username = req.getParameter("username");
if (username.equals("admin")){
req.getSession().setAttribute("USER_SESSION",req.getSession().getId());
resp.sendRedirect("/sys/success.jsp");
}else{//登录失败
resp.sendRedirect("/error.jsp");
}
}
3、此时在未登录成功的情况下,就能直接访问/Sys/success.jsp这个主页,需要解决,
方法一:比较low不推荐:在需要登录之后才能访问的页面上进行限制。
<%
Object userSession = request.getSession().getAttribute("USER_SESSION");
if (userSession==null){
response.sendRedirect("/Login.jsp");
}
%>
推荐解决方法二:使用过滤器!编写SysFilter类,这样做到”各司其职“。
public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws IOException, ServletException {
//解决方法二:用过滤器,这样各自职责分明
//ServletRequest HttpServletRequest 父子关系,进行强转
HttpServletRequest request = (HttpServletRequest) req;
HttpServletResponse response = (HttpServletResponse) resp;
Object userSession = request.getSession().getAttribute("USER_SESSION");
if (userSession==null){
response.sendRedirect("/error.jsp");
}
chain.doFilter(request,response);
}
此时,不登陆的前提下直接访问/sys/success.jsp页面就会转发至error页面