tomcat和http协议和servlet

javaweb

什么是javaweb?

JavaWeb 是指,所有通过 Java 语言编写可以通过浏览器访问的程序的总称,叫 JavaWeb。
JavaWeb 是基于请求和响应来开发的。

请求和响应

请求是指客户端给服务器发送数据,叫请求 Request。
响应是指服务器给客户端回传数据,叫响应 Response。
请求和响应是成对出现的,有请求就有响应。
在这里插入图片描述

常用的web服务器

Tomcat:由 Apache 组织提供的一种 Web 服务器,提供对 jsp 和 Servlet 的支持。它是一种轻量级的 javaWeb 容器(服务
器),也是当前应用最广的 JavaWeb 服务器(免费)。
Jboss:是一个遵从 JavaEE 规范的、开放源代码的、纯 Java 的 EJB 服务器,它支持所有的 JavaEE 规范(免费)。
GlassFish: 由 Oracle 公司开发的一款 JavaWeb 服务器,是一款强健的商业服务器,达到产品级质量(应用很少)。
Resin:是 CAUCHO 公司的产品,是一个非常流行的服务器,对 servlet 和 JSP 提供了良好的支持,
性能也比较优良,resin 自身采用 JAVA 语言开发(收费,应用比较多)。
WebLogic:是 Oracle 公司的产品,是目前应用最广泛的 Web 服务器,支持 JavaEE 规范,
而且不断的完善以适应新的开发要求,适合大型项目(收费,用的不多,适合大公司)。

Tomcat

目录介绍

bin 专门用来存放 Tomcat 服务器的可执行程序
conf 专门用来存放 Tocmat 服务器的配置文件
lib 专门用来存放 Tomcat 服务器的 jar 包
logs 专门用来存放 Tomcat 服务器运行时输出的日记信息
temp 专门用来存放 Tomcdat 运行时产生的临时数据
webapps 专门用来存放部署的 Web 工程。
work 是 Tomcat 工作时的目录,用来存放 Tomcat 运行时 jsp 翻译为 Servlet 的源码,和 Session 钝化的目录。

启动Tomcat服务器

找到 Tomcat 目录下的 bin 目录下的 startup.bat 文件,双击,就可以启动 Tomcat 服务器。

如何测试是否成功?
在浏览器中,输入以下地址测试:
1、http://localhost:8080
2、http://127.0.0.1:8080
3、http://真实 ip:8080

常见的启动失败的情况有,双击 startup.bat 文件,就会出现一个小黑窗口一闪而来。这个时候,失败的原因基本上都是因为没有配置好 JAVA_HOME 环境变量

Tomcat的停止

1、点击 tomcat 服务器窗口的 x 关闭按钮
2、把 Tomcat 服务器窗口置为当前窗口,然后按快捷键 Ctrl+C
3、找到 Tomcat 的 bin 目录下的 shutdown.bat 双击,就可以停止 Tomcat 服务器

Tomcat端口号的修改

Tomcat 默认的端口号是:8080
HTTP 协议默认的端口号是:80

找到 Tomcat 目录下的 conf 目录,找到 server.xml 配置文件,找到Connector标签,然后更改port即可

idea下的操作

在这里插入图片描述

HTTP协议

什么是 HTTP 协议

什么是协议?
协议是指双方,或多方,相互约定好,大家都需要遵守的规则,叫协议。
所谓 HTTP 协议,就是指,客户端和服务器之间通信时,发送的数据,需要遵守的规则,叫 HTTP 协议。
HTTP 协议中的数据又叫报文。

请求的 HTTP 协议格式

客户端给服务器发送数据叫请求。
服务器给客户端回传数据叫响应。
请求又分为 GET 请求,和 POST 请求两种

get请求
GET 请求
1、请求行
(1) 请求的方式 GET
(2) 请求的资源路径[+?+请求参数]
(3) 请求的协议的版本号 HTTP/1.1
2、请求头
key : value 组成 不同的键值对,表示不同的含义
在这里插入图片描述

