1. 基本概念
- 静态web:
- HTML、CSS
- 提供给所有人的数据始终不会变化
- 轮播图,伪动态
- 缺点:无法和数据库交互,数据无法持久化,用户无法交互
- 动态web:
- 技术栈:Servlet/JSP、ASP、PHP
- 每个人在不同的时间、地点看到的信息各不相同
- 缺点:动态web资源出现错误,需要重新编写后台程序
- web应用程序——Tomcat服务器:
- 组成部分:HTML、CSS、JS;JSP、Servlet;Java程序;jar包;配置文件(properties)
2. Web服务器
2.1 技术讲解
B/S:浏览和服务器;C/S:客户端和服务器
- ASP:
- 微软,国内最早流行,在HTML中嵌入了VB的脚本,ASP+COM
- 基本一个页面都有几千行的业务代码,维护成本高
- C#
- IIS
- PHP:
- 开发速度很快,功能很强大,跨平台,代码很简单
- 无法承载大访问量的情况(局限)
- JSP/Servlet:
- Sun公司主推的B/S架构
- 基于Java语言
- 可以承载三高问题带来的影响
- 语法像ASP
2.2 Web服务器
服务器是一种被动的操作,用来处理用户的请求并返回响应。
- IIS:
- 微软,ASP,Windows中自带的
Tomcat
默认端口号:Tomcat/8080;MySQL/3306;http/80;https/443
面试:网站是如何进行访问的?
- 输入一个域名;
- 客户端查询本机的C:\Windows\System32\drivers\etc\hosts配置文件有没有这个域名的映射:有则直接返回对应的IP地址;没有则通过交换机去DNS服务器找(不在本机);
2.3 Web网站发布
- webapps:
- ROOT
- test:网站的目录名
- WEB-INF
- classes:Java程序
- lib:web应用依赖的jar包
- web.xml:网站配置文件
- index.html/index.jsp:默认的首页
- static
- css
- js
- img
- ……
3. HTTP
超文本传输协议
请求行:
请求方式——
GET:请求能够携带的参数比较少,大小有限制,会在浏览器的URL地址栏显示数据内容,不安全,但高效;
POST:请求能够携带的参数没有限制,大小没有限制,不会在浏览器的URL地址栏显示数据内容,安全但不高效。
消息头:
Accept:告诉浏览器,它所支持的数据类型
Accept-Encoding:编码格式:GBK、UTF-8、GB2312、ISO8859-1
Accept-Language:告诉浏览器,它的语言环境
Cache-Control:缓存控制
Connection:告诉浏览器,请求完成时断开还是保持连接
响应状态码:
3XX:重定向304
4XX:找不到资源404
5XX:服务器代码错误500;网关错误502
面试:当你的浏览器地址栏输入地址并回车的一瞬间到页面能够展示,经历了什么?
4. Maven详解
Maven的高级之处在于它会帮你导入这个jar包依赖的其他jar包。
Maven由于约定大于配置,可能存在的问题是自己写的配置文件无法被导出或者生效,解决方案:在build中配置resources防止资源导出失败。
#POM.xml
<!--Package:项目的打包方式:
jar:java应用
war:Javaweb应用
-->
<build>
<resources>
<resource>
<directory>src/main/resources</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>true</filtering>
</resource>
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>true</filtering>
</resource>
</resources>
</build>
5. Servlet详解
Servlet就是sun公司开发动态web的一门技术。Sun在这些API中提供一个接口叫做Servlet,如果你想开发一个Servlet程序,只需要完成两个步骤:
- 编写一个类实现Servlet接口;
- 把开发好的Java类部署到web服务器中。
把实现了Servlet接口的Java程序叫做Servlet。
Servlet是由web服务器调用,web服务器收到浏览器请求后会调用Servlet的Service方法,该方法由我们自己编写的实现类重写,即接收并处理请求后给出响应信息,然后web容器(Tomcat)读取响应信息。
5.1 HelloServlet
- 构建一个普通的Maven项目,删掉src目录,后续在该项目里建立Module,这个空的工程就是Maven主工程;
- 导入依赖至主工程POM;
- 关于Maven父子工程的理解:父项目中会有module,子项目中会有parent;父项目中的jar包,子项目可以直接使用,反之则不能。
- Maven环境优化:修改web.xml为最新的;将maven的结构搭建完整。
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee
http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0"
metadata-complete="true">
</web-app>
- 编写一个servlet程序:编写一个普通类;实现servlet接口继承HttpServlet;
Servlet接口Sun公司有两个默认的实现类:HttpServlet、
- 编写Servlet的映射。为什么需要映射:所写Java程序需要通过浏览器访问,而浏览器需要连接web服务器,所以我们需要在web服务中注册我们写的servlet,还需要给他一个浏览器能够访问。
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.servlet.servlet01.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
- 配置tomcat,配置项目发布的路径。
5.2 Mapping问题
- 一个Servlet可以指定一个映射路径;
- 一个Servlet可以指定多个映射路径;
- 一个Servlet可以指定通用映射路径;
- 默认请求路径/*;
- 可以自定义前缀、后缀实现请求映射,注意:*前面不能加项目映射的路径;
- 优先级问题,指定了固有的映射路径优先级最高,如果找不到才会走默认的处理请求。
5.3 ServletContext
web容器在启动的时候,它会为每个web程序都创建一个对应的ServletContext对象,它代表了当前的web应用;
5.3.1 共享数据
我在这个Servlet中保存的数据,可以在另一个Servlet中共享
//.....
ServletContext context = this.getServletContext();
String username = "xx";
context.setAttribute("username",username);
//.....
ServletContext context = this.getServletContext();
String username = (String)context.getAttribute("username");
resp.setContentType("text/html");
resp.setCharacterEncoding("utf-8");
resp.getWriter().print("名字:"+username);
//配置XML
5.3.2 获取初始化参数及请求转发
<!--配置一些web应用初始化参数-->
<context-param>
<param-name>url</param-name>
<param-value>jdbc:mysql://localhost:3306/mybatis</param-value>
</context-param>
//......获取初始化参数
ServletContext context = this.getServletContext();
String url = context.getInitParameter("url");
resp.getWriter().print(url);
//......请求转发
ServletContext context = this.getServletContext();
context.getRequestDispatcher("/gp").forward(req,resp);
转发路径不会变;重定向路径会变。
5.3.3 读取资源文件
Properties类,以往使用绝对路径,但在web服务器只能使用相对路径。
在java、resources目录下新建properties文件,打包后统一在target的classes目录下,俗称该路径为classpath;需要一个文件流。
username=root
password=123456
InputStream is = this.getServletContext().getResourceAsStream("/WEB-INF/classes/db.properties");
Properties prop = new Properties();
prop.load(is);
String username = prop.getProperty("username");
String password = prop.getProperty("password");
5.4 HttpServletResponse
响应:Web服务器接收到客户端的HTTP请求,会针对这个请求分别创建一个代表请求的HttpServletRequest对象,代表响应的一个HttpServletResponse;如果要获取客户端请求过来的参数找HttpServletRequest,如果要给客户端响应一些信息找HttpServletResponse。
5.4.1 简单分类
负责向浏览器发送数据的方法:
ServletOutputStream getOutputStream() throws IOException;//
PrintWriter getWriter() throws IOException;//中文
负责向浏览器发送响应头的方法:
void setCharacterEncoding(String varl);
void setStatus(int varl);
//……
响应的状态码:
200、300、404、500等
5.4.2 常见应用
- 向浏览器输出消息——resp.getWriter().print();
- 下载文件——
- 获取下载文件的路径;
- 下载的文件名是啥;
- 想办法让浏览器能够支持下载我们需要的东西;
- 获取下载文件的输入流;
- 创建缓冲区;
- 获取OutputStream对象;
- 将FileOutputStream流写入到buffer缓冲区;
- 使用OutputStream将缓冲区中的数据输出到客户端。
- 验证码功能——
- 前端实现;
- 后端实现,需要用到Java的图片类,生成一个图片:
- 如何让浏览器5秒自动刷新一次;
- 在内存中创建一个图片;
- 得到图片;
- 设置图片的背景颜色;
- 给图片写数据;
- 告诉浏览器,这个请求用图片的方式打开;
- 网站存在缓存,不让浏览器缓存;
- 把图片写给浏览器。
- 实现重定向—— 一个web资源收到客户端请求后,它会通知客户端去访问另外一个web资源,常见场景:用户登录成功
/*resp.setHeader("Location","/response/img");
resp.setStatus(HttpServletResponse.SC_MOVED_TEMPORARILY);*/
void sendRedirect(String varl) throws IOException;
面试题:请聊聊重定向和转发的区别?
相同点:实现页面跳转
不同点:请求转发的时候,url不会变化,307;重定向会变化,302。
5.5 HttpServletRequest
HttpServletRequest代表客户端的请求,用户通过HTTP协议访问服务器,HTTP请求中的所有信息会被封装到HttpServletRequest,通过这个HttpServletRequest的方法,获得客户端的所有信息。
- 获取前端传递的参数——req.getParameter();req.getParameterValues();
- 请求转发——
req.setCharacterEncoding("utf-8");//处理后台接收中文乱码问题
resp.setCharacterEncoding("utf-8");req.getRequestDispatcher("/success.jsp").forward(req,resp);
//这里的/代表当前的web应用
6. Cookie、Session
6.1 会话
会话: 用户打开一个浏览器,点击了很多超链接,访问多个web资源,关闭浏览器,这个过程可以称之为会话;
有状态会话: 服务端给客户端一个信件,客户端下次访问服务端带上信件即可(cookie);服务器登记你来过了,下次来会自动匹配(session)。
6.2 保存会话的两种技术
cookie
——客户端技术(响应,请求)
session
——服务器技术,利用这个技术可以保存用户的会话信息,我们可以把信息或者数据放在session中
常见场景
网站登录后,第二次访问直接进入;
6.3 Cookie
- 从请求中获得cookie信息;
- 服务器响应给客户端cookie;
//Cookie,服务器端从客户端获取
Cookie[] cookies = req.getCookies();//获得cookie
cookie.getName();//获得cookie中的key
cookie.getValue();//获得cookie中的Value
new Cookie("lastLoginTime", System.currentTimeMillis()+"");//新建一个cookie
//Cookie有效期为一天
cookie.setMaxAge(24*60*60);//设置Cookie有效期
resp.addCookie(cookie);
cookie一般会保存在本地的用户目录下appdata;
一个网站cookie是否存在上限?
- 一个Cookie只能保存一个信息;
- 一个web站点可以给浏览器发送多个cookie,最多存放20个cookie;
- Cookie大小有限制4kb;
- 300个cookie浏览器上限。
删除Cookie:
- 不设置有效期,关闭浏览器,自动失效;
- 设置有效期时间为0。
传递中文字符:
URLEncoder.encode("你好","utf-8");//编码
URLDecoder.decode(cookie.getValue(),"utf-8");//解码
6.4 Session(重点)
服务器会给每一个**用户(浏览器)**创建一个Session对象,一个Session独占一个浏览器,只要浏览器没关,这个Session就存在。用户登录之后,整个网站都可以访问。——>保存用户的信息;保存购物车的信息。
Cookie和Session的异同:
Cookie是把用户的数据写给用户的浏览器,浏览器保存(可以保存多个);
Session把用户的数据写到用户独占Session中,服务器端保存(保存重要的信息,减少服务器资源的浪费);
Session对象由服务创建;
Session建议保存一个登录用户的信息;购物车信息;在整个网站中经常会使用的数据。
cookie存本地;session存服务器;servletcontext存全局。
HttpSession session = req.getSession();
session.removeAttribute("name");
session.invalidate();//手动注销session
<!--设置session默认的失效时间-->
<session-config>
<!--15分钟后session自动失效,以分钟为单位-->
<session-timeout>15</session-timeout>
</session-config>
7. JSP
7.1 JSP原理剖析
Java Server Pages:Java服务器端页面,也和servlet一样,用于动态web技术。
HTML只提供静态数据,JSP可以嵌入Java代码,为用户提供动态数据。
浏览器向服务器发送请求,不管访问什么资源,其实都是在访问Servlet,JSP最终也会被转换成为一个Java类。
JSP本质上就是一个Servlet。
JAVA代码原封不动输出到前端,HTML代码则用out.write()输出。
<dependencies>
<!--Servlet依赖-->
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<!--JSP依赖-->
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>javax.servlet.jsp-api</artifactId>
<version>2.3.3</version>
</dependency>
<!--JSTL表达式的依赖-->
<dependency>
<groupId>javax.servlet.jsp.jstl</groupId>
<artifactId>jstl-api</artifactId>
<version>1.2</version>
</dependency>
<!--standard标签库-->
<dependency>
<groupId>taglibs</groupId>
<artifactId>standard</artifactId>
<version>1.1.2</version>
</dependency>
</dependencies>
JSP声明<%!%>会被编译到JSP生成Java的类中。其他的<%=%>和<%%>会被生成到_jspService方法中。
JSP的注释不会在客户端显示,HTML则会。
<!--我是HTML的注释-->
<%--我是JSP的注释--%>
<%@JSP指令%>
<%--@include会将两个页面合二为一--%>
<%@include file="common/header.jsp"%>
<h1>网页主体</h1>
<%@include file="common/footer.jsp"%>
<%--jsp标签:拼接页面,本质还是三个,不会报错--%>
<jsp:include page="common/header.jsp"/>
<h1>网页主体</h1>
<jsp:include page="common/footer.jsp"/>
注意:
项目中新建文件夹放在web目录下,static:css、img、js、plugins等;
WEB-INF对用户不可见。
7.2 九大内置对象
- PageContext 存东西
- Request 存东西;
- 客户端向服务器发送请求,产生的数据,用户看完就没用了,比如新闻,看完没用
- Response
- Session 存东西;
- 客户端向服务器发送请求,产生的数据,用户用完一会还有用,比如购物车
- Application 【ServletContext】存东西;
- 客户端向服务器发送请求,产生的数据,一个用户用完了,其他用户还可能使用,比如聊天数据
- config 【ServletConfig】
- out
- page 几乎不用
- exception
<%
//作用域从底层到高层:page->request->session->application
//类似JVM:双亲委派机制
pageContext.setAttribute("name","zz");//保存数据只在一个页面有效
request.setAttribute("name","zz");//保存数据在一次请求中有效,请求转发会携带这个数据
session.setAttribute("name","zz");//保存数据只在一次会话中有效,从打开浏览器到关闭浏览器
application.setAttribute("name","zz");//保存数据只在服务器中有效,从打开服务器到关闭服务器
%>
<%
//请求转发
pageContext.forward("/index.jsp");//前端
request.getRequestDispatcher("/index.jsp").forward(request,response);//后端
%>
7.3 JSP标签、JSTL标签、EL表达式
在Tomcat中也需要引入JSTL的包,否则会报错:JSTL解析错误。
<!--JSTL表达式的依赖-->
<dependency>
<groupId>javax.servlet.jsp.jstl</groupId>
<artifactId>jstl-api</artifactId>
<version>1.2</version>
</dependency>
<!--standard标签库-->
<dependency>
<groupId>taglibs</groupId>
<artifactId>standard</artifactId>
<version>1.1.2</version>
</dependency>
EL表达式:${}
- 获取数据;
- 执行运算;
- 获取web开发的常用对象。
JSP标签:
<%--
等同于
http://localhost:8080/jsptag.jsp?name=kk&age=12
--%>
<jsp:forward page="/jsptag2.jsp">
<jsp:param name="name" value="kk"></jsp:param>
<jsp:param name="age" value="12"></jsp:param>
</jsp:forward>
JSTL标签:
JSTL标签库的使用就是为了弥补HTML标签的不足,它自定义了许多标签可供使用,标签的功能和Java代码一样。
- 核心标签
- 格式化标签
- SQL标签
- XML标签
8. JavaBean
JavaBean是实体类,有特定的写法:
- 必须要有一个无参构造;
- 属性必须私有化;
- 必须有对应的get/set方法;
- 一般用于和数据库的字段做映射。
实体类:一般和数据库中的表结构一一对应。
ORM:对象关系映射
- 表–>类
- 字段–>属性
- 行记录–>对象
9. MVC三层架构
Model模型、View视图、Controller控制器
9.1 早年
JSP
Servlet
JDBC
用户
View:视图
Controller:控制器
专注于处理请求及控制视图跳转
专注于显示数据
JavaBean:pojo:entity
数据库
弊端:程序臃肿,不利于维护;servlet的代码中要处理请求、响应、视图跳转、处理JDBC、处理业务代码、处理逻辑代码
9.2 MVC
操作
转发或重定向
Model
select
JDBC
用户
View:JSP展示数据模型及提供操作
Controller:Servlet接收请求及视图跳转
service:login:logout:查询全部用户
JavaBean:pojo:entity
数据库
Model模型:
- 业务处理:业务逻辑(service)
- 数据持久层:CRUD(Dao)
View视图:
- 展示数据;
- 提供链接发起Servlet请求(a,form,img…)
Controller控制器(Servlet):
- 接收用户的请求:(req:请求参数,Session信息…)
- 交给业务层处理对应的代码
- 控制视图的跳转
10. Filter
过滤器,用来过滤网站的数据:处理中文乱码;登录验证……
import javax.servlet.*;
import java.io.IOException;
public class ff implements Filter {
//初始化
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
//chain链:过滤中的所有代码,在过滤特定请求的时候都会执行;必须要让过滤器继续运行chain.doFilter(request,response);
@Override
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");
chain.doFilter(request,response);//让请求继续走,如果不写,程序到这里被拦截停止!
}
//销毁:web服务器关闭的时候,过滤器会销毁
@Override
public void destroy() {
}
}
<!--web.xml中配置-->
<filter>
<filter-name>CharacterEncodingFilter</filter-name>
<filter-class>com.example.CharacterEncodingFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>CharacterEncodingFilter</filter-name>
<!--只要是/servlet的任何请求,都会经过这个过滤器-->
<url-pattren>/servlet/*</url-pattern>
</filter-mapping>
11. 监听器
实现一个监听器的接口(有N种):
统计网站在线人数:统计session——HttpSessionListener
public class OnlineCountListener implements HttpSessionListener{
public void sessionCreated(HttpSessionEvent se){
ServletContext ctx = se.getSession().getServletContext();
Integer onlineCount = (Integer) ctx.getAttribute("OnlineCount");
if(onlineCount==null){
onlineCount = new Integer(1);
}else {
int count = onlineCount.intValue();
onlineCount = new Integer(count+1);
}
ctx.setAttribute("OnlineCount",OnlineCount);
}
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);
}
ctx.setAttribute("OnlineCount",OnlineCount);
}
}
<!--web.xml中配置-->
<listener>
<listener-class>com.example.OnlineCountListener</listener-class>
</listener>
session销毁:1.手动销毁 getSession().invalidate();2.自动销毁。