【JavaEE初阶系列】——Servlet运行原理以及Servlet API详解

news2025/4/7 16:04:00

目录

🚩Servlet运行原理

🚩Servlet API 详解

🎓HttpServlet核心方法

🎓HttpServletRequest核心方法

🎈核心方法的使用

🔴获取请求中的参数

💻query string

💻直接通过form表单

💻json

🎓HttpServletResponse核心方法

🎈setStatus(设置状态码)

🎈sendError发送错误码同时提示信息

🎈setHeader刷新浏览器时间戳

🎈setHeader和sendRedirect构造重定向响应


🚩Servlet运行原理

我们上一篇完成了一篇用servlet来写hello world程序,我们会发现 Servlet 的代码中我们并没有写 main 方法, 那么对应的 doGet 代码是如何被调用的呢? 响应又是如何 返回给浏览器的

servlet是一组api,操作http协议的,tomcat提供的。一组api也可以理解为一个HTTP服务器框架的。 


我们给一个程序,比喻成一辆汽车,main方法就是发动机,是否存在一辆车,没有发动机也可以跑呢?——火车车厢,只有火车头上面有发动机,后面的车厢都是不需要的,车头拉着跑就行了。

我们servlet的代码中没有写main方法,但是也是执行的,是因为我们的程序不是直接执行的,而是在tomcat上运行的,tomcat这个程序里面有main方法,tomcat是火车头,写的webapp就是车厢。

继承HttpServlet这个类,重写里面的方法,目的就是为了把咱们自己定义的代码,插入到“tomcat”中。

在tomcat中,就有形如以下这样的代码:

  • 先通过tcp socket读取请求,http也是基于tcp的 String requestString=readRequest();
  • 然后解析这个请求 HttpServletRequest req=parse(requestString)
  • 构造空的响应对象HttpServletResponse resp=new HttpServletResponse();
  • 创建一个HttpServlet对象 HttpServlet s=new HelloServlet(); s.doGet(req,resp);
  • 把得到的响应返回给浏览器
  • 。。。。。。

tomcat能够自动的new出一个咱们写的子类。执行到s.doGet(req,resp),就相当于把上述代码嵌套到tomcat代码中了。

相当于在处理一次请求的过程中,tomcat完成了99%的工作,剩余的1%由咱们写的代码来完成,(99%的工作是通用的公共的工作,剩下的1%是结合具体的业务,具有差异化的工作)。为了解决不同的业务场景,使用tomcat写网站,不同网站,涉及的业务是不同的。

我们写网站的时候,1.接收前端传来的http请求,并解析。2.根据http请求计算出一个http响应。3.把http响应返回到浏览器这边。  


🚩Servlet API 详解

🎓HttpServlet核心方法

我们写 Servlet 代码的时候, 首先第一步就是先创建类, 继承自 HttpServlet, 并重写其中的某些方法。

  • 方法名称:init     调用时机:在HttpServlet实例化之后被调用一次   (完成初始化操作)
  • 方法名称:destroy     调用时机:在HttpServlet实例不再使用的时候调用一次(用来释放资源)
  • 方法名称:service     每次收到http请求,就会自动执行的方法,处理请求,计算响应(服务器的主逻辑)
  • 方法名称:doGet       收到GET请求的时候调用(由service方法调用)
  • 方法名称:doPost      收到POST请求的时候调用(由service方法调用)
  • 方法名称:doPut/doDelete/doOptions/.....  收到其他请求的时候调用(由service方法调用)

这几个方法,都是可以在子类中重写的。在子类重写这些方法之后,这些方法也都是不需要咱们手动调用的,都是tomcat在合适的时机自行调用。

如果不重写service,在父类(HttpServlet中),自己的service,就会根据请求的方法,来分别调用下面的doGet,doPost,doPut....

  • init
  • destroy
  • service   这三个方法是根据servlet的生命周期来进行的。

所谓生命周期 就是 什么时间点做什么事情。