post请求
ii. POST 请求
1、请求行
(1) 请求的方式 POST
(2) 请求的资源路径[+?+请求参数]
(3) 请求的协议的版本号 HTTP/1.1
2、请求头

  1. key : value 不同的请求头,有不同的含义
    空行
    3、请求体 ===>>> 就是发送给服务器的数据
    在这里插入图片描述
常用请求头的说明

Accept: 表示客户端可以接收的数据类型
Accpet-Languege: 表示客户端可以接收的语言类型
User-Agent: 表示客户端浏览器的信息
Host: 表示请求时的服务器 ip 和端口号

哪些是get请求,哪些是post请求

GET 请求有哪些:
1、form 标签 method=get
2、a 标签
3、link 标签引入 css
4、Script 标签引入 js 文件
5、img 标签引入图片
6、iframe 引入 html 页面
7、在浏览器地址栏中输入地址后敲回车
POST 请求有哪些:
form 标签 method=post

响应的 HTTP 协议格式

1、响应行
(1) 响应的协议和版本号
(2) 响应状态码
(3) 响应状态描述符
2、响应头
(1) key : value 不同的响应头,有其不同含义
空行
3、响应体 ---->>> 就是回传给客户端的数据
在这里插入图片描述

常用的响应码说明

200 表示请求成功
302 表示请求重定向(明天讲)
404 表示请求服务器已经收到了,但是你要的数据不存在(请求地址错误)
500 表示服务器已经收到请求,但是服务器内部错误(代码错误)

Servlet

什么是Servlet

1、Servlet 是 JavaEE 规范之一。规范就是接口
2、Servlet 就 JavaWeb 三大组件之一。三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监听器。
3、Servlet 是运行在服务器上的一个 java 小程序,它可以接收客户端发送过来的请求,并响应数据给客户端。

实现Servlet程序

1、编写一个类去实现 Servlet 接口(需要导servlet-api包,maven如下)

        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>javax.servlet-api</artifactId>
            <version>3.1.0</version>
            <scope>provided</scope>
        </dependency>

2、实现 service 方法,处理请求,并响应数据
3、到 web.xml 中去配置 servlet 程序的访问地址

具体实现:
java代码:

public class HelloServlet implements Servlet {
// service 方法是专门用来处理请求和响应的
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws
ServletException, IOException {
	System.out.println("Hello Servlet 被访问了");
}
}

web.xml文件的配置:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee
http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0">
<!-- servlet 标签给 Tomcat 配置 Servlet 程序 -->
<servlet>
<!--servlet-name 标签 Servlet 程序起一个别名(一般是类名) -->
	<servlet-name>HelloServlet</servlet-name>
<!--servlet-class 是 Servlet 程序的全类名-->
	<servlet-class>com.atguigu.servlet.HelloServlet</servlet-class>
</servlet>
<!--servlet-mapping 标签给 servlet 程序配置访问地址-->
<servlet-mapping>
<!--servlet-name 标签的作用是告诉服务器,我当前配置的地址给哪个 Servlet 程序使用-->
	<servlet-name>HelloServlet</servlet-name>
<!--url-pattern 标签配置访问地址 <br/>
/ 斜杠在服务器解析的时候,表示地址为:http://ip:port/工程路径 <br/>
/hello 表示地址为:http://ip:port/工程路径/hello <br/>
-->
	<url-pattern>/hello</url-pattern>
</servlet-mapping>
</web-app>
url地址到Servlet程序的访问

在这里插入图片描述

Servlet的生命周期

1、执行 Servlet 构造器方法
2、执行 init 初始化方法
第一、二步,是在第一次访问的时候创建 Servlet 程序会调用。
3、执行 service 方法
第三步,每次访问都会调用。
4、执行 destroy 销毁方法
第四步,在 web 工程停止的时候调用。

通过继承HttpServlet实现Servlet程序(日常)

一般都继承HttpServlet去实现Servlet程序

1、编写一个类去继承 HttpServlet 类
2、根据业务需要重写 doGet 或 doPost 方法
3、到 web.xml 中的配置 Servlet 程序的访问地址

java代码:

