当前位置: 首页 > news >正文

Day40

Day40

监听器

概念:
监听器用于监听web应用中某些对象信息的创建、销毁、增加,修改,删除等动作的
发生,然后作出相应的响应处理。当范围对象的状态发生变化的时候,服务器自动调用
监听器对象中的方法。
常用于统计在线人数和在线用户,系统加载时进行信息初始化,统计网站的访问量等。

类别:

第一大类:监听请求域、会话域、全局域对象的创建和销毁。

第二大类:监听请求域、会话域、全局域对象的属性的创建、替换、销毁。

第三大类:监听会话域(String key - Object value)里value的绑定和解绑、钝化和活化。

第一大类

监听请求域:

servlet:

package com.qf.servlet;import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/Servlet01")
public class Servlet01 extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {this.doPost(request, response);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("执行Servlet");}
}

web.xml:

<listener><listener-class>com.qf.listen.MyServletRequestListener</listener-class></listener>

listener:

package com.qf.listen;import javax.servlet.ServletRequest;
import javax.servlet.ServletRequestEvent;
import javax.servlet.ServletRequestListener;public class MyServletRequestListener implements ServletRequestListener {@Overridepublic void requestDestroyed(ServletRequestEvent servletRequestEvent) {ServletRequest request = servletRequestEvent.getServletRequest();System.out.println("请求对象被销毁了"+request);}@Overridepublic void requestInitialized(ServletRequestEvent servletRequestEvent) {ServletRequest request = servletRequestEvent.getServletRequest();System.out.println("请求对象被创建了"+request);}
}

注:请求对象的生命周期为,创建:客户端发送请求,服务器会创建请求对象。销毁:服务器返回响应后,会把请求对象、响应对象一并销毁。

监听会话域

注意:查看生命周期要把index.jsp改名或者删除,因为项目启动会默认进入index.jsp,会创建会话对象。

web.xml:(手动调整session销毁时间)

<listener><listener-class>com.qf.listen.MyHttpSessionListener</listener-class></listener><session-config><session-timeout>1</session-timeout></session-config>

创建一个page01.jsp。

MyHttpSessionListener:

package com.qf.listen;import javax.servlet.http.HttpSession;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener;public class MyHttpSessionListener implements HttpSessionListener {@Overridepublic void sessionCreated(HttpSessionEvent httpSessionEvent) {HttpSession session = httpSessionEvent.getSession();System.out.println("会话对象被创建了"+session.getId());}@Overridepublic void sessionDestroyed(HttpSessionEvent httpSessionEvent) {System.out.println("会话对象被销毁了");}
}

注:会话对象的什么周期,创建:服务器使用到了session->request.getSession();销毁:session对象设置的过期时间到后就会被销毁。

思考1:客户端访问html资源是否会创建会话对象?答:不会

思考2:客户端访问jsp资源会创建会话对象?为什么?答:会,因为jsp里9大内置对象中有session对象。

思考3:session对象创建后,客户端发送下一次请求是为什么能找到之前的session对象?答:因为客户端在cookie中存储了session的id->JSESSIONID

思考4:session对象是存活在服务器中的,默认为30分钟的生命周期,可客户端存储的cookie里的JSESSIONID为会话结束时,也就意味着关闭浏览器后,再也找不到之前存储在服务器中的session对象,那怎么解决?

答:设置cookie中JSESSION的过期时间。

package com.qf.servlet;import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.*;
import java.io.IOException;
@WebServlet("/Servlet01")
public class Servlet01 extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {this.doPost(request, response);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("Servlet收到请求");HttpSession session = request.getSession();Cookie cookie = new Cookie("JSESSION",session.getId());cookie.setMaxAge(60*30);response.addCookie(cookie);response.getWriter().println("abc");}
}
监听全局域

web.xml:

<context-param><param-name>info</param-name><param-value>abc</param-value></context-param><listener><listener-class>com.qf.listen.MyServletContextListener</listener-class></listener>

MyServletContextListener:

package com.qf.listen;import javax.servlet.ServletContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;public class MyServletContextListener implements ServletContextListener {@Overridepublic void contextInitialized(ServletContextEvent servletContextEvent) {ServletContext servletContext = servletContextEvent.getServletContext();String info = servletContext.getInitParameter("info");System.out.println("全局对象被创建了"+"--"+info);}@Overridepublic void contextDestroyed(ServletContextEvent servletContextEvent) {System.out.println("全局对象被销毁了");}
}