我们在幼年/青年阶段我们需要学习,成年的时候我们需要工作,老年的时候我们需要养老和退休休息(当我们在执行servlet的时候,我们会根据特定的时间使用上述方法,如果在初始化的时候用init,在释放资源的时候我们用destroy,在收到http请求的时候我们用service方法。)

一个servlet不用了,说明tomcat要关闭了。

tomcat关闭有两种方式:

  • 1.直接干掉tomcat进程(比如直接在任务管理器中结束进程,或者直接点x)这时候我们完全用不到destroy方法
  • 2.通过8005管理端口,给tomcat发送一个“停机”指令,这个时候是能够执行destroy的。 

以往写的代码,都是自己定义方法,自己来调用,或者是直接调用别人定义好的方法。

从今往后,经常会涉及到咱们自己写好的代码,让别人帮咱们调用。所谓“框架”就是如此~~

一个程序的主体部分,都已经被其他大佬们写完了,有些细节内容,允许咱们插入咱们自己写的自定义的逻辑。

一台电脑整体的框架:cpu,主板,内存,硬盘,散热器,电源,机箱,显卡。大的框架确定,还可以有一些自定义的空间。

为什么写代码的时候,要使用框架呢?使用框架,就是不让我们随便写代码。就是限制程序员的自由,不能让程序员代码胡写。“设计模式”属于一种“软性的限制”,“使用框架”属于一种“硬性的限制”,所以框架就出现了。

使用postman,发起一个post请求,让我们看看效果。理解前端和后端之间的交互的对应关系,是一个很重要的事情,也是进入web开发的入门难点。


🎓HttpServletRequest核心方法

Tomcat 通过 Socket API 读取 HTTP 请求 ( 字符串 ), 并且按照 HTTP 协议的格式把字符串解析成
HttpServletRequest 对象 .

表示了一个HTTP请求。

类似于,第一行我们需要得到版本号,第二行我们需要知道uri......

方法 描述
  • String getProtocol() 返回请求协议的名称和版本。
  • String getMethod() 返回请求的 HTTP 方法的名称,例如,GETPOST PUT
  • String getRequestURI() 从协议名称直到 HTTP 请求的第一行的查询字符串中,返回该请求的 URL 的一部分。
  • String getContextPath() 返回指示请求上下文的请求 URI 部分。
  • String getQueryString() 返回包含在路径后的请求 URL 中的查询字符串。

  • Enumeration getParameterNames() 返回一个 String 对象的枚举,包含在该请求中包含的参数的名称。
  • String getParameter(String name) 以字符串形式返回请求参数的值,或者如果参数不存在则返回 null。
  • String[] getParameterValues(String name) 返回一个字符串对象的数组,包含所有给定的请求参数的值,如果参数不存在则返回 null。key存在重复的请求一个key有多个值?a=10&a=20;

query String本质上是键值对,一般都是根据key获取value,很少把queryString作为整体来使用。

  • Enumeration getHeaderNames() 返回一个枚举,包含在该请求中包含的所有的头名。
  • String getHeader(String name) 以字符串形式返回指定的请求头的值。

上述两个方法http请求header部分,也是键值对,也需要根据key获取到value

  • String getCharacterEncoding() 返回请求主体中使用的字符编码的名称。
  • String getContentType() 返回请求主体的 MIME 类型,如果不知道类型则返回 null
  • int getContentLength() 以字节为单位返回请求主体的长度,并提供输入流,或者如果长 度未知则返回 -1

上述三个方法,都是对getHeader进行了简单的封装。?a=10&b=20&c=30 query string更多的是希望能够知道a是几,b是几,c是几.

  • InputStream getInputStream() 用于读取请求的 body 内容. 返回一个 InputStream 对象

通过这个方法,得到一个流对象,读取这个对象,救恩那个得到整个请求的body。(很多时候,在代码中表示一个“不定长”的数据的时候,经常就会想到使用“流对象”)使用流对象,就允许咱们不必一次性处理完,一次处理一点,也可以一次全都处理,更好的兼容二进制数据。