public class helloSevlet extends HttpServlet {
    public void service(HttpServletRequest servletRequest, HttpServletResponse servletResponse) throws ServletException, IOException {
        System.out.println("hello servlet被访问拉");
        String method = servletRequest.getMethod();
        System.out.println(method);
        String aa = servletRequest.getParameter("aa");
        System.out.println(aa);
    }
}

xml配置文件:

<!DOCTYPE web-app PUBLIC
 "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
 "http://java.sun.com/dtd/web-app_2_3.dtd" >

<web-app>
  <display-name>Archetype Created Web Application</display-name>
  <welcome-file-list>
    <welcome-file>qinqin.jsp</welcome-file>
  </welcome-file-list>
  <servlet>
    <servlet-name>haha</servlet-name>
    <servlet-class>Test.helloSevlet</servlet-class>
    <load-on-startup>2</load-on-startup>
  </servlet>
  <servlet-mapping>
    <servlet-name>haha</servlet-name>
    <url-pattern>/hello</url-pattern>
  </servlet-mapping>
</web-app>

< load-on-startup></ load-on-startup>:
1)load-on-startup元素标记容器是否在启动的时候就加载这个servlet(实例化并调用其init()方法)。
2)它的值必须是一个整数,表示servlet应该被载入的顺序
2)当值为0或者大于0时,表示容器在应用启动时就加载并初始化这个servlet;
3)当值小于0或者没有指定时,则表示容器在该servlet被选择时才会去加载。
4)正数的值越小,该servlet的优先级越高,应用启动时就越先加载。
5)当值相同时,容器就会自己选择顺序来加载。

所以,< load-on-startup>x</ load-on-startup>,中x的取值1,2,3,4,5代表的是优先级,而非启动延迟时间。

通常大多数Servlet是在用户第一次请求的时候由应用服务器创建并初始化,但< load-on-startup>n</ load-on-startup> 可以用来改变这种状况,根据自己需要改变加载的优先级!

jsp页面:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<form action="http://localhost:8083/01_servlet_war/hello" method="get">
    <input type="text" name="aa" id="aa">
    <input type="submit" >
</form>
</body>
</html>

ServletConfig类(每个Servlet一个)

ServletConfig 类从类名上来看,就知道是 Servlet 程序的配置信息类。
Servlet 程序和 ServletConfig 对象都是由 Tomcat 负责创建,我们负责使用。
Servlet 程序默认是第一次访问的时候创建**,ServletConfig 是每个 Servlet 程序创建时,就创建一个对应的 ServletConfig 对象**。

ServletConfig的作用
1、可以获取 Servlet 程序的别名 servlet-name 的值
2、获取初始化参数 init-param
3、获取 ServletContext 对象

代码实例:
web.xml文件配置:

  <!-- servlet 标签给 Tomcat 配置 Servlet 程序 -->
  <servlet>
    <!--servlet-name 标签 Servlet 程序起一个别名(一般是类名) -->
    <servlet-name>haha</servlet-name>
    <!--servlet-class 是 Servlet 程序的全类名-->
    <servlet-class>Test.helloSevlet</servlet-class>
    <!--init-param 是初始化参数-->
    <init-param>
      <!--是参数名-->
      <param-name>username</param-name>
      <!--是参数值-->
      <param-value>root</param-value>
    </init-param>
    <init-param>
      <param-name>password</param-name>
      <param-value>root2</param-value>
    </init-param>
  </servlet>

java文件获取:
ServletConfig可以用getServletConfig来返回一个ServletConfig对象

    public void service(HttpServletRequest servletRequest, HttpServletResponse servletResponse) throws ServletException, IOException {
        ServletConfig servletConfig = getServletConfig();
        //找到servletname标签内容
        String servletName = servletConfig.getServletName();
        System.out.println(servletName);
        //找到参数名为username的值
        String username = servletConfig.getInitParameter("username");
        System.out.println(username);
        //获取servletContext对象
        ServletContext servletContext = servletConfig.getServletContext();
        System.out.println(servletContext);
    }

注意点:
如果重写了init方法,一定要super调用父类的init方法,因为父类的init方法涉及到对servletconfig的配置
在这里插入图片描述

