一天学完Servlet!!!(万字总结)

文章目录

    • 前言
    • Servlet打印Hello Servlet
    • Servlet生命周期
  • HttpServletRequest对象
    • 常用api方法
    • 请求乱码问题
    • 请求转发
    • request域对象
  • HttpServletResponse对象
    • 响应数据
    • 响应乱码问题
    • 请求重定向
    • 请求转发与重定向区别
  • Cookie对象
    • Cookie的创建与获取
    • Cookie设置到期时间
    • Cookie注意点
    • Cookie的路径
  • HttpSession对象
    • Session对象获取和常用Api
    • 标识符JSESSIONID
    • session域对象
    • session对象的销毁
  • ServletContext对象
    • 常用aip
    • ServletContext域对象
  • Servlet三大域对象总结
  • Servlet文件上传
    • 前端页面实现
    • 后台代码实现
  • Servlet文件下载
    • 超链接下载
    • 后台代码下载

前言

虽然Servlet现在已经被淘汰,但是作为SpringMVC的前置知识,还是需要全面了解一下的,故做此文,用于总结。

Servlet打印Hello Servlet

1、前置配置

下载Tomcat :
Tomcat下载连接

idea2022 创建Servlet项目流程:

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

public class Servlet01 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//打印内容在控制台System.out.println("hello servlet");//通过流输出到浏览器resp.getWriter().write("Hello Servlet!!!");}
}

运行效果:
浏览器:
在这里插入图片描述
控制台:
在这里插入图片描述

Servlet生命周期

生命周期简单来说,分为3步,
初始化(init()) ==> 调用方法(service()) ==>销毁方法(destroy())
放到代码中来看:

@WebServlet("/ser02")
public class servlet02 extends HttpServlet {private SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");/*** 系统方法,服务器自动调用,只执行一次* @throws ServletException*/@Overridepublic void init() throws ServletException {System.out.println("Servlet被创建了..." + formatter.format(new Date(System.currentTimeMillis())));}/*** 每次调用都会执行* @param req* @param resp* @throws ServletException* @throws IOException*/@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {System.out.println("Servlet被调用了..." + formatter.format(new Date(System.currentTimeMillis())));}/*** 系统方法,服务器自动调用,只执行一次*/@Overridepublic void destroy() {System.out.println("Servlet被销毁了..." + formatter.format(new Date(System.currentTimeMillis())));}
}

运行结果:
在这里插入图片描述

从图中可以看出,不管浏览器访问调用多少次,初始化和销毁方法只会执行一次,但是Service在每次访问都会执行。

HttpServletRequest对象

简单来讲,HttpServletRequest就是封装用户传过来的请求信息,对象由Tomcat封装好传过来,同时该对象是一个接口。service() 方法中传递的HttpServletRequest对象是该接口的实例化对象。

常用api方法

@WebServlet("/ser01")
public class Servlet01 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {// 1. 完整的请求urlString url = req.getRequestURL().toString();System.out.println("获取客户端请求的完整url: " + url);// 2. 部分url(站点名开始,到?前面结束)String uri = req.getRequestURI();System.out.println("客户端请求的部分url: " + uri);// 3. 获取请求行中的参数部分String queryString = req.getQueryString();System.out.println("获取请求行中的参数部分: " + queryString);// 4. 获取客户端请求方式String method = req.getMethod();System.out.println("获取客户端的请求方式: " + method);// 5. 获取http版本号String protocol = req.getProtocol();System.out.println("http版本号为: " + protocol);// 6. 获取webapp名字String contextPath = req.getContextPath();System.out.println("webapp名字为: " + contextPath);// 7. 获取请求参数// (1)、获取指定名称参数String username = req.getParameter("username");String pwd = req.getParameter("pwd");System.out.println("username: " + username + " pwd: " + pwd);// (2)、获取指定名称参数的所有值String[] hobbys = req.getParameterValues("hobby");System.out.println("hobbys: " + Arrays.toString(hobbys));if(hobbys != null && hobbys.length > 0) {for (String hobby : hobbys) {System.out.println("爱好: " + hobby);}}}
}

请求url:

http://localhost:8080/s01/ser01?username=suxuchao&pwd=123456&hobby=sing&hobby=dance&hobby=computer

运行结果:

获取客户端请求的完整url: http://localhost:8080/s01/ser01
客户端请求的部分url: /s01/ser01
获取请求行中的参数部分: username=suxuchao&pwd=123456&hobby=sing&hobby=dance&hobby=computer
获取客户端的请求方式: GET
http版本号为: HTTP/1.1
webapp名字为: /s01
username: suxuchao pwd: 123456
hobbys: [sing, dance, computer]
爱好: sing
爱好: dance
爱好: computer

请求乱码问题

在Tomcat8以上,GET请求有中文的话不会出现乱码,但是使用POST请求就会出现乱码,如下:

代码:

@WebServlet("/ser03")
public class Servlet03 extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {// 获取客户端传递的参数String username = req.getParameter("username");String pwd = req.getParameter("pwd");String method = req.getMethod();System.out.println("请求方法为: " + method);System.out.println("username: " + username);System.out.println("pwd: " + pwd);}
}

浏览器输入:
在这里插入图片描述

运行结果:

请求方法为: POST
username: å¼ ä¸‰
pwd: å¼ ä¸‰

解决方法:
首先我们需要知道,乱码是因为在解析过程中,request默认使用的编码是ISO-8859-1 (此编码不支持中文),所以解析一定会有乱码

		//设置编码req.setCharacterEncoding("UTF-8");

将该代码放在方法一开始,即可解决中文乱码问题

更正后结果:

请求方法为: POST
username: 张三
pwd: 张三

请求转发

请求转发有几个特点

  1. 服务端行为
  2. 地址栏的url不发生变化
  3. 从始至终只有一个请求
  4. request数据可以共享
  5. 代码中getRequestDispatcher只能转发一次

请求url:

http://localhost:8080/s01/ser03?username=张三

Servlet3代码:

		String username = req.getParameter("username");System.out.println("Servlet03: " + username);//请求转发跳转到Servlet04req.getRequestDispatcher("ser04").forward(req, resp);

Servlet4代码:

		// 接收客户端请求的参数String username = req.getParameter("username");System.out.println("Servlet04: " + username);

运行结果:

Servlet03: 张三
Servlet04: 张三

同时,重定向可以重定向.html/.jsp文件等页面:

req.getRequestDispatcher("login.jsp").forward(req, resp);

request域对象

request域对象中的数据在一次请求中有效,经过请求转发,request域中的数据仍然存在,在请求转发过程中可以通过request来传输/共享数据。

Servlet05代码:

		System.out.println("Servlet05...");//设置域对象内容req.setAttribute("name", "admin");req.setAttribute("age", 18);List<String> list = new ArrayList<>();list.add("aaa");list.add("bbb");req.setAttribute("list", list);req.getRequestDispatcher("ser06").forward(req, resp);

Servlet06代码:

 		System.out.println("Servlet06...");//获取域对象内容String name = (String) req.getAttribute("name");Integer age = (Integer) req.getAttribute("age");List<String> list = (List<String>) req.getAttribute("list");System.out.println("name: " + name);System.out.println("age: " + age);System.out.println("list: " + Arrays.toString(list.toArray()));

请求url:

http://localhost:8080/s01/ser05

访问结果:

Servlet05...
Servlet06...
name: admin
age: 18
list: [aaa, bbb]

HttpServletResponse对象

简单来讲,HttpServletResponse就是封装服务器处理后,要响应给客户端的数据,封装的数据有 (发送数据,发送响应头,发送响应状态码) 的方法。service() 方法中传递的HttpServletResponse对象是该接口的实例化对象

响应数据

使用字符流:

	// 获取字符输出流PrintWriter writer = resp.getWriter();// 输出数据writer.write("Hello");

使用字节流:

	//字节输出流ServletOutputStream outputStream = resp.getOutputStream();outputStream.write("Hi".getBytes());

注意:
两个流只能选择一个来用,否则就会报错。

响应乱码问题

s02代码:

	PrintWriter writer = resp.getWriter();writer.write("<h2>你好</h2>");