上述介绍的方法,都是get系列的方法(都是读方法),没有set系列(没有写方法)。当前拿到的HttpServletRequest,这里的数据都是来自于客户端发来的,这些数据的内容已经确定下来了,程序员是不需要更改的。

🎈核心方法的使用

调用api,把得到的结果构造成一个字符串,同意返回给客户端。
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;
import java.util.Enumeration;

@WebServlet("/show")
public class ShowRequestServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //调用api,把得到的结果构造成一个字符串,统一返回给客户端
        StringBuilder stringBuilder=new StringBuilder();
        stringBuilder.append(req.getProtocol());//得到版本号
        stringBuilder.append("<br>");
        stringBuilder.append(req.getMethod());//获取方法 get还是post还是delete
        stringBuilder.append("<br>");
        stringBuilder.append(req.getRequestURI());//地址
        stringBuilder.append("<br>");
        stringBuilder.append(req.getContextPath());//上下文路径
        stringBuilder.append("<br>");
        stringBuilder.append(req.getQueryString());//请求字符串
        stringBuilder.append("<br>");

        //获取所有的header
        Enumeration<String> headerName=req.getHeaderNames();//头部信息
        while (headerName.hasMoreElements()){
            String key=headerName.nextElement();
            String value=req.getHeader(key);
            stringBuilder.append(key+":"+"  "+value);
            stringBuilder.append("<br>");

        }
        //告诉浏览器,咱们的数据返回的是什么类型,任何一次服务器返回,都要做的事情
        resp.setContentType("text/html; charset=utf8");
        //把上述内容整体返回到客户端这边
        resp.getWriter().write(stringBuilder.toString());
    }
}

🔴获取请求中的参数
💻query string

在服务器这边获取到请求中的参数(Query String)

query string中的键值对,都是程序员自定义的,实际开发中都会非常广泛的使用到query string这样的机制。

请求就是相当于输入网址,后面写上 ?username=zyf&password=0528 那么就获取,然后打印出来,如果输入的地址(请求)没有username和passeord那么就是null
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;

//从query string中获取用户名和密码
@WebServlet("/getParameter")
public class QueryStringGetParameterServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //此处约定,请求中给定的query string 是形如:username=zyf&password=0528
        //上述query string就会被tomcat给自动解析成一个Map结构
        //getParameter就会在map中查找
        //请求就是相当于输入网址,后面写上 ?username=zyf&password=0528 那么就获取,然后打印出来,如果输入的地址(请求)没有username和passeord那么就是null
        String username=req.getParameter("username");
        String password=req.getParameter("password");
        //对两者内容,进行处理,这里就是简单的打印出来
        System.out.println("username="+username);
        System.out.println("password="+password);
        resp.getWriter().write("ok");
    }
}

如果请求中没有加上query string,此时得到的结果是null null


💻直接通过form表单

body的格式就是query string格式,Content-Type:application/x-www-form-urlencoded

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("/postParameter")
public class FormPostParameterServlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //约定 前端形如这样的请求:
        //POST: /postParameter
        //Content-Type: x-www-form-urlencoded

        //username=zyf&password=528
        //就需要在后端代码中,把body中的 值 给拿到,获取值的方法还是getParameter
        String username=req.getParameter("username");
        String password=req.getParameter("password");
        System.out.println("username="+username);
        System.out.println("password="+password);
        resp.getWriter().write("ok");
    }
}