注:1.生命周期为项目启动时创建,服务器正常关闭时销毁。

经验:应用场景:

监听全局域对象的创建和销毁,而全局域对象在项目启动时创建,就意味着监听器里的初始化方法在项目启动时就会被调用,所以我们可以把项目初始化的工作放在该方法里。在配置文件中context-param里设置初始数据(建议放在最上面),在监听器的初始化方法中可以用getInitParameter()方法获取数据。

第一大类使用案例-统计在线人数

在全局域监听器中创建:

        servletContext.setAttribute("count",0);

在会话域监听器中统计:

package com.qf.listen;import javax.servlet.ServletContext;
import javax.servlet.http.HttpSession;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener;public class MyHttpSessionListener implements HttpSessionListener {@Overridepublic void sessionCreated(HttpSessionEvent httpSessionEvent) {HttpSession session = httpSessionEvent.getSession();System.out.println("会话对象被创建了"+session.getId());ServletContext servletContext = session.getServletContext();int count = (int) servletContext.getAttribute("count");count++;servletContext.setAttribute("count",count);}@Overridepublic void sessionDestroyed(HttpSessionEvent httpSessionEvent) {System.out.println("会话对象被销毁了");HttpSession session = httpSessionEvent.getSession();ServletContext servletContext = session.getServletContext();int count = (int) servletContext.getAttribute("count");count--;servletContext.setAttribute("count",count);}
}

第二大类

请求属性监听器

MyServletRequestAttributeListener(用注解配置):