s03代码:

	ServletOutputStream os = resp.getOutputStream();os.write("<h2>你好</h2>".getBytes("UTF-8"));

运行结果:

<h2>??</h2>
<h2>浣犲ソ</h2>

解决方式:
在代码最开头添加一行这个代码,类比request的解决方法

resp.setCharacterEncoding("UTF-8");

但是我们发现此时两种输出流打印的结果最后都变成了相同的乱码,如下:

<h2>浣犲ソ</h2>

这是因为我们只设置了服务端的编码,request请求数据只在服务端之间传递,因此这样设置可以解决问题,但是response格式的话,我们不但要设置服务端,还有设置客户端的编码,解决方法如下:

	// 设置服务端的编码resp.setCharacterEncoding("UTF-8");// 设置客户端的编码resp.setHeader("content-type", "text/html;charset=UTF-8");

或者更简单的解决方法

	//同时设置编码resp.setContentType("text/html;charset=UTF-8");

这样就可以解决乱码和无法识别html的问题了。

请求重定向

  1. 重定向是客户端行为,由服务端指导
  2. 地址栏url会发生改变
  3. 存在两次请求
  4. 两次请求的数据不共享

Servlet04代码:

	System.out.println("Servlet04....");resp.setContentType("text/html;charset=UTF-8");String username = req.getParameter("username");System.out.println("username: " + username);resp.sendRedirect("s05");

Servlet05代码:

	System.out.println("Servlet05....");resp.setContentType("text/html;charset=UTF-8");String username = req.getParameter("username");System.out.println("username: " + username);

第一次请求url:

http://localhost:8080/s01/s04?username=zhangsan

重定向后的url:

http://localhost:8080/s01/s05

打印结果:

Servlet04....
username: zhangsan
Servlet05....
username: null

请求转发与重定向区别

在这里插入图片描述

Cookie对象

Cookie就是通过浏览器的一些程序,将只在客户端进行操作的数据,存在cookie中,以此来减轻服务器的压力,例如:记住密码的操作。
cookie就是简单的键值对形式存储,key和value之间用"="连接,不同kv之间用“;” 来分割。

Cookie的创建与获取

创建Cookie对象

	// Cookie的创建Cookie cookie = new Cookie("username", "zhangsan");// 发送(响应)Cookie对象resp.addCookie(cookie);

在这里插入图片描述

获取Cookie对象:

	// 获取Cookie数组Cookie[] cookies = req.getCookies();// 判断Cookie是否为空if(cookies != null && cookies.length > 0) {for (Cookie cookie : cookies) {String name = cookie.getName();String value = cookie.getValue();System.out.println("name: " + name);System.out.println("value: " + value);}}

运行结果:

name: username
value: zhangsan

Cookie设置到期时间

如下:

	// 到期时间为负数(浏览器关闭就失效)Cookie cookie1 = new Cookie("username01", "zhangsan");cookie1.setMaxAge(-1);res.addCookie(cookie1);// 到期时间为整数(cookie存在指定时间)Cookie cookie2 = new Cookie("username02", "lisi");cookie2.setMaxAge(30);res.addCookie(cookie2);// 到期时间为0(删除cookie)Cookie cookie3 = new Cookie("username03", "wangwu1");cookie3.setMaxAge(0);res.addCookie(cookie3);

Cookie注意点

  1. Cookie只在当前浏览器有用,Cookie无法跨浏览器,且只保存在当前电脑

  2. Cookie存中文问题

        // 使用URLEncoder进行编码String name = "姓名";String value = "张三";name = URLEncoder.encode(name);value = URLEncoder.encode(value);// 创建Cookie对象Cookie cookie1 = new Cookie(name, value);resp.addCookie(cookie1);//获取CookieCookie[] cookies = req.getCookies();if(cookies != null && cookies.length > 0) {for (Cookie cookie : cookies) {String dname = URLDecoder.decode(cookie.getName());String dvalue = URLDecoder.decode(cookie.getValue());System.out.println("dname: " + dname);System.out.println("dvalue: " + dvalue);}}

存储效果:在这里插入图片描述

  1. 同名Cookie问题

如果服务端发送重复的Cookie,那么会覆盖原有的Cookie.

  1. 浏览器存放Cookie的数量