ServletContext类(全局变量,大家用一个)

什么是 ServletContext?
1、ServletContext 是一个接口,它表示 Servlet 上下文对象
2、一个 web 工程,只有一个 ServletContext 对象实例
3、ServletContext 对象是一个域对象
4、ServletContext 是在 web 工程部署启动的时候创建。在 web工程停止的时候销毁

什么是域对象?
域对象,是可以像 Map 一样存取数据的对象,叫域对象。
这里的域指的是存取数据的操作范围,整个 web 工程。

存数据取数据删除数据
Mapput()get()remove()
域对象setAttribute()getAttribute()removeAttribute()

ServletContext 类的四个作用
1、获取 web.xml 中配置的上下文参数 context-param
2、获取当前的工程路径,格式: /工程路径
3、获取工程部署后在服务器硬盘上的绝对路径(Tomcat中的路径)
4、像 Map 一样存取数据

xml文件中配置:

  <!--context-param 是上下文参数(它属于整个 web 工程)-->
  <context-param>
    <param-name>username</param-name>
    <param-value>root1</param-value>
  </context-param>
  
  <context-param>
    <param-name>namespace</param-name>
    <param-value>root2</param-value>
  </context-param>

java文件获取:

    public void service(HttpServletRequest servletRequest, HttpServletResponse servletResponse) throws ServletException, IOException {
        ServletConfig servletConfig = getServletConfig();
        ServletContext servletContext = servletConfig.getServletContext();

        String username = servletContext.getInitParameter("username");
        System.out.println("context-param 参数 username 的值是:"+username);

        System.out.println( "当前工程路径:" + servletContext.getContextPath() );
		//当前工程路径:/01_servlet_war
        System.out.println("工程部署的路径是:" + servletContext.getRealPath("/"));
        //工程部署的路径是:E:\java\apache-tomcat-8.5.43\webapps\01_servlet_war\
        System.out.println("工程下 css 目录的绝对路径是:" + servletContext.getRealPath("/css"));
        //工程部署的路径是:E:\java\apache-tomcat-8.5.43\webapps\01_servlet_war\css
        
        servletContext.setAttribute("key",11);
        Object key = servletContext.getAttribute("key");
        System.out.println(key);
    }
HttpServletRequest 类

HttpServletRequest 类有什么作用
每次只要有请求进入 Tomcat 服务器,Tomcat 服务器就会把请求过来的 HTTP 协议信息解析好封装到 Request 对象中。然后传递到 service 方法(doGet 和 doPost)中给我们使用。我们可以通过 HttpServletRequest 对象,获取到所有请求的信息。

HttpServletRequest类的常用方法
i. getRequestURI():获取请求的资源路径
ii. getRequestURL() :获取请求的统一资源定位符(绝对路径)
iii. getRemoteHost() :获取客户端的 ip 地址
iv. getHeader() :获取请求头
v. getParameter() :获取请求的参数
vi. getParameterValues(): 获取请求的参数(多个值的时候使用)
vii. getMethod(): 获取请求的方式 GET 或 POST
viii. setAttribute(key, value):设置域数据
ix. getAttribute(key): 获取域数据
x. getRequestDispatcher():获取请求转发对象
xi.setCharacterEncoding():设置请求体的字符集(post请求)

代码实例:

    public void service(HttpServletRequest servletRequest, HttpServletResponse servletResponse) throws ServletException, IOException {
        // i.getRequestURI() 获取请求的资源路径
        System.out.println("URI => " + servletRequest.getRequestURI());
        // ii.getRequestURL() 获取请求的统一资源定位符(绝对路径)
        System.out.println("URL => " + servletRequest.getRequestURL());
        // iii.getRemoteHost() 获取客户端的 ip 地址
        /**
         * 在 IDEA 中,使用 localhost 访问时,得到的客户端 ip 地址是 ===>>> 127.0.0.1<br/>
         * 在 IDEA 中,使用 127.0.0.1 访问时,得到的客户端 ip 地址是 ===>>> 127.0.0.1<br/>
         * 在 IDEA 中,使用 真实 ip 访问时,得到的客户端 ip 地址是 ===>>> 真实的客户端 ip 地址<br/>
         */
        System.out.println("客户端 ip 地址 => " + servletRequest.getRemoteHost());
        // iv.getHeader() 获取请求头
        System.out.println("请求头 User-Agent ==>> " + servletRequest.getHeader("User-Agent"));
        // vii.getMethod() 获取请求的方式 GET 或 POST
        System.out.println("请求的方式 ==>> " + servletRequest.getMethod());
    }