package com.qf.listen02;import javax.servlet.ServletRequestAttributeEvent;
import javax.servlet.ServletRequestAttributeListener;
import javax.servlet.annotation.WebListener;@WebListener
public class MyServletRequestAttributeListener implements ServletRequestAttributeListener {@Overridepublic void attributeAdded(ServletRequestAttributeEvent servletRequestAttributeEvent) {String name = servletRequestAttributeEvent.getName();Object value = servletRequestAttributeEvent.getValue();System.out.println("请求域对象添加了属性:"+name+"---"+value);}@Overridepublic void attributeRemoved(ServletRequestAttributeEvent servletRequestAttributeEvent) {String name = servletRequestAttributeEvent.getName();Object value = servletRequestAttributeEvent.getValue();System.out.println("请求域对象删除了属性:"+name+"---"+value);}@Overridepublic void attributeReplaced(ServletRequestAttributeEvent servletRequestAttributeEvent) {String name = servletRequestAttributeEvent.getName();Object value = servletRequestAttributeEvent.getValue();//获取被替换的值System.out.println("请求域对象替换了属性:"+name+"---"+value);}
}

page.jsp:

<%//操作请求域的属性request.setAttribute("msg","aaabbbccc");//添加属性request.setAttribute("msg","bbbcccddd");//替换属性request.removeAttribute("msg");//删除属性
%>

会话属性监听器

MyHttpSessionAttributeListener:

package com.qf.listen02;import javax.servlet.http.HttpSessionAttributeListener;
import javax.servlet.http.HttpSessionBindingEvent;public class MyHttpSessionAttributeListener implements HttpSessionAttributeListener {@Overridepublic void attributeAdded(HttpSessionBindingEvent httpSessionBindingEvent) {String name = httpSessionBindingEvent.getName();Object value = httpSessionBindingEvent.getValue();System.out.println("会话域对象添加了属性");}@Overridepublic void attributeRemoved(HttpSessionBindingEvent httpSessionBindingEvent) {String name = httpSessionBindingEvent.getName();Object value = httpSessionBindingEvent.getValue();System.out.println("会话域对象删除了属性");}@Overridepublic void attributeReplaced(HttpSessionBindingEvent httpSessionBindingEvent) {String name = httpSessionBindingEvent.getName();Object value = httpSessionBindingEvent.getValue();System.out.println("会话域对象替换了属性");}
}

page02.jsp:

<%//操作会话域的属性session.setAttribute("msg","aaabbbccc");//添加属性session.setAttribute("msg","bbbcccddd");//替换属性session.removeAttribute("msg");//删除属性
%>

全局属性监听器

MyServletContextAttributeListener:

package com.qf.listen02;import javax.servlet.ServletContextAttributeEvent;
import javax.servlet.ServletContextAttributeListener;public class MyServletContextAttributeListener implements ServletContextAttributeListener {@Overridepublic void attributeAdded(ServletContextAttributeEvent servletContextAttributeEvent) {String name = servletContextAttributeEvent.getName();Object value = servletContextAttributeEvent.getValue();System.out.println("全局对象创建了属性:"+"--"+name+"--"+value);}@Overridepublic void attributeRemoved(ServletContextAttributeEvent servletContextAttributeEvent) {String name = servletContextAttributeEvent.getName();Object value = servletContextAttributeEvent.getValue();System.out.println("全局对象创建了属性:"+"--"+name+"--"+value);}@Overridepublic void attributeReplaced(ServletContextAttributeEvent servletContextAttributeEvent) {String name = servletContextAttributeEvent.getName();Object value = servletContextAttributeEvent.getValue();System.out.println("全局对象创建了属性:"+"--"+name+"--"+value);}
}

page03.jsp:

<%//操作全局域的属性application.setAttribute("msg","aaabbbccc");//添加属性application.setAttribute("msg","bbbcccddd");//替换属性application.removeAttribute("msg");//删除属性
%>

第三大类

HttpSessionBindingListener:监听JavaBean对象在session中的绑定(添加)和解绑(删除),即监听某一个具体的类对象,第二大类是监听所有的类对象。

HttpSessionActivationListener:监听JavaBean对象在session中的钝化(序列化)和活化(反序列化)

User:(实现HttpSessionBindingListener,HttpSessionActivationListener,Serializable)

package com.qf.listen03;import javax.servlet.http.HttpSessionActivationListener;
import javax.servlet.http.HttpSessionBindingEvent;
import javax.servlet.http.HttpSessionBindingListener;
import javax.servlet.http.HttpSessionEvent;
import java.io.Serializable;public class User implements HttpSessionBindingListener, HttpSessionActivationListener, Serializable {private String username;private String password;private String name;private String nickName;public User(String username, String password, String name, String nickName) {this.username = username;this.password = password;this.name = name;this.nickName = nickName;}public User() {}public String getUsername() {return username;}public void setUsername(String username) {this.username = username;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getNickName() {return nickName;}public void setNickName(String nickName) {this.nickName = nickName;}@Overridepublic String toString() {return "User{" +"username='" + username + '\'' +", password='" + password + '\'' +", name='" + name + '\'' +", nickName='" + nickName + '\'' +'}';}@Overridepublic void sessionWillPassivate(HttpSessionEvent httpSessionEvent) {System.out.println(this+"被session钝化了");}@Overridepublic void sessionDidActivate(HttpSessionEvent httpSessionEvent) {System.out.println(this+"被session活化了");}@Overridepublic void valueBound(HttpSessionBindingEvent httpSessionBindingEvent) {System.out.println(this+"绑定到session中了");}@Overridepublic void valueUnbound(HttpSessionBindingEvent httpSessionBindingEvent) {System.out.println(this+"从session中解绑了");}
}

在web里面创建META-INF文件夹,写一个context.xml:

<?xml version="1.0" encoding="UTF-8"?>
<Context><Manager className="org.apache.catalina.session.PersistentManager" maxIdleSwap="1"><Store className="org.apache.catalina.session.FileStore" directory="C:\\text"/></Manager>
</Context>

Servlet02:

package com.qf.servlet;import com.qf.listen03.User;import javax.servlet.ServletException;
import javax.servlet.annotation.WebListener;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;
@WebServlet("/Servlet02")
public class Servlet02 extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {this.doPost(request, response);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("Servlet02接收到请求了...");HttpSession session = request.getSession();//注意:将user对象添加到session中,可以叫做User对象绑定到session中 --- 绑定session.setAttribute("user",new User("1233211234567","123132","zs","fwkt"));//注意:将user对象从session中删除,可以叫做User对象从session中解绑了 --- 解绑session.removeAttribute("user");}
}

Servlet03:

package com.qf.servlet;import com.qf.listen03.User;import javax.servlet.ServletException;
import javax.servlet.annotation.WebListener;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;@WebServlet("/Servlet03")
public class Servlet03 extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {this.doPost(request, response);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("Servlet03接收到请求了...");HttpSession session = request.getSession();session.setAttribute("user",new User("1233211234567","123132","zs","fwkt"));}
}

注意:将User对象添加到session中,可以叫做User对象绑定到session中 — 绑定

将User对象从session中删除,可以叫做User对象从session中解绑 — 解绑

面试题:session的钝化与活化:

在这里插入图片描述

过滤器

简介

Filter:过滤器,拦截访问web资源的请求与响应操作

Servlet API中提供了Filter接口,开发web应用时,如果编写的Java类实现了这个接口,则把这个Java类称之为过滤器。

创建与使用

在这里插入图片描述

场景:welcome.html向Servlet01发送请求:

package com.qf.servlet;import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/Servlet01.action")
public class Servlet01 extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {this.doPost(request, response);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {request.setCharacterEncoding("UTF-8");response.setContentType("text/html;charset=UTF-8");System.out.println("Servlet收到请求了");}
}

过滤器:

package com.qf.filter;import javax.servlet.*;
import java.io.IOException;public class Filter01 implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {System.out.println("Filter01 --- init()");}@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {System.out.println("Filter01 --- doFilter() -- 放行前");filterChain.doFilter(servletRequest,servletResponse);System.out.println("Filter01 --- doFilter() -- 放行后");}@Overridepublic void destroy() {System.out.println("Filter01 --- destroy()");}
}