💻json
        请求格式如下req(json字符串)——>得到Request对象
        Post /json
        Content-type: application/json
        {
           username:zhangsan",
           password:"123"


        }  此处约定响应的格式(也按照json来组织;
        {
          ok:true
        }

body的格式是json,Content-Type:application/json

json本质上也是键值对,规则和form表单截然不同,解析方式自然也变了,由于json支持“嵌套”,因此自己手写解析json的代码,并不容易。


用json处理请求中的参数,有些麻烦。

  • 1.下载导入jackson到项目中,通过maven

  • 2.使用jackson,一个类两个方法

ObjectMapper

把json字符串,映射成一个java对象——readValue

把一个java对象,映射成json字符串——writeValueAsString

网络传输,使用json字符串,java代码中各种逻辑,java对象。

站在服务器的角度,收到的请求,就是json字符串,就需要把json字符串,先映射到java对象中,再进行一系列的业务逻辑处理。

处理完了之后,可能还需要把得到的java对象,映射回json字符串,并且通过响应来返回。


把json字符串,映射成一个java对象——readValue

jackson从json字符串映射到java对象的过程:

  • 1.参数就是json字符串(json字符串是在http的body中的,就需要通过HttpServletRequest中的getInputStream来获取到)此处把这个流对象,直接传给readValue,readValue内部就会读取到InputStram中的所有数据(http请求的body,上面的json字符串)
  • 2.按照json的格式,进行解析,把json字符串,解析成Map(键值对)
  • 3.把Map转换成java对象,在方法的第二个参数,传入了Request.class告诉readValue方法内部,要把当前Map转成啥样的java对象。readValue内部,就可以通过反射api,创建出Request.class实例,并且,根据Request.class提供的属性的名字,来查询上述Map,把得到的结果,赋值给对应的属性。

举个例子:当前Request.class中就知道了,Request两个属性,分别叫做username和password。

拿到username去刚才的Map里查询~~查到的结果叫做cl,于是把cl赋值给username

拿着password去刚才的Map里查询~查到的结果叫做1005,于是把1005赋值给password属性

最终就得到了一个完整的Request对象,username就是cl,password是1005


把一个java对象,映射成json字符串——writeValueAsString

这个代码就是readValue的反向操作能把java对象映射成json字符串。writeValueAsString

  • 1.通过传入的参数,获取到类对象,通过反射拿到哪些属性,就有一个属性 ok
  • 2.根据属性的名字拿到属性的值 拿到了值 true
  • 3.把上述 属性名字 和 属性值 按照json格式构造成字符串就是返回值 形如"{ok:true}"

postman对于json格式要求比较严格,key必须加上引号(但是比如前端的ajax这里就可以不加引号)


🎓HttpServletResponse核心方法

类似于这种响应报文。

方法 描述
  • void setStatus(int sc) 为该响应设置状态码。
  • void setHeader(String name, String value) 设置一个带有给定的名称和值的 header. 如果 name 已经存在, 则覆盖旧的值.
  • void addHeader(String name, String value) 添加一个带有给定的名称和值的 header. 如果 name 已经存在, 不覆盖旧的值, 并列添加新的键值对
  • void setContentType(String type) 设置被发送到客户端的响应的内容类型。
  • void setCharacterEncoding(String charset) 设置被发送到客户端的响应的字符编码(MIME 字符集)例如, UTF-8。
  • void sendRedirect(String location) 使用指定的重定向位置 URL 发送临时重定向响应到客户端。
  • PrintWriter getWriter() 用于往 body 中写入文本格式数据.
  • OutputStream getOutputStream() 用于往 body 中写入二进制格式数据.

🎈setStatus(设置状态码)

@WebServlet("/status")
public class StatusServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //不显示设置。默认是200
       // resp.setStatus(404);
        resp.sendError(404,"这个页面是个错误的页面");
    }
}

🎈sendError发送错误码同时提示信息


🎈setHeader刷新浏览器时间戳


通过setHeader给响应中设置一些特殊的header,比如,可以设置refresh:1,让浏览器,每秒钟自动刷新一次。

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("/refresh")
public class RefreshServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //通过setHeader给响应设置一些特殊的header
        //refresh使浏览器每隔xs都刷新一次
        resp.setHeader("refresh","1");
        resp.getWriter().write(" "+System.currentTimeMillis());//时间戳(当前时间)
    }
}

虽然是每秒刷新一次,但是也不是精准的1000ms,略多一点,浏览器发请求,到服务器返回响应,也需要时间。


🎈setHeader和sendRedirect构造重定向响应

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("/Redirect")
public class RedirectServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //1.设置状态码302(重定向)
        //2.header里面设置一个属性location描述跳转到哪个页面中去