获取请求的参数

表单如下:

<form action="http://localhost:8083/01_servlet_war/hello" method="get">
    <input type="text" name="aa">
    兴趣爱好:<input type="checkbox" name="hobby" value="cpp">C++
    <input type="checkbox" name="hobby" value="java">Java
    <input type="checkbox" name="hobby" value="js">JavaScript<br/>
    <input type="submit" >
</form>

java代码来获取请求的参数:

    public void service(HttpServletRequest servletRequest, HttpServletResponse servletResponse) throws ServletException, IOException {
        //获取参数
        String aa = servletRequest.getParameter("aa");
        String[] hobbies = servletRequest.getParameterValues("hobby");
        System.out.println(aa);   //qqq
        System.out.println(Arrays.asList(hobbies));  //[cpp, java, js]
    }
doGet请求的中文乱码解决:

使用string的getBytes来编码和解码

// 获取请求参数
String username = req.getParameter("username");
//1 先以 iso8859-1 进行编码
//2 再以 utf-8 进行解码
username = new String(username.getBytes("iso-8859-1"), "UTF-8");
POST 请求的中文乱码解决

使用request的setCharacterEncoding方法来设置请求体的字符集为 UTF-8,达到解决乱码的目的
注意: 要放在所有接受参数(getParameter)的前面

// 设置请求体的字符集为 UTF-8,从而解决 post 请求的中文乱码问题
req.setCharacterEncoding("UTF-8");
System.out.println("-------------doPost------------");
// 获取请求参数
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobby = req.getParameterValues("hobby");
请求的转发

什么是请求的转发?
请求转发是指:服务器收到请求后,从一个资源跳转到另一个资源的操作叫请求转发。
我们通常采用getRequestDispatcher来获取请求转发对象
在这里插入图片描述
请求转发的特点:

  1. 浏览器地址栏没有变化
  2. 他们是一次请求
  3. 他们共享request域中的数据
  4. 可以转发到WEB-INF目录下(正常直接输网站不能进入)
  5. 不可以访问工程以外的资源

示例代码:
1 form表单提交请求到servlet1程序
2 servlet1程序:
servlet1转发给servlet2

    public void service(HttpServletRequest servletRequest, HttpServletResponse servletResponse) throws ServletException, IOException {
        //获取参数
        String aa = servletRequest.getParameter("aa");
        servletRequest.setAttribute("key",aa);
        //请求转发必须要以斜杠打头,/ 斜杠表示地址为:http://ip:port/工程名/ , 映射到 IDEA 代码的 web 目录
        RequestDispatcher requestDispatcher = servletRequest.getRequestDispatcher("/xml");
        requestDispatcher.forward(servletRequest, servletResponse);
    }

3 servlet2程序:

    protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        Object key = req.getAttribute("key");
        System.out.println(key);
        System.out.println("xmlloveyzl");
    }
base标签的作用

在这里插入图片描述
这时候我们就需要base标签,base标签可以设置当前页面中所有相对路径工作时,参照哪个路径来进行跳转

示例代码:
一共有三块组成,a.html和在aa包下的b.html和java代码块,默认路径是http://localhost:8083/01_servlet_war

a.html:

<a href="aa/b.html">去b</a>
<a href="http://localhost:8083/01_servlet_war/htmlServlet">去servlet</a>

java代码:

    protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("出现啦");
        req.getRequestDispatcher("/aa/b.html").forward(req, resp);
    }

b.html:

<a href="../a.html">返回</a>