配置文件:

<welcome-file-list><welcome-file>welcome.html</welcome-file></welcome-file-list><filter><filter-name>Filter01</filter-name><filter-class>com.qf.filter.Filter01</filter-class></filter><filter-mapping><filter-name>Filter01</filter-name><url-pattern>*.action</url-pattern></filter-mapping>

生命周期

创建:项目启动时创建 – 构造方法、init()

销毁:项目正常关闭时销毁 – destroy()

注意:Filter对象是单例的。

当有多个过滤器时,

配置文件:

<filter><filter-name>Filter01</filter-name><filter-class>com.qf.filter.Filter01</filter-class></filter><filter-mapping><filter-name>Filter01</filter-name><url-pattern>*.action</url-pattern></filter-mapping><filter><filter-name>Filter02</filter-name><filter-class>com.qf.filter.Filter01</filter-class></filter><filter-mapping><filter-name>Filter02</filter-name><url-pattern>*.action</url-pattern></filter-mapping><filter><filter-name>Filter03</filter-name><filter-class>com.qf.filter.Filter01</filter-class></filter><filter-mapping><filter-name>Filter03</filter-name><url-pattern>*.action</url-pattern></filter-mapping>

思考:多个Filter的创建顺序?多个Filter调用顺序?

创建顺序:无序,底层是多线程抢资源。调用顺序为web.xml里的配置顺序(推荐)。如果是使用注解配置,是按照类名字典排序。

案例1 编码过滤器

WEB-学生管理系统中写一个Encoderfilter类进行编码过滤(就不必在servlet中设置了)