//        resp.setStatus(302);
      // resp.setHeader("Location","https://www.baidu.com");
        resp.sendRedirect("https://www.baidu.com");
    }
}
  • 重定向状态码是3xx(比如302)
  • header需要有一个Location属性,描述要跳转到哪里
resp.setHeader("Location","https://www.baidu.com");
  • 直接跳转到所写字符串ip地址
resp.sendRedirect("https://www.baidu.com");


知不足而奋进,望远山而前行。

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

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

相关文章

如何更好地使用Kafka? - 事先预防篇

要确保Kafka在使用过程中的稳定性&#xff0c;需要从kafka在业务中的使用周期进行依次保障。主要可以分为&#xff1a;事先预防&#xff08;通过规范的使用、开发&#xff0c;预防问题产生&#xff09;、运行时监控&#xff08;保障集群稳定&#xff0c;出问题能及时发现&#…

【自动驾驶|毫米波雷达】初识毫米波雷达射频前端硬件

第一次更新&#xff1a;2024/5/4 目录 整体概述 混频器&#xff08;MIXER&#xff09; 低通滤波器&#xff08;LPF&#xff1a;Low-Pass filter&#xff09; 数模转换器&#xff08;ADC&#xff1a;Analog to Digital Converter&#xff09; 毫米波雷达功能框图 整体概述 完…

分布式与一致性协议之ZAB协议(六)

ZAB协议 成员发现 成员发现是通过跟随者和领导者交互来完成的&#xff0c;目标是确保大多数节点对领导者的关系没有异议&#xff0c;也就是确立领导者的领导地位。成员发现的实现流程如图所示。 1.领导者选举结束&#xff0c;节点进入跟随者状态或者领导者状态后&#xff0…

微软 AI 研究团队推出 SIGMA:一个开源研究平台,旨在推动混合现实与人工智能交叉领域的研究与创新

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

YzmCMS 7.0任意函数调用RCE 漏洞研究分析

YzmCMS是一款基于YZMPHP开发的一套轻量级开源内容管理系统,YzmCMS简洁、安全、开源、免费,可运行在Linux、Windows、MacOSX、Solaris等各种平台上,专注为公司企业、个人站长快速建站提供解决方案。 YzmCMS 某些接口调用了 db_pdo类的where方法 导致了远程命令执行漏洞&#xf…

PLC学习笔记

PLC学习笔记 前言一、一些基操知识二、GX works2编程2.1 位逻辑1.2 中间寄存器1.3 PLC的扫描方式 总结 前言 我这个人真的是太渴望知识了~ 一、一些基操知识 一般X表示输入&#xff0c;Y表示输出。一般八个为一组X0~X7M表示中间寄存器&#xff0c;M0~M7时间T、计数C 二、GX …

操作系统:线程相关知识

目录 1.生产消费者模型 1.1.概念引入 1.2.基于阻塞队列的生产消费模型 1.3.POSIX信号量 1.3.1.再识信号量 1.3.2.信号量接口的学习 1.4.基于环行队列的生产消费模型 1.5.深刻理解生产消费模型 2.可重入函数与线程安全 1.生产消费者模型 1.1.概念引入 生产者-消费者模型…

AI换脸免费软件Rope中文汉化蓝宝石版本全新UI界面,修复部分已知错误【附下载地址与详细使用教程】

rope蓝宝石版&#xff1a;点击下载 注意&#xff1a;此版本支持N卡、A卡、CPU&#xff0c;且建议使用中高端显卡&#xff0c;系统要求win10及以上。 Rope-蓝宝石 更新内容&#xff1a; 0214版更新&#xff1a; ①&#xff08;已修复&#xff09;恢复到以前的模型荷载参数。有…

Hadoop3:HDFS的架构组成

一、官方文档 我这里学习的是Hadoop3.1.3版本&#xff0c;所以&#xff0c;查看的也是3.1.3版本的文档 Architecture模块最下面 二、HDFS架构介绍 HDFS架构的主要组成部分&#xff0c;是一下四个部分 1、NameNode(NN) 就是Master节点&#xff0c;它是集群管理者。 1、管…

Stack数据结构设计模板