浏览器对于Cookie的存放数量也是有上限的,Cookie存放在浏览器,而且一般由服务端创建和设定,后期结合Session来进行会话追踪。

Cookie的路径

简单总结,如果访问路径中带上了Cookie设置的路径,就可以访问到Cookie,否则就访问不到。

http://localhost:8080/s01/cookie01

对于上述url,
如果Cookie设置如下就可以访问到

cookie.setPath("/");
cookie.setPath("/s01");
cookie.setPath("/s01/cookie01");

如果设置如下就访问不到

cookie.setPath("/s02");
cookie.setPath("/s01/cookie02");

HttpSession对象

  1. Session 数据本身只存储在服务端
  2. 客户端仅持有用于查找 Session 的 ID(相当于钥匙)
  3. 若客户端禁用 Cookie,需通过 URL 参数传递 Session ID
  4. 每一个Session只在当前浏览器保存,开新浏览器就无法获得当前Session

Session对象获取和常用Api

	// 获取Session对象HttpSession session = req.getSession();// 获取Session的会话标识符String id = session.getId();System.out.println(id);// 获取Session的创建时间System.out.println(session.getCreationTime());// 获取最后一次访问时间System.out.println(session.getLastAccessedTime());// 判断是否是新的Session对象System.out.println(session.isNew());

运行结果:

A07655B90CBD6033B4D2F4FCC3C5FBE3
1745488955622
1745488955622
true

标识符JSESSIONID

服务器每次会先检查从客户端传过来数据中有没有JSESSIONID标识符,该标识符存放于cookie,服务器收到JSESSIONID之后会先查看是否含有该值的Session对象,如果没有,则认为这是一次新对话,创建一个新Session对象,如果由,就认为是标志过的会话,返回该Session对象,实现数据共享。在这里插入图片描述

session域对象

整体与request的域对象的api是一样的,但是与request域对象的区别是,Session域对象只要你这次会话没有结束,即浏览器没有和服务器断开,无论开多少个页面都能访问到Session中的数据,request则是只要换了一次请求,就无法共享域对象数据。

        // 获取Session域对象HttpSession session = req.getSession();// 设置域对象session.setAttribute("username", "zhangsan");session.setAttribute("pwd", "123456");//移除域对象session.removeAttribute("pwd");//获取域对象String username = (String) session.getAttribute("username");

session对象的销毁

  1. 默认销毁时间
    Tomcat/config/web.xml,在该文件下,默认session在不操作的情况下,存活时间为30min。
    <session-config><session-timeout>30</session-timeout></session-config>
  1. 自己设定销毁时间
	HttpSession session = req.getSession();session.setMaxInactiveInterval(15); // 15秒过期
  1. 立即销毁
        session.invalidate();// 立即销毁
  1. 关闭浏览器失效

Session 底层依赖Cookie来实现,而Cookie存在浏览器内存中,因此关闭浏览器导致Cookie失效,也会导致Session失效。

ServletContext对象

  1. 每个web程序,有且仅有一个Servlet Context对象,又称Application对象,在WEB程序启动后,一般会创建一个对应的ServletContext对象。
  2. 对象中保存了当前服务器的信息还有getRealPath获取资源真实路径等。

常用aip

代码:

// 获取servletContext对象
// (1)、通过request对象获取
ServletContext servletContext1 = req.getServletContext();
// (2)、通过Session对象获取
ServletContext servletContext2 = req.getSession().getServletContext();
// (3)、通过ServletConfig对象获取
ServletConfig servletConfig = getServletConfig();
ServletContext servletContext3 = servletConfig.getServletContext();
// (4)、直接获取
ServletContext servletContext4 = getServletContext();//常用方法
//1. 获取当前服务器的版本信息
String serverInfo = req.getServletContext().getServerInfo();
System.out.println("获取当前服务器的版本信息: " + serverInfo);
//2. 获取当前项目真实路径
String realPath = req.getServletContext().getRealPath("/");
System.out.println(" 获取当前服务器的真实路径: " + realPath);

运行结果:

获取当前服务器的版本信息: Apache Tomcat/8.5.47
获取当前服务器的真实路径: D:\code_study\servlet01\target\servlet01-1.0-SNAPSHOT\

