JAVAWeb中的Servlet学习
一 Servlet简介
1.1动态资源和静态资源
静态资源
- 无需在程序运行时通过代码运行生成的资源,在程序运行之前就写好的资源.例如:html css js img ,音频文件和视频文件
动态资源
- 需要在程序运行时通过代码运行生成的资源,在程序运行之前无法确定的数据,运行时动态生成,例如Servlet,Thymeleaf …。
- 动态资源指的不是视图上的动画效果或者是简单的人机交互效果
1.2Servlet简介
Servlet(serverapplet)是运行在服务端(tomcat)的java小程序,是sun公司提供的一套定义动态资源规范;从代码上将Servlet就是一个接口。
- 用来接收、处理客户端请求、响应给浏览器的动态资源。在整个Web应用中,Servlet主要负责接收处理请求、协同调度功能以及响应数据。我们可以把Servlet称为web应用中的控制器。
- 不是所有的JAVA类都能用于处理客户端请求,能处理客户端请求并做出响应的一套技术标准就是Servlet
- Servlet是运行在服务端的,所以Servlet必须在WEB项目中开发且在Tomcat这样的服务容器中运行
请求响应HttpServletRequest和HttpServletResponse之间的对对应关系
二 Servlet开发流程
2.1 目标
校验注册信息。
2.2 开发过程
1 创建javaWEB项目,同时将tomcat添加当前项目依赖
2 重写service方法,service(HttpServletRequest req, HttpServletResponse resp)
3 在service方法中,定义业务处理代码
4 在web。xml中,配置Servlet对应的请求映射路径
2.3 servlet-api.jar 导入问题
servlet-api 编码的时候需要,运行的时候在服务器的环境中,由服务器(Tomcat)提供,因此我们的JAVAWEB项目中,在打包/构建的时候,是无需携带servlet-api的jar包
2.4 Content-Type响应头的问题
MIME类型响应头 媒体类型,文件类型,响应的数据类型
MIME类型用于告诉客户端响应的数据是什么类型的数据,客户端以此类型决定用什么方式解析响应体
2.5 Servlet_url-pattern的一些特殊写法问题
<!-- 一个servlet-name可以对应多个url-pattern一个servlet标签可以对应多个servlet-mapping标签url-pattern1 精确匹配/s12 模糊匹配*作为通配符/ 匹配全部,不包含jsp文件/* 匹配全部,包含jsp文件/a/* 匹配前缀,后缀模糊/*.action 匹配后缀,前缀模糊
--><servlet><servlet-name>servlet1</servlet-name><servlet-class>com.zhg.servlet.Servlet1</servlet-class></servlet><servlet-mapping><servlet-name>servlet1</servlet-name><url-pattern>/s1</url-pattern><url-pattern>/s2</url-pattern></servlet-mapping><servlet-mapping><servlet-name>servlet1</servlet-name><url-pattern>/s3</url-pattern></servlet-mapping>
2.6 Servlet 注解方式配置
@WebServlet("/s1")
public class UserServlet extends HttpServlet
三 Servlet的声明周期
3.1 生命周期简介
什么是Servlet的生命周期
- 应用程序中的对象不仅在空间上有层次结构的关系,在时间上也会因为处于程序运行过程中的不同阶段而表现出不同状态和不同行为–这就是对象的生命周期。
- 简单的叙述生命周期,就是对象在容器中从开始创建到销毁的过程。
Servlet容器
- Servlet对象是Servlet容器创建的,生命周期方法都是由容器(目前我们使用的是Tomcat)调用的。这一点和我们之前所编写的代码有很大不同。在今后的学习中我们会看到,越来越多的对象交给容器或框架来创建,越来越多的方法由容器或框架来调用,开发人员要尽可能多的将精力放在业务逻辑的实现上。
Servlet主要的生命周期执行特点
生命周期 | 对应方法 | 执行时机 | 执行次数 |
---|---|---|---|
构造对象 | 构造器 | 第一次请求或容器启动 | 1 |
初始化 | init() | 构造完毕后 | 1 |
处理服务 | service(HttpServletRequest req, HttpServletResponse resp) | 每次请求 | 多次 |
销毁 | destory() | 容器关闭 | 1 |
/*** 1实例化 构造器* 2初始化 init* 3接受请求,处理请求 service* 4销毁 destroy** Servlet在Tomcat中是单例的* Servlet的成员变脸在多个线程之中是共享的* 不建议在service方法中修改成员变量 在并发请求时,会引发线程安全问题** defaultServlet 默认servlet 静态资源由defaultServlet处理*/
@WebServlet("/servletifeCycle")
public class ServletifeCycle extends HttpServlet {public ServletifeCycle() {System.out.println("new ");}@Overridepublic void init(ServletConfig config) throws ServletException {System.out.println("init");}@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {System.out.println("service");}@Overridepublic void destroy() {System.out.println("destroy");}
}
四 Servlet继承结构
4.1 顶级的Servlet接口
/* Servlet继承结构
* 1顶级的Servlet接口
* public interface Servlet {
* //初始化方法,构造完毕后,由Tomcat自动调用完成初始化功能的方法
* void init(ServletConfig var1) throws ServletException;
*
* //获得ServletConfig对象的方法
* ServletConfig getServletConfig();
*
* //接受用户请求,响应信息
* void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;
*
* //返回Servlet字符串描述的信息
* String getServletInfo();
*
* //Servlet在回收前,由TOmcat调用的销毁方法,往往用于资源的释放
* void destroy();
* }
*/
4.2 抽象的类 GenericServlet
/*2 抽象的类 GenericServlet 侧重service方法意外的其他方法的基础处理
* public abstract class GenericServlet implements Servlet{
*
* private transient ServletConfig config;
*
* public void destroy() {
* //将抽象方法,重写为普通方法,在方法内部没有任何的实现代码
* }
*
* // tomcat在调用init方法时,会读取配置信息进入一个ServletConfig对象并将该对象传入init方法
* public void init(ServletConfig config) throws ServletException {
* //将config对象存储为当前的属性
* this.config = config;
* //调用重载的无参的init
* this.init();
* }
*
* //重载的初始化方法,我们重新初始化方法时对应的方法
* public void init() throws ServletException {
* }
*
* //返回ServletConfig的方法
* public ServletConfig getServletConfig() {
* return this.config;
* }
*
* // 再次抽象声明service方法
* public abstract void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;
* }
*/
4.3 HttpServlet
/* 3 抽象类 侧重service方法的处理
* public abstract class HttpServlet extends GenericServlet{
*
*
* public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
* HttpServletRequest request;
* HttpServletResponse response;
* try {
* //参数父转子
* request = (HttpServletRequest)req;
* response = (HttpServletResponse)res;
* } catch (ClassCastException var6) {
* throw new ServletException(lStrings.getString("http.non_http"));
* }
*
* //调用重载的service
* this.service(request, response);
* }
*
* protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
* //获取请求的方式
* String method = req.getMethod();
* //更具请求方式,调用对应的do方法
* if (method.equals("GET")) {
* long lastModified = this.getLastModified(req);
* if (lastModified == -1L) {
* this.doGet(req, resp);
* } else {
* long ifModifiedSince;
* try {
* ifModifiedSince = req.getDateHeader("If-Modified-Since");
* } catch (IllegalArgumentException var9) {
* ifModifiedSince = -1L;
* }
*
* if (ifModifiedSince < lastModified / 1000L * 1000L) {
* this.maybeSetLastModified(resp, lastModified);
* this.doGet(req, resp);
* } else {
* resp.setStatus(304);
* }
* }
* } else if (method.equals("HEAD")) {
* long lastModified = this.getLastModified(req);
* this.maybeSetLastModified(resp, lastModified);
* this.doHead(req, resp);
* } else if (method.equals("POST")) {
* this.doPost(req, resp);
* } else if (method.equals("PUT")) {
* this.doPut(req, resp);
* } else if (method.equals("DELETE")) {
* this.doDelete(req, resp);
* } else if (method.equals("OPTIONS")) {
* this.doOptions(req, resp);
* } else if (method.equals("TRACE")) {
* this.doTrace(req, resp);
* } else {
* String errMsg = lStrings.getString("http.method_not_implemented");
* Object[] errArgs = new Object[1];
* errArgs[0] = method;
* errMsg = MessageFormat.format(errMsg, errArgs);
* resp.sendError(501, errMsg);
* }
*
* }
*
* //
* protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
* String msg = lStrings.getString("http.method_get_not_supported");
* //故意响应请求方式不允许的信息
* this.sendMethodNotAllowed(req, resp, msg);
* }
*
* //
* private void sendMethodNotAllowed(HttpServletRequest req, HttpServletResponse resp, String msg) throws IOException {
* String protocol = req.getProtocol();
* if (protocol.length() != 0 && !protocol.endsWith("0.9") && !protocol.endsWith("1.0")) {
* resp.sendError(405, msg);//故意响应405请求方式不允许的信息
* } else {
* resp.sendError(400, msg);//故意响应400请求方式不允许的信息
* }
*
* }
* }
*/
4.4 自定义Servlet
/* 4 自定义Servlet
*
* class Servlet extends HttpServlet{
*
* @Override
* protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
* //接受用户请求信息
*
* //做出响应
* }
* }
* 1 部分程序员推荐在servlet中重写do...方法处理请求 理由:service方法中可能做了一些处理,如果我们直接重写service的话,弗雷service方法处理功能则失效
* 2 目前直接重写service也没有什么问题
* 3 使用SpringMvc框架后,无需继承重写HttpServlet方法,处理请求的方法也无需do*** service
* 4 如果doGet 和doPost方法中,实现方式一样,可以一个方法直接调用另一个方法
*/
五 ServletConfig
5.1 ServletConsig是什么
- 为Servlet提供初始配置参数的一种对象,每个Servlet都有自己独立唯一的ServletConfig对象
- 容器会为每个Servlet实例化一个ServletConfig对象,并通过Servlet生命周期的init方法传入给Servlet作为属性
5.2 ServletConfig为Servlet提供配置参数
配置Servlet
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee https://jakarta.ee/xml/ns/jakartaee/web-app_6_0.xsd"version="6.0"><servlet><servlet-name>servlet2</servlet-name><servlet-class>com.zhg.servlet.Servlet2</servlet-class><!--配置servlet的初始参数--><init-param><param-name>key1</param-name><param-value>value1</param-value></init-param></servlet><servlet-mapping><servlet-name>servlet2</servlet-name><url-pattern>/servlet2</url-pattern></servlet-mapping>
</web-app>
定义Servlet
public class Servlet2 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {ServletConfig servletConfig = getServletConfig();//获取初始配置信息//更具参数名获取参数值String key1 = servletConfig.getInitParameter("key1");System.out.println(key1);//获取所有的初始参数的名字Enumeration<String> initParameterNames = servletConfig.getInitParameterNames();//initParameterNames.hasMoreElements() 判断有没有下一个参数,如果有返回truewhile (initParameterNames.hasMoreElements()){//1获取下一个参数名 2右边向下移动String name = initParameterNames.nextElement();String value = servletConfig.getInitParameter(name);System.out.println(name+"-->"+value);}}
}
注解方式配置
@WebServlet(urlPatterns = "/servlet2",initParams = {@WebInitParam(name = "key1",value = "value1"),@WebInitParam(name = "key2",value = "value2")}
)
public class Servlet2 extends HttpServlet
六 ServletContext
6.1 ServletContext是什么
- ServletContext对象有称呼为上下文对象,或者叫应用域对象
- 容器会为每个app创建一个独立的唯一的ServletContext对象
- ServletContext对象为所有的Servlet所共享
- ServletContext可以为所有的Servlet提供初始配置参数
6.2 Servlet怎么使用
-
配置ServletContext参数
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee https://jakarta.ee/xml/ns/jakartaee/web-app_6_0.xsd"version="6.0"><context-param><param-name>encoding</param-name><param-value>UTF-8</param-value></context-param><context-param><param-name>admin</param-name><param-value>pwd</param-value></context-param></web-app>
-
在Servlet中获取ServletContext并获取参数
@WebServlet("/servlet3")
public class Servlet3 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//获取ServletContextServletContext servletContext1 = getServletContext();ServletContext servletContext2 = getServletConfig().getServletContext();ServletContext servletContext3 = req.getServletContext();//验证三个ServletContext是否是同一个对象System.out.println(servletContext1 == servletContext2);System.out.println(servletContext1 == servletContext3);String encoding = servletContext1.getInitParameter("encoding");System.out.println(encoding);Enumeration<String> initParameterNames = servletContext1.getInitParameterNames();while (initParameterNames.hasMoreElements()) {String name = initParameterNames.nextElement();String value = servletContext1.getInitParameter(name);System.out.println(name + "-->" + value);}}
}
6.3ServletContext其他重要API
获取其他资源的真实路径
String realPath = servletContext.getRealPath(“资源web目录中的路径”)
获取项目的上下文路径
String contextPath = servletContext.getContextPath();
域对象的相关API
- 域对象:一些用于存储数据和传递数据的对象,传递数据不同的范围,我们称之为不同的域;不同的域对象代表不同的域,共享数据的范围也不同
- ServletContext代表应用,所以ServletContext域也叫作应用域,是webapp中最大的域,可以在本应用内实现数据的共享和传递
- webapp中的三大域对象,分别是应用域,会话域,请求域
API | 功能解释 |
---|---|
void setAttribute(String var1, Object var2); | 向域中存储/修改数据 |
Object getAttribute(String var1); | 获得域中的数据 |
void removeAttribute(String var1); | 移除域中的数据 |
七 HttpServletRequest
7.1 HttpServletRequest简介
HttpServletRequest是什么
- HttpServletRequest是一个接口,其父接口是ServletRequest
- HttpServletRequest是Tomcat将请求报文转换封装而来的对象,在Tomcat调用service方法时传入
- HttpServletRequest代表客户端发来的请求,所有请求中的信息都可以通过该对象获得
7.2 HttpServletRequest常见API
HttpServletRequest怎么用
-
获取请求行信息相关
API 功能解释 StringBuffer getRequestURL(); 获取客户端请求的url String getRequestURI(); 获取客户端请求项目中的具体资源 int getServerPort(); 获取客户端发送请求时的端口 int getLocalPort(); 获取本应用在所在容器的端口 int getRemotePort(); 获取客户端程序的端口 String getScheme(); 获取请求协议 String getProtocol(); 获取请求协议及版本号 string getMethod(); 获取请求方式 -
获取请求头信息相关
API | 功能解释 |
---|---|
String getHeader(String headerName); | 根据头名称获取请求头 |
Enumeration getHeaderNames0; | 获取所有的请求头名字 |
String getContentType(); | 获取content-type请求头 |
-
获得请求参数相关
API 功能解释 String getParameter(String parameterName); 根据请求参数名获取请求单个参数值 String[] getParameterValues(String parameterName); 根据请求参数名获取请求多个参数值数组 Enumeration getParameterNames0; 获取所有请求参数名 Map<String, String[> getParameterMap(); 获取所有请求参数的键值对集合 BufferedReader getReader( )throws lOException; 获取读取请求体的字符输入流 ServletInputStream getInputStream() throws IOException; 获取读取请求体的字节输入流 int getContentLength(); 获得请求体长度的字节数 -
其他API
API 功能解释 String getServletPath(); 获取请求的Servlet的映射路径 ServletContext getServletContext(); 获取ServletContext对象 Cookie[] getCookies(); 获取请求中的所有cookie HttpSession getSession(); 获取Session对象 void setCharacterEncoding(String encoding); 设置请求体字符集
@WebServlet("/servlet5")
public class Servlet5 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//行相关 GET/POST uri http/1.1System.out.println(req.getMethod());//获取请求方式 GETSystem.out.println(req.getScheme());//获取协议 httpSystem.out.println(req.getProtocol());//获取协议及版本 HTTP/1.1System.out.println(req.getRequestURI());//获取请求路径 项目内的资源路径 /demo2/servlet5System.out.println(req.getRequestURL());//获取请求路径 项目内资源的完整路径 http://localhost:8080/demo2/servlet5/*URI 统一资源标识符 interface URI{} 资源定位的要求和规范URL 统一资源定位符 class URL implements URI{} http协议下,一个具体的资源路径*/System.out.println(req.getLocalPort()); //获取服务器端口号System.out.println(req.getServerPort()); //获取客户端发送请求时的端口号System.out.println(req.getRemotePort()); //获取客户端软件的端口号//头相关//根据名字单独获取头信息System.out.println(req.getHeader("User-Agent"));Enumeration<String> headerNames = req.getHeaderNames();//获取所有的头信息while (headerNames.hasMoreElements()){String name = headerNames.nextElement();String value = req.getHeader(name);System.out.println(name+"-->"+value);}}
}
@WebServlet("/servlet6")
public class Servlet6 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//获取键值对形式的参数//根据参数名获取单个参数String username = req.getParameter("username");System.out.println(username);System.out.println("-----------------------------");//根据参数名获取多个参数值String[] hobbies = req.getParameterValues("hobby");System.out.println(Arrays.toString(hobbies));System.out.println("-----------------------------");//获取所有参数名Enumeration<String> parameterNames = req.getParameterNames();while (parameterNames.hasMoreElements()){String name = parameterNames.nextElement();String[] values = req.getParameterValues(name);System.out.println(name+"-->"+Arrays.toString(values));}System.out.println("-----------------------------");//获取所有参数的map集合 key=参数名 value=参数值Map<String, String[]> parameterMap = req.getParameterMap();for (Map.Entry<String, String[]> entry : parameterMap.entrySet()) {String key = entry.getKey();String[] values = entry.getValue();System.out.println(key+"-->"+Arrays.toString(values));}/*** 以上API用于获取key=value形式的参数** 获得请求体中非键值对数据 JSON串 文件** 获取一个从请求体中读取字符的字符输入流* BufferedReader reader = req.getReader(); JSON串** 获得一个从请求中读取二进制数据字节的输入流* ServletInputStream inputStream = req.getInputStream(); 文件*/String servletPath = req.getServletPath();System.out.println(servletPath);}
}
八 HttpServletResponse
8.1 HttpServletResponse简介
HttpServletResponse是什么
- HttpServletResponse是一个接口,其父接口是ServletResponse
- HttpServletResponse是Tomcat预先创建的,在Tomcat调用service方法时传入
- HttpServletResponse代表对客户端的响应,该对象会被转换成响应的报文发送给客户端,通过该对象我们可以设置响应信息
8.2 HttpServletResponse常见API
HttpServletResponse怎么用
-
设置响应行相关
API 功能解释 void setStatus(int code) 设置响应状态码 -
设置响应头相关
API | 功能解释 |
---|---|
void setHeader(String headerName,String headerValue) | 设置/修改响应头键值对 |
void setContentType(String contentType); | 设置content-type响应头及响应字符集(设置MIME类型) |
-
设置响应体相关
API 功能解释 Printwriter getWriter() throws IOException; 获得向响应体放入信息的字符输出流 ServletOutputStream getOutputStream( throws lOException; 获得向响应体放入信息的字节输出流 void setContentLength(int length); 设置响应体的字节长度,其实就是在设置content-length响应头 -
其他API
API 功能解释 void sendError(int code, String message) throws IOException; 向客户端响应错误信息的方法,需要指定响应码和响应信息 void addCookie(Cookie cookie); 向响应体中增加cookie void setCharacterEncoding(String encoding); 设置响应体字符集 MIME类型
-
MIME类型,可以理解为文档类型,用户表示传递的数据是属于什么类型的文档
-
浏览器可以根据MIME类型决定该用什么样的方式解析接收到的响应体数据
-
可以这样理解:前后端交互数据时,告诉对方发给对方的是html/css/js/图片/声音/视频…
-
tomcat/conf/web.xml中配置了常见文件的拓展名和MIMIE类型的对应关系
-
常见的MIME类型举例如下
文件扩展名 | MIME类型 |
---|---|
.html | text/html |
.css | text/css |
.js | application/javascript |
.png/.jpeg/… | image/jpeg |
.mp3/.mpe/.mpeg/… | audio/mpeg |
.mp4 | video/mp4 |
.m1v/.m2v/.mpe/… | video/mpeg |
@WebServlet("/servlet7")
public class Servlet7 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//设置响应头相关的API HTTP/1.1 200/404/405/500/...resp.setStatus(200);String info = "<h1>hello world</h1>";//设置响应头相关的API
// resp.setHeader("aaa","valuea");
// resp.setHeader("Content-Type","text/html");
// resp.setHeader("Content-Length","1234");resp.setContentType("text/html");resp.setContentLength(info.getBytes().length);//设置响应体内容API// 获得一个响应体中输入文本字符输出流PrintWriter writer = resp.getWriter();writer.write(info);}
}
九请求转发和响应重定向
9.1 概述
什么是请求转发和响应重定向
- 请求转发和响应重定向是web应用中间接访问项目资源的两种手段,也是Servlet控制页面跳转的两种手段。
- 请求转发通过HttpServletRequest实现,响应重定向通过HttpServletResponse实现
9.2 请求转发
请求转发运行逻辑图
请求转发特点
- 请求转发通过HttpServletRequest对象获取请求转发器实现
- 请求转发是服务器内部的行为,对客户端是屏蔽的
- 客户端只发送了一次请求,客户端地址栏不变
- 服务端只产生了一对请求和响应对象,这一对请求和响应对象会继续传递给下一个资源
- 因为全程只有一个HttpServletRequset对象,所以请求参数可以传递,请求域中的数据也可以传递
- 请求转发可以转发给其他Servlet动态资源,也可以转发给一些静态资源以实现页面跳转
- 请求转发可以转发给WEB-INF下受保护的资源
@WebServlet("/servlet8")
public class Servlet8 extends HttpServlet {/*** 1 请求转发是通过HttpServletRequest对象来完成* 2 请求转发是服务器内部转发,对客户端是屏蔽的* 3 客户端只产生了一次请求,服务端只产生了一对req 和 res* 4 浏览器地址栏不会发生改变* 5 请求的参数是可以继续传递的* 6 目标资源可以是servlet动态资源 也可以是html静态资源* 7目标资源可以是WEB-INF下的受保护资源* 8 目标资源不可以是本项目的外部资源**/@Overridepublic void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {System.out.println("servlet8");System.out.println(req.getParameter("key"));//转发到servlet9req.getRequestDispatcher("/servlet9").forward(req,res);}
}
@WebServlet("/servlet9")
public class Servlet9 extends HttpServlet {@Overridepublic void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {System.out.println("servlet9");System.out.println(req.getParameter("key"));}
}
9.3 响应重定向
响应重定向运行逻辑图
响应重定向特点
- 响应重定向通过HttpServletResponse对象的sendRedired方法实现
- 响应重定向是服务端通过302响应码和路径,告诉客户端自己去找其他资源,是在服务端提示下的客户端的行为
- 客户端至少发送了两次请求,客户端地址栏是要变化的
- 服务端产生了多对请求和响应对象,且请求和响应对象不会传递给下一个资源
因为全程产生了多个HttpServletRequset对象,所以请求参数不可以传递,请求域中的数据也不可以传递 - 重定向可以是其他Servlet动态资源,也可以是一些静态资源以实现页面跳转
- 重定向不可以到WEB-INF下受保护的资源
- 重定向可以到本项目以外的外部资源
@WebServlet("/servlet10")
public class Servlet10 extends HttpServlet {/*1 重定向是通过HttpServletResponse对象实现2 响应重定向是在服务端提示下,客户端的行为3 客户端的地址栏是变化的4 请求产生多次 后端就会有多个 req resp 此时请求中的参数不能自动传递5 目标资源可以是试图资源6 目标资源不可以WEB-INF下的资源7 目标资源可以是本项目的外部资源*/@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//接受用户请求System.out.println("Servlet10 service");//响应重定向 设置响应状态码为302 同时设置响应头Locationresp.sendRedirect("servlet11");}
}
@WebServlet("/servlet11")
public class Servlet11 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {System.out.println("Servlet11 service");}
}
十 乱码和路径问题
10.1 乱码问题
乱码问题产生的根本原因是什么
- 数据的编码和解码使用的不是同一个字符集
- 使用了不支持某个语言文字的字符集
各个字符集的兼容性
10.1.1 HTML乱码问题
设置项目文件的字符集要使用一个支持中文的字符集
试图文件通过来告诉浏览器通过什么字符集来解析当前文档
10.1.2 Tomcat控制台乱码
在tomcat10.1 版本中,修改tomcat/conf/loggin.properties中,所有的UTF-为GBK即可
sout乱码问题,设置JVM加载.class文件时使用UTF-8字符集
10.1.3 请求乱码问题
10.1.3.1 GET请求乱码
GET请求方式乱码分析
GET方式提交参数的方式是将参数放到URL后面,如果使用的不是UTF-8,那么会对参数进行URL编码处理
- HTML中的影响了GET方式提交参数的URL编码
- tomcat10.1.7的的URI编码默认为UTF-8
- 当GET方式提交的参数URL编码和tomcat10.1默认的URl编码不一致时,就会出现乱码
GET请求方式乱码演示
-
浏览器解析的文档的
-
GET方式提交时,会对数据进行URL编码处理,是将GBK转码为“百分号码”
-
tomcat10.1.7默认使用UTF-8对URI进行解析,造成前后端使用的字符集不一致,出现乱码
GET请求方式乱码解决
方式1:设置GET方式提交的编码和Tomcat的URI默认解析编码一致
<meta charset="UTF-8">
方式2:方式2:设置Tomcat的URI解析字符集和GET请求发送时所使用URL转码时的字符集一致即可,修改conf/serverxml中Connecter添加URIEncoding=““GBK””(不推荐)
10.1.3.2 POST方式请求乱码
POST请求方式乱码分析
- POST请求将参数放在请求体中进行发送
- 请求体使用的字符集受到了<meta charset=““字符集””/>的影响
- Tomcat10.1默认使用UTF-8字符集对请求体进行解析
- 如果请求体的URL转码和Tomcat的请求体解析编码不一致,就容易出现乱码
POST请求方式乱码演示
- POST请求请求体受到了的影响
- 请求体中,将GBK数据进行URL编码
- tomcat10 默认UTF-8为请求体的解码字符集
POST请求方式乱码解决
-
方式1:请求时,使用UTF-8字符集提交请求体
-
-
方式2:后端获取参数前,设置解析请求体使用的字符集和请求发送时使用的字符集一致
10.1.3.3 响应乱码问题
响应乱码分析
- 在Tomcat10.1中,向响应体中放入的数据默认使用了工程编码UTF-8
- 浏览器在接收响应信息时,使用了不同的字符集或者是不支持中文的字符集就会出现乱码
演示
-
服务端通过response对象向响应体添加数据
-
浏览器接收数据解析乱码
响应乱码解决
- 方式1:手动设定浏览器对本次响应体解析时使用的字符集(不推荐) edge和chrome浏览器没有提供直接的比较方便的入口,不方便
- 方式2:后端通过设置响应体的字符集和浏览器解析响应体的默认字符集一致(不推荐)
- 方式3:通过设置content-type响应头,告诉浏览器以指定的字符集解析响应体(推荐)
10.2 路径问题
相对路径和绝对路径
-
相对路径
。相对路径的规则是:以当前资源所在的路径为出发点去寻找目标资源
。相对路径不以/开头
。在file协议下,使用的是磁盘路径
。在http协议下,使用的是url路径
。相对路径中可以使用./表示当前资源所在路径,可以省略不写
。相对路径中可以使用…/表示当前资源所在路径的上一层路径,需要时要手动添加
- 绝对路径
。绝对路径的规则是:使用以一个固定的路径做出出发点去寻找目标资源,和当前资源所在的路径没有关系。
。绝对路径要以/开头
。绝对路径的写法中,不以当前资源的所在路径为出发点,所以不会出现./和…/
。不同的项目和不同的协议下,绝对路径的基础位置可能不同,要通过测试确定
。绝对路径的好处就是:无论当前资源位置在哪,寻找目标资源路径的写法都一致
10.2.1 前段路径问题
前端项目结构
10.2.1.1 相对路径情况分析
相对路径情况1:web/index.html中引入web/static/img/logo.png
-
访问路径index.html中的url为:http://localhost:8080/demo2/index.html
-
当前资源为 :index.html
-
当前资源的所在路径为: http://localhost:8080/demo2/
-
要获取的目标资源url为:http://localhost:8080/demo2/static/img/logo.png
-
index.html中定义的 :< img src=“static/img/logo.png”/>
-
寻找方式就是在当前资源所在路径 (http://localhost:8080/demo2/)后拼接src属性值(static/img/logo.png),正好是目标资源正常获取的url(http://localhost:8080/demo2/static/img/logo.png)
<!DOCTYPE html> <html lang="en"> <head><meta charset="UTF-8"><title>Title</title> </head> <body> <!--引入图片1 相对路径以当前资源的所在路径为出发点去找目标资源语法: 不以/开头./表示当前资源的路径../表示当前资源的上一层路径2 绝对路径--><img src="static/img/logo.png"></body> </html>
相对路径情况2:web/page/a/test.html中引入web/static/img/logo.png
- 访问test.html的url为 : http://localhost:8080/demo2/page/a/test.html
- 当前资源为 : test.html
- 当前资源的所在路径为 : http://localhost:8080/demo2/page/a
- 要获取的目标资源url为 : http://localhost:8080/demo2/static/img/logo.png
- test.html中定义的了 :
<img src="../../static/img/logo.png"/>
- 寻找方式就是在当前资源所在路径(http://localhost:8080/demo2/page/a)后拼接src属性值(…/…/static/img/logo.png),其中 …/可以抵消一层路径,正好是目标资源正常获取的url(http://localhost:8080/demo2/static/img/logo.png)
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body><!-- ../代表上一层路径 --><img src="../../static/img/logo.png">
</body>
</html>
相对路径情况3:web/WEB-INF/views/view1.html中引入web/static/img/logo.png
- view1.html在WEB-INF下,需要通过Servlet请求转发获得
@WebServlet("/view1Servlet")
public class View1Servlet extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {RequestDispatcher requestDispatcher = req.getRequestDispatcher("WEB-INF/views/view1.html");requestDispatcher.forward(req,resp);}
}
- 访问view1.html的url为 : http://localhost:8080/demo2/view1Servlet
- 当前资源为 : view1Servlet
- 当前资源的所在路径为 : http://localhost:8080/demo2/
- 要获取的目标资源url为 : http://localhost:8080/demo2/static/img/logo.png
- view1.html中定义的了 :
<img src="static/img/logo.png"/>
- 寻找方式就是在当前资源所在路径(http://localhost:8080/demo2/)后拼接src属性值(static/img/logo.png),正好是目标资源正常获取的url(http://localhost:8080/demo2/static/img/logo.png)
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body><img src="static/img/logo.png">
</body>
</html>
10.2.1.2 绝对路径情况分析
绝对路径情况1:web/index.html中引入web/static/img/logo.png
- 访问路径index.html的url为: http://localhost:8080/demo2/index.html
- 绝对路径的基准路径为:http://localhost:8080
- 要获取的目标资源url为:http://localhost:8080/demo2/static/img/logo.png
- index.html中定义的:< img src=“/demo2/static/img/logo.png” />
- 寻找方式就是在基准路径(http://localhost:8080)后面拼接src属性值(/demo2/static/img/logo.png),得到的正式目标资源访问的正确路径
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body>
<!--引入图片1 相对路径以当前资源的所在路径为出发点去找目标资源语法: 不以/开头./表示当前资源的路径../表示当前资源的上一层路径2 绝对路径--><img src="/demo2/static/img/logo.png"></body>
</html>
绝对路径情况2:web/page/a/test.html中引入web/static/img/log.png
- 访问路径index.html的url为: http://localhost:8080/demo2/page/a/test.html
- 绝对路径的基准路径为:http://localhost:8080
- 要获取的目标资源url为:http://localhost:8080/demo2/static/img/logo.png
- index.html中定义的:< img src=“/demo2/static/img/logo.png” />
- 寻找方式就是在基准路径(http://localhost:8080)后面拼接src属性值(/demo2/static/img/logo.png),得到的正式目标资源访问的正确路径
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body>
<img src="/demo2/static/img/logo.png"></body>
</html>
绝对路径情况3:webWEB-INF/views/view1.html中引入web/static/img/log.png
-
view1.html在WEB-INF下,需要通过Servlet请求转发获得
@WebServlet("/view1Servlet") public class View1Servlet extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//请求转发到req.getRequestDispatcher("WEB-INF/views/view1.html").forward(req,resp);} }
-
访问view1.html的url为:http://localhost:8080/demo2/view1Servlet
-
绝对路径的基准路径为:http://localhost:8080
-
要获取的目标资源url为:http://localhost:8080/demo2/static/img/logo.png
-
view1.html中定义的: < img src=“/demo2/static/img/logo.png” />
-
寻找方式就是在基准路径(http://localhost:8080)后面拼接src属性值(/static/img/logo.png),得到的正是目标资源访问的正确路径
base标签的使用
base标签定义页面相对路径公共前缀
- base 标签定义在head标签中,用于定义相对路径的公共前缀
- base 标签定义的公共前缀只在相对路径上有效,绝对路径中无效
- 如果相对路径开头有 ./ 或者…/修饰,则base标签对该路径同样无效
index.html 和a/b/c/test.html 以及view1Servlet 中的路径处理
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title><!--定义相对路径的公共前缀,将相对路径转化成了绝对路径--><base href="/demo2/">
</head>
<body><img src="static/img/logo.png">
</body>
</html>
10.2.1.4 缺省项目上下文路径
项目上下文路径变化问题
- 通过 base标签虽然解决了相对路径转绝对路径问题,但是base中定义的是项目的上下文路径
- 项目的上下文路径是可以随意变化的
- 一旦项目的上下文路径发生变化,所有base标签中的路径都需要改
解决方案
- 将项目的上下文路径进行缺省设置,设置为 /,所有的绝对路径中就不必填写项目的上下文了,直接就是/开头即可
10.2.2 重定向中的路径问题
目标 :由/x/y/z/servletA重定向到a/b/c/test.html
@WebServlet("/x/y/z/servletA")
public class ServletA extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {}
}
10.2.2.1相对路径写法
- 访问ServletA的url为 : http://localhost:8080/demo2/x/y/z/servletA
- 当前资源为 : servletA
- 当前资源的所在路径为 : http://localhost:8080/demo2/x/x/z/
- 要获取的目标资源url为 : http://localhost:8080/demo2/a/b/c/test.html
- ServletA重定向的路径 : …/…/…/a/b/c/test/html
- 寻找方式就是在当前资源所在路径(http://localhost:8080/demo2/x/y/z/)后拼接(…/…/…/a/b/c/test/html),形成(http://localhost:8080/demo2/x/y/z/…/…/…/a/b/c/test/html)每个…/抵消一层目录,正好是目标资源正常获取的url(http://localhost:8080/demo2/a/b/c/test/html)
@WebServlet("/x/y/z/servletA")
public class ServletA extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {// 相对路径重定向到test.htmlresp.sendRedirect("../../../a/b/c/test.html");}
}
10.2.2.2绝对路径写法
-
访问ServletA的url为 : http://localhost:8080/demo2/x/y/z/servletA
-
绝对路径的基准路径为 : http://localhost:8080
-
要获取的目标资源url为 : http://localhost:8080/demo2/a/b/c/test.html
-
ServletA重定向的路径 : /demo2/a/b/c/test.html
-
寻找方式就是在基准路径(http://localhost:8080)后面拼接(/demo2/a/b/c/test.html),得到( http://localhost:8080/demo2/a/b/c/test.html)正是目标资源访问的正确路径
-
绝对路径中需要填写项目上下文路径,但是上下文路径是变换的
- 可以通过 ServletContext的getContextPath()获取上下文路径
- 可以将项目上下文路径定义为 / 缺省路径,那么路径中直接以/开头即可
//绝对路径中,要写项目上下文路径 //resp.sendRedirect("/demo2/a/b/c/test.html"); // 通过ServletContext对象动态获取项目上下文路径 //resp.sendRedirect(getServletContext().getContextPath()+"/a/b/c/test.html"); // 缺省项目上下文路径时,直接以/开头即可 resp.sendRedirect("/a/b/c/test.html");
10.2.3 请求转发中的路径问题
目标 :由x/y/servletB请求转发到a/b/c/test.html
@WebServlet("/x/y/servletB")
public class ServletB extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {}
}
10.2.3.1 相对路径写法
-
访问ServletB的url为 : http://localhost:8080/demo2/x/y/servletB
-
当前资源为 : servletB
-
当前资源的所在路径为 : http://localhost:8080/demo2/x/x/
-
要获取的目标资源url为 : http://localhost:8080/demo2/a/b/c/test.html
-
ServletA请求转发路径 : …/…/a/b/c/test/html
-
寻找方式就是在当前资源所在路径(http://localhost:8080/demo2/x/y/)后拼接(…/…/a/b/c/test/html),形成(http://localhost:8080/demo2/x/y/…/…/a/b/c/test/html)每个…/抵消一层目录,正好是目标资源正常获取的url(http://localhost:8080/demo2/a/b/c/test/html)
@WebServlet("/x/y/servletB") public class ServletB extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {RequestDispatcher requestDispatcher = req.getRequestDispatcher("../../a/b/c/test.html");requestDispatcher.forward(req,resp);} }
10.2.3.2绝对路径写法
-
请求转发只能转发到项目内部的资源,其绝对路径无需添加项目上下文路径
-
请求转发绝对路径的基准路径相当于http://localhost:8080/demo2
-
在项目上下文路径为缺省值时,也无需改变,直接以/开头即可
@WebServlet("/x/y/servletB") public class ServletB extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {RequestDispatcher requestDispatcher = req.getRequestDispatcher("/a/b/c/test.html");requestDispatcher.forward(req,resp);} }
10.2.3.3目标资源内相对路径处理
-
此时需要注意,请求转发是服务器行为,浏览器不知道,地址栏不变化,相当于我们访问test.html的路径为http://localhost:8080/demo2/x/y/servletB
-
那么此时 test.html资源的所在路径就是http://localhost:8080/demo2/x/y/所以test.html中相对路径要基于该路径编写,如果使用绝对路径则不用考虑
<!DOCTYPE html> <html lang="en"> <head><meta charset="UTF-8"><title>Title</title> </head> <body><!--当前资源路径是 http://localhost:8080/demo2/x/y/servletB当前资源所在路径是 http://localhost:8080/demo2/x/y/目标资源路径=所在资源路径+src属性值 http://localhost:8080/demo2/x/y/../../static/img/logo.pnghttp://localhost:8080/demo2/static/img/logo.png得到目标路径正是目标资源的访问路径 --> <img src="../../static/img/logo.png"> </body> </html>
MVC架构模式
MVC(Model View Controller)是软件工程中的一种**
软件架构模式
,它把软件系统分为模型
、视图
和控制器
**三个基本部分。用一种业务逻辑、数据、界面显示分离的方法组织代码,将业务逻辑聚集到一个部件里面,在改进和个性化定制界面及用户交互的同时,不需要重新编写业务逻辑。
-
M:Model 模型层,具体功能如下
- 存放和数据库对象的实体类以及一些用于存储非数据库表完整相关的VO对象
- 存放一些对数据进行逻辑运算操作的的一些业务处理代码
-
V:View 视图层,具体功能如下
- 存放一些视图文件相关的代码 html css js等
- 在前后端分离的项目中,后端已经没有视图文件,该层次已经衍化成独立的前端项目
-
C:Controller 控制层,具体功能如下
- 接收客户端请求,获得请求数据
- 将准备好的数据响应给客户端
MVC模式下,项目中的常见包
-
M:
- 实体类包(pojo /entity /bean) 专门存放和数据库对应的实体类和一些VO对象
- 数据库访问包(dao/mapper) 专门存放对数据库不同表格CURD方法封装的一些类
- 服务包(service) 专门存放对数据进行业务逻辑运算的一些类
-
C:
- 控制层包(controller)
-
V:
- web目录下的视图资源 html css js img 等
- 前端工程化后,在后端项目中已经不存在了
非前后端分离的MVC
前后端分离的MVC
相关文章:

JAVAWeb中的Servlet学习
一 Servlet简介 1.1动态资源和静态资源 静态资源 无需在程序运行时通过代码运行生成的资源,在程序运行之前就写好的资源.例如:html css js img ,音频文件和视频文件 动态资源 需要在程序运行时通过代码运行生成的资源,在程序运行之前无法确定的数据,运行时动态生成,例如Servle…...

docker安装ddns-go(外网连接局域网)
docker先下载镜像,目前最新版是v6.7.6 也可以csdn资源下载 再导入dockers https://download.csdn.net/download/u014756339/90096748 docker load -i ddns-go.tar 启动 docker run -d --name ddns-go --restartalways --nethost -v /opt/ddns-go:/root jeessy/…...
时间复杂度度详解
时间复杂度是衡量算法性能的重要指标,用来描述算法随着输入规模 n 增大,运行时间的增长趋势。以下是时间复杂度的核心概念与常见分类的详细讲解。 1. 时间复杂度的定义 时间复杂度反映了算法执行的 基本操作数量 与输入规模 n 的关系。它通常使用大 O表示法来表示,即: 其中…...

如何处理和优化大文件上传和下载
如何处理和优化大文件上传和下载 简单来说 文件过大会导致内存溢出,上传和下载过慢会影响用户体验,不合理的设计可能引发安全问题,还有网络问题,数据完整性,服务器压力等 文件过大,内存溢出,…...

QT 线程锁
在 Qt 中,线程锁是用来同步多线程访问共享资源的机制,防止数据竞争和线程安全问题。Qt 提供了几种线程锁和同步工具,主要包括以下几种: 1. QMutex 功能:QMutex 是 Qt 中最常用的互斥锁(mutex)…...

光猫开DMZ教程
本教程以移动光猫未例,具体操作以实际光猫为准 1、登录移动光猫管理后台 打开浏览器,在浏览器地址栏输入移动光猫登录管理地址192.168.1.1或者tplogin.cn 按“回车键”打开登录页面,然后输入路由器管理密码登录。 移动光猫登录页面 超级密…...

分区之间的一种度量方法-覆盖度量(Covering Metric)
分区之间的一种度量方法——覆盖度量(Covering Metric),用于量化一个分区如何被另一个分区覆盖或近似。以下是逐步详细解释: 1. 背景与符号说明 分区的概念: 分区是将一个集合(这里是 { 1 , … , n } \{…...

cocos creator接入字节跳动抖音小游戏JSAPI敏感词检测(进行文字输入,但输入敏感词后没有替换为*号)
今天更新了某个抖音小游戏的版本,增加了部分剧情,半天过后一条短信审核未通过,emmm…抖音总是能给开发者惊喜…打开电脑看看这次又整什么幺蛾子… 首先是一脸懵逼,后端早已接入了官方的内容安全检测能力了(https://de…...

13.Java IO 流(文件流、字符流、字符处理流、字节处理流、对象处理流、标准流、转换流、打印流、Properties 配置文件、其他流)
一、文件引入 1、文件的概念 文件是保存数据的地方(例如,文档,图片,音视频等) 2、文件流 流:数据在数据源(文件)和程序(内存)之间经历的路径 输入流&…...

掌握 DOM 操作:让你的网页动起来
文章目录 前言一、什么是 DOM?二、DOM 树的结构三、使用 JavaScript 操作 DOM总结前言 在现代 Web 开发中,动态交互几乎是每个网站的标配。而这种交互的实现,离不开 DOM(Document Object Model) 的操作。本次课程深入讲解了 DOM 的基础知识以及如何使用 JavaScript 操作 …...

JVM整理部分面试题
1.如何主动触发垃圾回收? 在Java中,垃圾回收是自动进行的,由Java虚拟机(JVM)负责管理。但是,有时候我们可能希望手动触发垃圾回收以释放一些无用的对象。这可以通过调用System.gc()方法来实现 手动触发垃…...

ubuntu20 使用 pyspacemouse获取 spacemouse wireless 输入
1. 设置设备权限 (1) 默认情况下,普通用户可能没有权限访问 HID 设备,可以通过设置 udev 规则解决: cd /etc/udev/rules.d sudo touch 99-spacemouse.rules sudo gedit 99-spacemouse.rules在新建的99-spacemouse.rules中添加以下内容 SUB…...

windows下Qt5自动编译配置QtMqtt环境(11)
文章目录 [toc]1、概述2、准备1.1 下载源码1.2 配置环境1.3 解释原理 3、编译4、验证5、参考6、视频 更多精彩内容👉内容导航 👈👉Qt网络编程 👈 1、概述 Qt默认是不包含mqtt库的,如果需要使用到mqtt库就只能自己编译配…...

速盾:高防cdn有哪些冷知识?
高防CDN(Content Delivery Network)是一种用于提供高可靠性、高性能的互联网服务的技术。它通过将内容分发到离用户最近的服务器上,并优化网络流量,以提升网站的响应速度和承载能力。除了这些基本的了解,下面是一些高防…...

全面UI组件库Telerik 2024 Q4全新发布——官方宣布支持.NET 9
Telerik DevCraft包含一个完整的产品栈来构建您下一个Web、移动和桌面应用程序。它使用HTML和每个.NET平台的UI库,加快开发速度。Telerik DevCraft提供最完整的工具箱,用于构建现代和面向未来的业务应用程序,目前提供UI for ASP.NET MVC、Ken…...

电脑投屏到电脑:Windows,macOS及Linux系统可以相互投屏!
本篇其实是电脑远程投屏到另一台电脑的操作介绍。本篇文章的方法可用于Windows,macOS及Linux系统的相互投屏。 为了避免介绍过程中出现“这台电脑”投屏到“那台电脑”的混乱表述,假定当前屏幕投出端是Windows系统电脑,屏幕接收端是Linux系统…...

12.6深度学习_模型优化和迁移_模型移植
八、模型移植 1. 认识ONNX https://onnx.ai/ Open Neural Network Exchange(ONNX,开放神经网络交换)格式,是一个用于表示深度学习模型的标准,可使模型在不同框架之间进行转移。 ONNX的规范及代码主要由微软…...

Grid++Report:自定义模板设计(自由表格使用),详细教程
实现效果 步骤 一、新建空白 初始状态都是空白页,如果不是,点击右上角->文件->新建空白 二、页面设置 右击页面灰色部分->页面设置 根据需求自定义页面 三、报表头设计 1、新增报表头 右击屏幕->新增->报表节->报表头 点击报表头…...

[Collection与数据结构] 位图与布隆过滤器
🌸个人主页:https://blog.csdn.net/2301_80050796?spm1000.2115.3001.5343 🏵️热门专栏: 🧊 Java基本语法(97平均质量分)https://blog.csdn.net/2301_80050796/category_12615970.html?spm1001.2014.3001.5482 🍕 Collection与…...

idea中新建一个空项目
目的,为了在同一个目录下有多个小的项目:使用IDE为idea2022。 步骤: 点击新建项目,点击创建空项目,这里选择空项目是将其作为其他项目的一个容器,如图所示: 然后点击文件->项目结构…...

【Python】【Conda 】Conda 与 venv 虚拟环境优缺点全解:如何做出明智选择
目录 引言一、基本概念1.1 Conda 虚拟环境1.2 Python venv 虚拟环境 二、主要区别对比三、优缺点分析3.1 Conda 虚拟环境的优缺点3.2 Python venv 虚拟环境的优缺点 四、使用场景推荐4.1 使用 Conda 虚拟环境的场景4.2 使用 Python venv 虚拟环境的场景 五、虚拟环境管理工具对…...

深度学习在故障检测中的应用:从理论到实践
随着工业设备和信息系统的复杂性增加,故障检测成为企业运维的重要任务。然而,传统的基于规则或统计学的故障检测方法难以应对复杂多变的故障模式。深度学习作为一种强大的数据分析工具,为故障检测提供了新的解决思路。本文将介绍深度学习模型…...

自然语言处理与人工智能
自然语言处理(NLP)与人工智能(AI) 自然语言处理(NLP)是人工智能(AI)领域的一个重要分支,旨在使计算机能够理解、解释和生成自然语言。随着深度学习技术的进步࿰…...

量化交易系统开发-实时行情自动化交易-8.15.Ptrade/恒生平台
19年创业做过一年的量化交易但没有成功,作为交易系统的开发人员积累了一些经验,最近想重新研究交易系统,一边整理一边写出来一些思考供大家参考,也希望跟做量化的朋友有更多的交流和合作。 接下来会对于Ptrade/恒生平台介绍。 P…...

非常简单实用的前后端分离项目-仓库管理系统(Springboot+Vue)part 4
三十三、出入库管理 Header.vue导一下,RecordController加一个 //将入库数据和原有数据相加吧//新增PostMapping("/save")public Result save(RequestBody Record record) {return recordService.save(record) ? Result.success() : Result.fail();} GoodsManage.v…...

基于MATLAB的信号处理工具:信号分析器
信号(或时间序列)是与特定时间相关的一系列数字或测量值,不同的行业和学科将这一与时间相关的数字序列称为信号或时间序列。生物医学或电气工程师会将其称为信号,而统计学家或金融定量分析师会使用时间序列这一术语。例如…...

Codeforces Round 784 (Div. 4)
题目链接 A. Division? 题意 思路 模拟即可 示例代码 void solve() {int n;cin >> n;int ans;if(n > 1900) ans 1;else if(n > 1600) ans 2;else if(n > 1400) ans 3;else ans 4;cout << "Division " << ans << \n;}B. T…...

OpenNebula 开源虚拟平台,对标 VMware
Beeks Group 主要为金融服务提供商提供虚拟专用服务器和裸机服务器。该公司表示,转向 OpenNebula 不仅大幅降低了成本,还使其虚拟机效率提升了 200%,并将更多裸机服务器资源用于客户端负载,而非像以往使用 VMware 时那样用于虚拟机…...

软件项目标书参考,合同拟制,开发合同制定,开发协议,标书整体技术方案,实施方案,通用套用方案,业务流程,技术架构,数据库架构全资料下载(原件)
1、终止合同协议书 2、项目合作协议 3、合同交底纪要 4、合同管理台账 软件资料清单列表部分文档清单:工作安排任务书,可行性分析报告,立项申请审批表,产品需求规格说明书,需求调研计划,用户需求调查单&…...

Jenkins环境一站式教程:从安装到配置,打造高效CI/CD流水线环境-Ubuntu 22.04.5 环境离线安装配置 Jenkins 2.479.1
文章目录 Jenkins环境一站式教程:从安装到配置,打造高效CI/CD流水线环境-Ubuntu 22.04.5 环境离线安装配置 Jenkins 2.479.1一、环境准备1.1 机器规划1.2 环境配置1.2.1 设置主机名1.2.2 停止和禁用防火墙1.2.3 更新系统 二、安装配置Jenkins2.1 安装JDK…...