第三章 栈、队列、数组 1.栈 1.1 顺序栈 #define MaxSize 20 typedef int ElemType; //顺序栈的定义 typedef struct {ElemType data[MaxSize];int top; }SqStack; // 初始化顺序栈 void InitSqStack(SqStack &S){S.top -1; }; // 入栈(增) bool Push(SqStack &S,El…

RabbitMQ之消费者并发消费

为什么要引入消费者的并发消费&#xff1f; 当生产者的推送速度是远远超过消费者的能力的&#xff0c;可以提高消费者的消费速度。比如在java中我们可以启动多个 JVM 进程&#xff0c;实现多进程的并发消费&#xff0c;从而加速消费的速度&#xff0c;在mq中也可以通过设置配置…

Git系列:Git Stash临时保存与恢复工作进度

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…

python从0开始学习(五)

目录 前言 1、顺序结构 2、选择结构 2.1双分支结构 2.2多分枝结构 2.3嵌套使用 2.4多个条件的链接 总结 前言 在上篇文章中&#xff0c;我们学习了python中的运算符&#xff0c;本篇文章继续往下讲解。本篇文章主要讲解程序的组织结构。 1、顺序结构 顺序结构是程序按照…

UDP通讯的demo

udp通讯的demo&#xff0c;这个只是简单的实现。 后面我还会加入udp组播功能。 因为懒&#xff0c;所以我自己发&#xff0c;自己接收了。 经过测试&#xff0c;可以看到&#xff0c;发送消息和接收消息功能都没问题。 广播&#xff1a; 这个是点对点的通过对方的ip和端口发…

设计模式学习笔记 - 回顾总结:在实际软件开发中常用的设计思想、原则和模式

概述 本章&#xff0c;先来回顾下整个专栏的知识体系&#xff0c;主要包括面向对象、设计原则、编码规范、重构技巧、设计模式五个部分。 面向对象 相对于面向过程、函数式编程&#xff0c;面向对象是现在最主流的编程范式。纯面向过程的编程方法&#xff0c;现在已经不多见了…

网络网络层之(4)IPv4协议

网络网络层之(1)IPv4协议 Author: Once Day Date: 2024年4月4日 一位热衷于Linux学习和开发的菜鸟&#xff0c;试图谱写一场冒险之旅&#xff0c;也许终点只是一场白日梦… 漫漫长路&#xff0c;有人对你微笑过嘛… 全系列文档可参考专栏&#xff1a;通信网络技术_Once-Day的…

设置 kafka offset 消费者位移

文章目录 1.重设kafka消费者位移2.示例2.1 通过 offset 位置2.2 通过时间2.3 设置到最早 1.重设kafka消费者位移 维度策略含义位移Earliest把位移调整到当前最早位移处位移Latest把位移调整到当前最新位移处位移Current把位移调整到当前最新提交位移处位移Specified-Offset把位…

奥威-金蝶BI现金流量表模板,可借鉴、可套用

企业现金流一旦出了问题都是大问题&#xff0c;会直接影响到企业的日常运作&#xff0c;甚至直接关系到企业能不能继续存活&#xff0c;因此现金流量表是企业财务分析中重要报表之一&#xff0c;也是企业监控财务监控情况的重要手段之一。那么这么重要的一份现金流量表该怎么做…

Angular中创建和使用服务

Angular中的服务 文章目录 Angular中的服务前言一、创建服务二、使用服务 前言 Angular 服务是 Angular 应用程序中用于封装可重用逻辑的类。服务在应用程序的多个组件之间共享数据和功能&#xff0c;而不依赖于应用程序的UI。服务可以用于诸如数据处理、与后端通信、用户身份…

Verilog中4位数值比较器电路

某4位数值比较器的功能表如下。 请用Verilog语言采用门级描述方式&#xff0c;实现此4位数值比较器 参考代码如下&#xff1a; &#xff08;CSDN代码块不支持Verilog&#xff0c;代码复制到notepad编辑器中&#xff0c;语言选择Verilog&#xff0c;看得更清楚&#xff09; t…