ServletContext域对象

该域对象数据不建议存过多数据,因为如果不手动移除,服务器启动后会一直存在。

ServletContext servletContext = req.getServletContext();
// 设置域对象
servletContext.setAttribute("name", "zhangsan");
// 获取域对象
String name = (String) servletContext.getAttribute("name");
// 移除域对象
System.out.println("name = " + name);
servletContext.removeAttribute("name");

Servlet三大域对象总结

  1. request域对象
    在一次request请求中生效。
  2. Session域对象
    在一次会话中生效。
  3. ServletContext域对象
    在服务器启动期间一直生效,除非重启服务器。

Servlet文件上传

前端页面实现

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>文件上传</title>
</head>
<body>
<!--文件上传1.准备表单2.设置表单的提交类型3.设置表单类型问文件上传表单 enctype="multipart/form-data"4.设置文件提交的地址5.准备表单元素1. 普通的表单项 type="text"2. 文件项 type="file"
-->
<form method="post" enctype="multipart/form-data" action="uploadServlet">姓名: <input type="text" name="username"> <br>文件: <input type="file" name="myfile"> <br><button>提交</button>
</form>
</body>
</html>

后台代码实现

@WebServlet("/uploadServlet")
@MultipartConfig //文件上传表单一定要加这个注释
public class uploadServlet extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {System.out.println("文件上传....");// 设置请求编码格式req.setCharacterEncoding("UTF-8");// 获取普通表单项String username = req.getParameter("username");System.out.println("username: " + username);// 获取Part对象(Serlet将mutipart/form-data的POST请求封装成Part对象)Part part = req.getPart("myfile");// 通过part对象得到上传的文件名String fileName = part.getSubmittedFileName();System.out.println("上传文件名: " + fileName);String realPath = req.getServletContext().getRealPath("/");System.out.println("文件存放路径: " + realPath);// 上传文件到指定目录part.write(realPath + "/" + fileName);}
}

最后文件上传的路径可以改成你想要指定的。
注意:上传文件一定要加@MultipartConfig 注解,否则文件无法上传成功

Servlet文件下载

就是将浏览器上的文件下载到本地

超链接下载

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>文件下载</title>
</head>
<body><!--浏览器可识别的文件可以直接浏览器中查看,不可识别的文件需要先浏览器下载。--><!--浏览器可以识别的资源--><a href="download/hello.txt">文本文件</a><a href="download/ai.jpg">图片文件</a><!--浏览器不可以识别的资源--><a href="download/yasuo.zip">压缩文件</a><hr><!--添加了download属性,可以直接在浏览器中下载对应文件--><a href="download/hello.txt" download>文本文件</a><a href="download/ai.jpg" download="ai.jpg">图片文件</a>
</body>
</html>

后台代码下载