配置文件(标准写法要求匹配所有的action,但是由于之前项目中servlet后没有加.action,故此处偷懒直接写/*):

<welcome-file-list><welcome-file>welcome.html</welcome-file></welcome-file-list><filter><filter-name>EncoderFilter</filter-name><filter-class>com.qf.filter.EncoderFilter</filter-class></filter><filter-mapping><filter-name>EncoderFilter</filter-name><url-pattern>/*</url-pattern></filter-mapping>

过滤器:

package com.qf.filter;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;public class EncoderFilter implements Filter {private String code;@Overridepublic void init(FilterConfig filterConfig) throws ServletException {code = filterConfig.getInitParameter("code");}@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest) servletRequest;HttpServletResponse response = (HttpServletResponse) servletResponse;request.setCharacterEncoding(code);response.setContentType("text/html;charset="+code);filterChain.doFilter(request,response);}@Overridepublic void destroy() {}
}
案例2 登录权限过滤器

写一个LoginFilter。

配置文件:

<filter><filter-name>LoginFilter</filter-name><filter-class>com.qf.filter.LoginFilter</filter-class></filter><filter-mapping><filter-name>LoginFilter</filter-name><url-pattern>/*</url-pattern></filter-mapping>

LoginFilter:

package com.qf.filter;import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;public class LoginFilter implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {}@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest) servletRequest;HttpServletResponse response = (HttpServletResponse) servletResponse;String requestURI = request.getRequestURI();System.out.println(requestURI);
//        StringBuffer requestURL = request.getRequestURL();
//        System.out.println(requestURL);HttpSession session = request.getSession();String username = (String) session.getAttribute("username");String role = (String) session.getAttribute("role");String name = (String) session.getAttribute("name");if(requestURI.equals("/Day37_war_exploded/register.jsp")||requestURI.equals("/Day37_war_exploded/login.jsp")||requestURI.equals("/Day37_war_exploded/CodeServlet")||requestURI.equals("/Day37_war_exploded/LoginServlet")||requestURI.equals("/Day37_war_exploded/")||requestURI.equals("/Day37_war_exploded/welcome.html")){filterChain.doFilter(request,response);}else{if(username!=null&&role!=null&&name!=null){if("student".equals(role) && requestURI.contains("QueryAllStuServlet")) {response.sendRedirect("index.jsp");}else{filterChain.doFilter(request,response);}}}}@Overridepublic void destroy() {}
}
案例3 敏感词过滤器

场景:在学生详情页面写一个建议,跳转到proposal.jsp,里面写一个表单,提交到ProposalServlet

<%--Created by IntelliJ IDEA.User: GuDate: 2024-06-17Time: 17:37To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head><title>Title</title>
</head>
<body><h2>建议</h2><form action="ProposalServlet" method="post">建议:<input type="text" name="proposal"/><input type="submit" value="提交"/><button type="button" οnclick="fun01()">返回</button></form><script type="text/javascript">function fun01(){window.location = "index.jsp";}</script>
</body>
</html>

ProposalServlet(输出建议):

package com.qf.servlet;import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/ProposalServlet")
public class ProposalServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {this.doPost(request, response);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {String proposal = (String) request.getParameter("proposal");response.getWriter().println(proposal);}
}

MyHttpServletRequestWrapper包装类,继承HttpServletRequestWrapper类(构造方法、重写getParameter方法):

package com.qf.pojo;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;public class MyHttpServletRequestWrapper extends HttpServletRequestWrapper {public MyHttpServletRequestWrapper(HttpServletRequest request) {super(request);}@Overridepublic String getParameter(String name) {String parameter = super.getParameter(name);if(parameter!=null){parameter = parameter.replaceAll("<","&lt;");parameter = parameter.replaceAll(">","&gt;");parameter = parameter.replaceAll("傻逼","**");}return parameter;}
}

SensitiveWordsFilter:

package com.qf.filter;import com.qf.pojo.MyHttpServletRequestWrapper;import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;public class SensitiveWordFilter implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {}@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest) servletRequest;MyHttpServletRequestWrapper requestWrapper = new MyHttpServletRequestWrapper(request);filterChain.doFilter(requestWrapper,servletResponse);}@Overridepublic void destroy() {}
}

相关文章:

Day40

Day40 监听器 概念&#xff1a; 监听器用于监听web应用中某些对象信息的创建、销毁、增加&#xff0c;修改&#xff0c;删除等动作的 发生&#xff0c;然后作出相应的响应处理。当范围对象的状态发生变化的时候&#xff0c;服务器自动调用 监听器对象中的方法。 常用于统计在线…...

linux基础 - 内核的基础概念

目录 零. 前言 一. 源码简介 二. 存储管理 物理内存管理&#xff1a; 虚拟内存管理&#xff1a; 内存分配与回收&#xff1a; 三. CPU 和进程管理 进程管理&#xff1a; CPU 管理&#xff1a; 四. 文件系统 文件系统的概念 常见的 Linux 文件系统类型 文件系统的工…...

centos7系统使用docker-compose安装部署jenkins

CentOS7系统使用docker-compose安装部署jenkins&#xff0c;并实现前后端自动构建 记录一次工作中部署jenkins的真实经历&#xff0c;总结了相关经验 1.准备环境 1.java 由于最新的jenkins需要jdk11以上才能支持&#xff0c;而系统里的jdk是1.8的&#xff0c;因此等jenkins安…...

传染病报卡内容——丙型

--丙型 select a.morbiditdate 发病日期, diagnosedate 诊断日期, a.deathdate 死亡日期, a.casetypequality 病例分类,a.hcvrna "HCR_RNA定量" from zl_sdmb.t_报卡记录 t, c1_infectiousv1_6 a where t.id a.fileid and t.卡片种类 传…...

本地快速部署大语言模型开发平台Dify并实现远程访问保姆级教程

文章目录 前言1. Docker部署Dify2. 本地访问Dify3. Ubuntu安装Cpolar4. 配置公网地址5. 远程访问6. 固定Cpolar公网地址7. 固定地址访问 前言 本文主要介绍如何在Linux Ubuntu系统使用Docker快速部署大语言模型应用开发平台Dify,并结合cpolar内网穿透工具实现公网环境远程访问…...

《Cloud Native Data Center Networking》(云原生数据中心网络设计)读书笔记 -- 02 Clos拓扑

本章回答以下问题&#xff1a; 什么是 Clos 拓扑&#xff0c;它与“接入 - 汇聚 - 核心”拓扑有何不同?Clos 拓扑的特征是什么?Clos 拓扑对数据中心网络的影响是什么? Clos拓扑 云原生数据中心基础设施的先行者们想要构建一种支持大规模水平扩展网络。 基本的Clos拓扑如图…...

VUE3版本新特性

VUE3版本新特性 VUE3和VUE2的区别路由的使用vite安装项目新特性使用 1.VUE3和VUE2的区别 2020年9月18日&#xff0c;Vue.js发布版3.0版本&#xff0c;代号&#xff1a;One Piece 于 2022 年 2 月 7 日星期一成为新的默认版本! Vue3性能更高,初次渲染快55%, 更新渲染快133% 。…...

【Oracle篇】Oracle数据库坏块处理:rman修复坏块实践与案例分析(第七篇,总共八篇)

&#x1f4ab;《博主介绍》&#xff1a;✨又是一天没白过&#xff0c;我是奈斯&#xff0c;DBA一名✨ &#x1f4ab;《擅长领域》&#xff1a;✌️擅长Oracle、MySQL、SQLserver、阿里云AnalyticDB for MySQL(分布式数据仓库)、Linux&#xff0c;也在扩展大数据方向的知识面✌️…...

学懂C#编程:从一个简单的例子理解事件处理

在C#中&#xff0c;事件是一种特殊的委托类型&#xff0c;用于在对象上发生某些事情时通知订阅者。事件的处理通常包括定义事件&#xff0c;创建触发事件的条件&#xff0c;以及订阅该事件的事件处理程序。 以下是一个简单的C#事件处理示例&#xff1a; using System;// 定义…...

深入理解指针(2)

4. const 修饰指针 4.1 const修饰变量 变量是可以修改的&#xff0c;如果把变量的地址交给⼀个指针变量&#xff0c;通过指针变量的也可以修改这个变量。 但是如果我们希望⼀个变量加上⼀些限制&#xff0c;不能被修改&#xff0c;怎么做呢&#xff1f;这就是const的作⽤。 …...

C#.Net筑基-集合知识全解

01、集合基础知识 .Net 中提供了一系列的管理对象集合的类型&#xff0c;数组、可变列表、字典等。从类型安全上集合分为两类&#xff0c;泛型集合 和 非泛型集合&#xff0c;传统的非泛型集合存储为Object&#xff0c;需要类型转。而泛型集合提供了更好的性能、编译时类型安全…...

AI PPT生成器,一键在线智能生成PPT工具

PPT作为商业沟通和教育培训中的重要工具&#xff0c;PPT制作对于我们来说并不陌生。但是传统的PPT制作不仅耗时&#xff0c;而且想要做出精美的PPT&#xff0c;需要具备一定的设计技能。下面小编就来和大家分享几款AI PPT工具&#xff0c;只要输入主题&#xff0c;内容就可以在…...

stm32学习笔记---零基础入门介绍2

目录 STM32介绍 STM32家族系列 ARM介绍 ARM内核型号种类 我们学习用的STM32 片上资源/外设&#xff08;Peripheral&#xff09; 命名规则 系统结构 引脚定义 STM32的启动配置 STM32最小系统电路和其他部分电路 最小系统板的实物图 附&#xff1a;安装软件准备 声明…...

搭建取图系统app源码开发,满足广泛应用需求

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 前言 图片已成为信息传递的重要媒介&#xff0c;广泛应用于各个领域。为满足日益增长的图片需求&#xff0c;搭建一款高效的取图系统&#xff0c;可以为用户提供便捷、全面的…...

语音质量评价方法之MOS

引言 在语音增强、语音合成、语音转换、声音转换、语音克隆、语音修复等等领域&#xff0c;常常要对输出的语音进行评价。对语音的质量评价一般关注两个方面&#xff0c;即主观评价和客观评价。主观评价就是人凭借听觉感受对语音进行打分&#xff0c;客观评价比较广泛&#xf…...

gorm简介

【1】ORM: 即Object-Relational Mapping,它的作用是在关系型数据库和对象之间作一个映射&#xff0c;这样我们在具体的操作数据库的时候&#xff0c;就不需要再去和复杂的SQL语句打交道&#xff0c;只要像平时操作对象一样操作它们就可以了。 【2】GORM gorm是go语言的一个orm…...

MySQL:SELECT list is not in GROUP BY clause 报错 解决方案

一、前言 一大早上测试环境&#xff0c;发现测试环境的MySQL报错了。 SELECT list is not in GROUP BY clause and contains nonaggregated column二、解决方案 官方文档中提到&#xff1a; 大致意思&#xff1a; 用于GROUP BY的SQL / 92标准要求满足以下条件&#xff1a; SE…...

IPython的使用技巧

1、解释说明 IPython是一个强大的Python交互式shell&#xff0c;它提供了丰富的功能&#xff0c;如自动补全、历史记录、内置帮助等。IPython使得在命令行下编写和测试Python代码变得更加方便和高效。 2、使用示例 安装IPython&#xff1a; pip install ipython启动IPython…...

Spring Boot 多线程例子

在Spring Boot中&#xff0c;多线程可以通过Java的并发工具来实现。以下是一些常见的多线程实现方法&#xff1a; 1. 使用Async注解和CompletableFuture&#xff1a; 首先&#xff0c;需要在Spring Boot应用的主类上添加EnableAsync注解&#xff0c;以启用异步支持。 java Spr…...

java干货 线程池的分析和使用

文章目录 一、了解线程池1.1 什么是线程池1.2 为什么需要线程池 二、四种线程池的使用2.1 newFixedThreadPool2.2 newCachedThreadPool2.3 newSingleThreadExecutor2.4 newScheduledThreadPool 三、自定义线程池3.1 线程池七大核心参数3.2 线程池内部处理逻辑 一、了解线程池 …...

文本张量入门

张量&#xff0c;英文为Tensor&#xff0c;是机器学习的基本构建模块&#xff0c;是以数字方式表示数据的形式。 张量的基本类型: 创建一个标量&#xff08;0维张量&#xff09;&#xff0c;也就是一个单独的数字 scalar torch.tensor(7) scalar.ndim # 返回张量的维度 0 # …...

js文字如何轮播?

<div class"td-style"> <span class"td-text">内容1内容1内容1内容1内容1内容1</span> </div> css&#xff1a; <style> .td-style { width: 160px; height: 72px; overflow: hidden; white-…...

Linux 五种IO模型

注&#xff1a;还有一种信号驱动IO&#xff0c;使用较少暂不讨论&#xff1b; 一&#xff0c;区分阻塞、非阻塞和同步、异步 看了很多文章对这两组概念解释和对比&#xff0c;说的太复杂了&#xff0c;其实没必要&#xff0c;两句话就能说清楚。 首先&#xff0c;对于读数据rec…...

深度解析响应式异步编程模型

上一篇文章中我们聊了一下线程池,基于线程池的多线程编程是我们在高并发场景下提升系统处理效率的有效手段,但却不是唯一的。今天我们来看一下另一种异步开发的常用手段-响应式编程模型 传统多线程模型的缺陷 多线程模型是目前应用最为广泛的并发编程手段,但凡遇到什么性能…...

一个软件是如何开发出来的呢?

一、前言 如今&#xff0c;AI大爆发的时代&#xff0c;作为一名IT从业者&#xff0c;你是否也想尝试开发一套自己的系统&#xff0c;实现那些看似有可能实现的天马行空的想法&#xff0c;变成一个优秀甚至伟大的产品&#xff0c;甚至带来某个行业的革新&#xff0c;那作为一名…...

宝塔板面有哪些优势

哈喽呀&#xff0c;大家好呀&#xff0c;淼淼又来和大家见面啦&#xff0c;在当今数字化时代&#xff0c;随着云计算和互联网技术的飞速发展&#xff0c;服务器管理成为了许多企业和个人开发者不可或缺的一部分。然而&#xff0c;传统服务器管理方式的复杂性和技术门槛往往令初…...

Mybatis中BaseEntity作用

新建各种对象的时候&#xff0c;一般来说&#xff0c;有几个属性是所有对象共有的&#xff0c;比如说id,is_del&#xff0c;is_enable这些&#xff0c;然后设置一个基础对象&#xff0c;以后新建所有对象的时候都继承它&#xff0c;就省的每次都要写这些共有的属性了...

IDEA2023中使用run Dashboard面板?实现批量运行微服务

1、直接点击Add service--->Run Configuration Type---->Spring Boot 2、这样就出现了run Dashboard面板&#xff0c;可同时运行多个工程模块&#xff0c;shift选中所有启动类组命名&#xff08;Group Configurations&#xff09; 3、启动所有的项目...

分数受限,鱼和熊掌如何兼得?专业or学校,这样选最明智!

文章目录 引言一、专业解析二、名校效应分析三、好专业和好学校的权衡结论个人建议 引言 24年高考帷幕落下&#xff0c;一场新的思考与选择悄然来临。对于每一位高考考生&#xff0c;学校和专业都是开启大学新生活的两个前置必选项。但有时候“鱼与熊掌不可兼得”&#xff0c;…...

CentOS 8.5 - 配置ssh的免密登录

文章目录 生成ssh密钥公钥内容放入服务器 生成ssh密钥 在本地主机安装 ssh工具&#xff0c;并生成公钥、私钥。 # 命令行输入 ssh-keygen -r rsa# 会在当前用户的家目录下生成一个.ssh目录公钥内容放入服务器 将上一步生成的id_rsa.pub公钥的内容复制到远程服务器 # 编辑文…...

电子政务网站建设出版社/网络营销的特点举例说明

很多朋友觉得PID是遥不可及&#xff0c;很神秘&#xff0c;很高大上的一种控制&#xff0c;对其控制原理也很模糊&#xff0c;只知晓概念性的层面&#xff0c;知其然不知其所以然&#xff0c;那么本期从另类视角来探究微分、积分电路的本质&#xff0c;意在帮助理解PID的控制原…...

设计网站横幅/网站建设费用都选网络

具体操作如下1. 关闭正在运行的 MySQL。2. 打开 DOS 窗口&#xff0c;转到mysql\bin目录。3. 输入mysqld --skip-grant-tables回车。如果没有出现提示信息&#xff0c;那就对了。如果指令没有阻塞输入或者 tasklist 无 mysqld 进程&#xff0c;这说明不成功&#xff0c;用 3.1 …...

西安有什么旅游景点/北京网站优化常识

需要用到离线缓存将数据信息存入数据库&#xff0c;在没有网络的时候进行加载&#xff0c;而iOS用的就是sqlite3数据库&#xff0c;用原生的sql我们也能实现&#xff0c;但是书写起来比较麻烦&#xff0c;尤其是其它语言转过来的程序员会感觉吃力&#xff0c;我们一般使用第三方…...

中文域名转换英文域名/详细描述如何进行搜索引擎的优化

1.麝香中的质量控制成分是 A.麝香酮 B.降麝香酮 C.麝香吡啶 D.羟基麝香吡啶 E.胆甾醇 2.胆汁酸结构中决定其化学性质的特征官能团是 A.酮基 B.脂基 C.醛基 D.羧基 E.炔基 3.蟾酥中蟾毒配基含量占药材总量的 A.5&#xff05;&#xff5e;20&#xff05; B.15&#xff05;&a…...

网站建设招聘信息/北京百度推广代运营

转至&#xff1a;http://sap.iteye.com/blog/121584今天突然想到的&#xff0c;肯定很多人知道&#xff0c;但是也肯定有一大堆人不知道。 转换公式 (n1)/2 比如DEC定义为13位&#xff0c;其中3位小数 则ABAP的对应P型应该定义为 (131)/2 7 data: p(7) type p decimals 3. 具体…...

北京建设工程网站/网络营销策划师

目前DEDECMS的优化做的已经很好了&#xff0c;但是在URL的长度控制上&#xff0c;个人感觉还是有欠缺的。这里专门来说一下对于织梦CMS缩短URL长度的方法。一&#xff1a;去掉a目录和html目录大家都知道&#xff0c;DEDECMS5.5版已经把根目录html给改成了a目录了&#xff0c;本…...