上述的代码,从a页面能正常进入b页面,也能正常跳转回去;从java代码中能正常进入b页面,但是不能正常跳转回去。
原因如下:
a的路径如下:http://localhost:8083/01_servlet_war/a.html
java的servlet路径如下:http://localhost:8083/01_servlet_war/htmlServlet
a调用时b中的…/a.html路径为正确的路径
servlet调用时b中的路径为正确的路径http://localhost:8083/a.html,因为是在原路径基础上…了

解放方案:
加上base,base设置相对相对路径为aa包下,返回就是正常的了

<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <base href="http://localhost:8083/01_servlet_war/aa/">
</head>
<body>
<a href="../a.html">返回</a>
</body>
Web 中的相对路径和绝对路径

在 javaWeb 中,路径分为相对路径和绝对路径两种:
相对路径是:
. 表示当前目录
… 表示上一级目录
资源名 表示当前目录/资源名

绝对路径:
http://ip:port/工程路径/资源路径

在实际开发中,路径都使用绝对路径,而不简单的使用相对路径。
1、绝对路径
2、base+相对

web 中 / 斜杠的不同意义

在 web 中 / 斜杠 是一种绝对路径。
/ 斜杠 如果被浏览器解析,得到的地址是:http://ip:port/
< a href="/">斜杠< /a>

/ 斜杠 如果被服务器解析,得到的地址是:http://ip:port/工程路径
1、< url-pattern>/servlet1< /url-pattern>
2、servletContext.getRealPath(“/”);
3、request.getRequestDispatcher(“/”);

特殊情况: response.sendRediect(“/”); 把斜杠发送给浏览器解析。得到 http://ip:port/

HttpServletResponse 类
HttpServletResponse 类的作用

HttpServletResponse 类和 HttpServletRequest 类一样。每次请求进来,Tomcat 服务器都会创建一个 Response 对象传递给 Servlet 程序去使用。HttpServletRequest 表示请求过来的信息,HttpServletResponse 表示所有响应的信息,我们如果需要设置返回给客户端的信息,都可以通过 HttpServletResponse 对象来进行设置

两个输出流的说明
字节流getOutputStream();常用于下载(传递二进制数据)
字符流getWriter();常用于回传字符串(常用)

两个流同时只能使用一个。
使用了字节流,就不能再使用字符流,反之亦然,否则就会报错。

如何往客户端回传数据
    protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        PrintWriter writer = resp.getWriter();
        writer.println("真牛啊xml");
    }
响应的乱码解决

上述的代码运行时可以发现乱码,原因是客户端和服务器端的编码格式不一致。解决方式如下:

方式一:(不推荐使用)

// 设置服务器字符集为 UTF-8
resp.setCharacterEncoding("UTF-8");
// 通过响应头,设置浏览器也使用 UTF-8 字符集
resp.setHeader("Content-Type", "text/html; charset=UTF-8");

方式二:(推荐使用)
它会同时设置服务器和客户端都使用 UTF-8 字符集,还设置了响应头

// 它会同时设置服务器和客户端都使用 UTF-8 字符集,还设置了响应头
// 此方法一定要在获取流对象之前调用才有效
resp.setContentType("text/html; charset=UTF-8");
请求重定向

请求重定向,是指客户端给服务器发请求,然后服务器告诉客户端说。我给你一些地址。你去新地址访问。叫请求重定向(因为之前的地址可能已经被废弃)。
response默认是没有工程路径的。
在这里插入图片描述
请求重定向的特点:

  1. 浏览器地址栏会发生变化
  2. 两次请求
  3. 不共享Request域中数据(因为是两次请求)
  4. 不能访问WEB-INF下的资源
  5. 可以访问工程外的资源

请求重定向两种方式。

方式一:(不推荐使用)

// 设置响应状态码 302 ,表示重定向,(已搬迁)
resp.setStatus(302);
// 设置响应头,说明 新的地址在哪里
resp.setHeader("Location", "http://localhost:8080");

方式二:(推荐使用)

resp.sendRedirect("/01_servlet_war/aa/b.html");
//或
resp.sendRedirect("https://www.baidu.com");
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值