@WebServlet("/downloadServlet")
public class DownloadServlet extends HttpServlet {@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {System.out.println("文件下载开始....");// 1. 设置请求的编码req.setCharacterEncoding("UTF-8");// 2. 获取参数String fileName = req.getParameter("fileName");// 3. 参数的非空判断if(fileName == null || "".equals(fileName.trim())) {resp.setContentType("text/html;charset=utf-8");resp.getWriter().write("请输入要下载的文件名");resp.getWriter().close();return;}// 得到图片存放的路径String path = req.getServletContext().getRealPath("/download/");// 通过路径得到file对象File file = new File(path + fileName);// 判断文件对象是否存在if(file.exists() && file.isFile()) {// 设置相应类型(浏览器无法处理或激活某个程序来处理的MIME类型)resp.setContentType("application/x-msdownload");// 设置响应头resp.setHeader("content-Disposition", "attachment;filename" + fileName);// 得到file文件输入流InputStream is = new FileInputStream(file);// 得到字节输出流ServletOutputStream os = resp.getOutputStream();// 定义字节数组byte[] bytes = new byte[1024];//定义长度int len = 0;// 循环输出while((len = is.read(bytes)) != -1) {// 输出os.write(bytes, 0, len);}// 关闭资源os.close();is.close();} else {resp.getWriter().write("文件不存在,请重试!");resp.getWriter().close();}}
}

至此,Servlet基本流程就学习结束。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/78074.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Springboot整合 xxljob,自定义添加、修改、删除、停止、启动任务

目录 一、模拟登录方式 二、注解方式 三、访问者调用 四、测试 本次自定义方式分为两种&#xff1a;一种是模拟登录&#xff0c;另一种是使用注解的方式 一、模拟登录方式 修改xxl-job-admin工程&#xff0c;在controller里面添加一个MyApiController&#xff0c;在里面添…

STM32F407使用ESP8266实现阿里云OTA(中)

文章目录 前言一、程序分析二、程序讲解1. main函数2. Get_Version()函数3. esp_Init()函数4. Check_Updata()函数结语前言 从上一章STM32F407使用ESP8266实现阿里云OTA(上)中我们已经对连接阿里云和从阿里云获取升级包的流程非常的熟悉了。所以本章我们进行STM32的程序开发…

Docker部署DeepSeek常见问题及解决方案

在使用Docker部署DeepSeek的过程中,许多开发者可能会遇到一些常见问题。本文整理了几个高频问题及其解决方案,帮助大家更顺利地完成部署。 镜像拉取失败 问题现象 执行 docker pull 命令时,提示超时或镜像不存在。 可能原因 1. 网络环境不稳定,导致连接Docker Hub失败…

Linux 内核 IPv4 套接字创建机制与协议表管理深度解析

一、inet_create:IPv4 套接字创建的核心引擎 1.1 核心功能与执行流程 inet_create 是 Linux 内核处理 socket(AF_INET, type, protocol) 系统调用的核心实现,主要完成以下关键任务: 协议匹配与初始化:根据套接字类型和协议号匹配协议处理模块 资源分配:创建并初始化套接…

网络:手写HTTP

目录 一、HTTP是应用层协议 二、HTTP服务器 三、HTTP服务 认识请求中的uri HTTP支持默认首页 响应 功能完善 套接字复用 一、HTTP是应用层协议 HTTP下层是TCP协议&#xff0c;站在TCP的角度看&#xff0c;要提供的服务是HTTP服务。 这是在原来实现网络版计算器时&am…

论文笔记(七十八)Do generative video models understand physical principles?

Do generative video models understand physical principles? 文章概括Physics-IQ基准数据集评估协议为什么要创建一个真实世界的Physics-IQ数据集模型物理理解的评估指标动作发生在哪里&#xff1f;空间IoU&#xff08;Spatial IoU&#xff09;动作在哪里、何时发生&#xf…

AXP2101入门

目录 核心功能与特性封装与配置安全与可靠性 AXP2101 是一款由全志公司开发的单电池 NVDC 电源管理集成电路&#xff08;PMIC&#xff09;&#xff0c;专为锂离子/锂聚合物单电池应用设计&#xff0c;适用于需要多通道电源输出的设备。 核心功能与特性 1.输入与充电管理 输入…

DAY8:Oracle高可用架构深度解析与Data Guard单节点搭建实战

引言 在数据库领域&#xff0c;高可用性&#xff08;High Availability&#xff09;是保障业务连续性的核心要求。Oracle作为企业级数据库的领导者&#xff0c;提供了RAC、Data Guard、GoldenGate三大核心方案。本文将深入剖析这些技术的实现原理&#xff0c;并手把手指导搭建…

游戏引擎学习第243天:异步纹理下载

仓库 https://gitee.com/mrxiao_com/2d_game_6 https://gitee.com/mrxiao_com/2d_game_5 回顾并为今天设定阶段 目前的开发工作主要回到了图形渲染相关的部分。我们之前写了自己的软件渲染器&#xff0c;这个渲染器性能意外地好&#xff0c;甚至可以以相对不错的帧率运行过场…

BBRv2,v3 吞吐为什么不如 BBRv1

为什么 BBRv2/3 测试下来吞吐远不如 2016 年底的 BBRv1&#xff0c;这个事曾经提到过很多次&#xff0c;今天分析一下原理。注意三个事实&#xff1a; BBR 是一种拥塞控制算法&#xff1b;BBR 已经迭代到了 v3 版本&#xff1b;BBRv3 的 “性能” 远不如 BBRv1. 第二点有点不…

前端项目搭建集锦:vite、vue、react、antd、vant、ts、sass、eslint、prettier、浏览器扩展,开箱即用,附带项目搭建教程

前端项目搭建集锦&#xff1a;vite、vue、react、antd、vant、ts、sass、eslint、prettier、浏览器扩展&#xff0c;开箱即用&#xff0c;附带项目搭建教程 前言&#xff1a;一、Vue项目下载快速通道二、React项目下载快速通道三、BrowserPlugins项目下载快速通道四、项目搭建教…

蓝桥杯 15.小数第n位

小数第n位 原题目链接 题目描述 我们知道&#xff0c;整数做除法时&#xff0c;有时会得到有限小数&#xff0c;有时会得到无限循环小数。 如果我们把有限小数的末尾加上无限多个 0&#xff0c;它们就具有了统一的形式。 本题的任务是&#xff1a;在上述约定下&#xff0c…

【Docker】在Ubuntu平台上的安装部署

写在前面 docker作为一种部署项目的辅助工具&#xff0c;真是太好用了需要魔法&#xff0c;不然无法正常运行笔者环境&#xff1a;ubuntu22.04 具体步骤 更新系统包索引 sudo apt update安装必要依赖包 sudo apt install -y apt-transport-https ca-certificates curl softwa…

Spring Boot默认缓存管理

Spring框架支持透明地向应用程序添加缓存&#xff0c;以及对缓存进行管理&#xff0c;其管理缓存的核心是将缓存应用于操作数据的方法&#xff0c;从而减少操作数据的执行次数&#xff0c;同时不会对程序本身造成任何干扰。Spring Boot继承了Spring框架的缓存管理功能&#xff…

数模学习:一,层次分析法

基本定位&#xff1a; 适用于解决评价&#xff0c;选择类问题&#xff08;数值不确定&#xff0c;需要自己结合资料数据等自己填写&#xff09;。 引入&#xff1a; 若要解决选择类的问题&#xff0c;打分的方式最为常用——即采用权重表&#xff1a; 指标权重选择1选择2..…

模板偏特化 (Partial Specialization)

C 模板偏特化 (Partial Specialization) 模板偏特化允许为模板的部分参数或特定类型模式提供定制实现&#xff0c;是 静态多态&#xff08;Static Polymorphism&#xff09; 的核心机制之一。以下通过代码示例和底层原理&#xff0c;全面解析模板偏特化的实现规则、匹配优先级…

sql 根据时间范围获取每日,每月,年月的模版数据

1&#xff1a;获取每日模版数据&#xff08;参数也支持跨年&#xff09; SELECT a.selected_date cdate FROM(SELECT adddate(1970-01-01,t4.i * 10000 t3.i * 1000 t2.i * 100 t1.i * 10 t0.i) selected_dateFROM( SELECT 0 i UNION SELECT 1 UNION SELECT 2 UNION SELEC…

windows上的RagFlow+ollama知识库本地部署

一、 docker的安装与部署 1. 下载Docker Desktop 访问Docker官网并下载适用于Windows的Docker Desktop安装程序。 RagFlow对docker的要求: Docker ≥ 24.0.0 & Docker Compose ≥ v2.26. docker 下载地址: https://www.docker.com/ Get Docker | Docker Docs 如下图所…

多模态大语言模型arxiv论文略读(三十四)

SHIELD : An Evaluation Benchmark for Face Spoofing and Forgery Detection with Multimodal Large Language Models ➡️ 论文标题&#xff1a;SHIELD : An Evaluation Benchmark for Face Spoofing and Forgery Detection with Multimodal Large Language Models ➡️ 论文…

Unity InputSystem触摸屏问题

最近把Unity打包后的windows软件放到windows触摸屏一体机上测试&#xff0c;发现部分屏幕触摸点击不了按钮&#xff0c;测试了其他应用程序都正常。 这个一体机是这样的&#xff0c;一个电脑机箱&#xff0c;外接一个可以触摸的显示屏&#xff0c;然后UGUI的按钮就间歇性点不了…