HTTP入门学习

HTTP

概念

Hyper Text Transfer Protocol 超文本传输协议

传输协议:定义了,客户端和服务器端通信时,发送数据的格式

特点:

1. 基于TCP/IP的高级协议

2. 默认端口号:80

3. 基于请求/响应模型的:一次请求对应一次响应

4. 无状态的:每次请求之间相互独立,不能交互数据

历史版本:

1.0:每一次请求响应都会建立新的连接

1.1:复用连接

请求消息数据格式

请求行

请求方式 请求url 请求协议/版本

GET /login.html HTTP/1.1

请求方式:

HTTP协议有7中请求方式,常用的有2种

GET:

1. 请求参数在请求行中,在url后。

2. 请求的url长度有限制的

3. 不太安全

POST:

1. 请求参数在请求体中

2. 请求的url长度没有限制的

3. 相对安全

请求头:客户端浏览器告诉服务器一些信息

请求头名称: 请求头值

常见的请求头:

1. User-Agent:浏览器告诉服务器,我访问你使用的浏览器版本信息

可以在服务器端获取该头的信息,解决浏览器的兼容性问题

2. Referer:http://localhost/login.html

告诉服务器,我(当前请求)从哪里来?

作用:

(1)防盗链:

(2)统计工作:

3. 请求空行

空行,就是用于分割POST请求的请求头,和请求体的。

4. 请求体(正文):

封装POST请求消息的请求参数的

字符串格式:

POST /login.html HTTP/1.1

Host: localhost

User-Agent: Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:60.0) Gecko/20100101 Firefox/60.0

Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8

Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2

Accept-Encoding: gzip, deflate

Referer: http://localhost/login.html

Connection: keep-alive

Upgrade-Insecure-Requests: 1

username=zhangsan

响应消息数据格式

Request

request对象和response对象的原理

1. request和response对象是由服务器创建的。我们来使用它们

2. request对象是来获取请求消息,response对象是来设置响应消息

request对象继承体系结构

ServletRequest -- 接口

| 继承

HttpServletRequest -- 接口

| 实现

org.apache.catalina.connector.RequestFacade 类(tomcat)

request功能

获取请求消息数据
1.获取请求行数据

GET /day14/demo1?name=zhangsan HTTP/1.1

方法:

1. 获取请求方式 :GET
* String getMethod()
2. (*)获取虚拟目录:/day14
* String getContextPath()
3. 获取Servlet路径: /requestDemo1
* String getServletPath()
4. 获取get方式请求参数:name=zhangsan
* String getQueryString()
5. (*)获取请求URI:/day14/requestDemo1
* String getRequestURI(): /day14/requestDemo1
* StringBuffer getRequestURL() :http://localhost/day14/requestDemo1
* URL:统一资源定位符 : http://localhost/day14/demo1
* URI:统一资源标识符 : /day14/requestDemo1
6. 获取协议及版本:HTTP/1.1
* String getProtocol()
7. 获取客户机的IP地址:
* String getRemoteAddr()

@Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1. 获取请求方式 :GET
        //* String getMethod()
        String method = request.getMethod();
        System.out.println(method);
        //2. (*)获取虚拟目录:/day14
        //* String getContextPath()
        String contextPath = request.getContextPath();
        System.out.println(contextPath);
        // 3. 获取Servlet路径: /demo1
        //* String getServletPath()
        String servletPath = request.getServletPath();
        System.out.println(servletPath);
        //4. 获取get方式请求参数:name=zhangsan
        //* String getQueryString()
        String queryString = request.getQueryString();
        System.out.println(queryString);
        // 5. (*)获取请求URI:/day14/demo1
        //* String getRequestURI():     /day14/demo1
        //* StringBuffer getRequestURL()  :http://localhost/day14/demo1
        String requestURI = request.getRequestURI();
        System.out.println(requestURI);
        StringBuffer requestURL = request.getRequestURL();
        System.out.println(requestURL);
        // 6. 获取协议及版本:HTTP/1.1
        //* String getProtocol()
        String protocol = request.getProtocol();
        System.out.println(protocol);
        // 7. 获取客户机的IP地址:
        //* String getRemoteAddr()
        String remoteAddr = request.getRemoteAddr();
        System.out.println(remoteAddr);
    }
2. 获取请求头数据

方法:

(*)String getHeader(String name):通过请求头的名称获取请求头的值

Enumeration<String> getHeaderNames():获取所有的请求头名称

3. 获取请求体数据:

请求体:只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数

步骤:

1. 获取流对象

BufferedReader getReader():获取字符输入流,只能操作字符数据

ServletInputStream getInputStream():获取字节输入流,可以操作所有类型数据

在文件上传知识点后讲解

2. 再从流对象中拿数据

@Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //获取请求消息体
        //1.获取字符流
        BufferedReader br = req.getReader();
        //2.获取数据
        String line=null;
        while ((line=br.readLine())!=null){
            System.out.println(line);
        }
    }
其他功能
1. 获取请求参数通用方式:

不论get还是post请求方式都可以使用下列方法来获取请求参数

1. String getParameter(String name):根据参数名称获取参数值 username=zs&password=123

//获取请求参数
String username = req.getParameter("username");
System.out.println("post");
System.out.println(username);

2. String[] getParameterValues(String name):根据参数名称获取参数值的数组 hobby=xx&hobby=game

/*根据参数名称获取参数值的数组*/
String[] hobbies = req.getParameterValues("hobby");
/*for (String hobby : hobbies) {
System.out.println(hobby);
}

3. Enumeration<String> getParameterNames():获取所有请求的参数名称

/*获取所有请求的参数名称*/
Enumeration<String> parameterNames = req.getParameterNames();
while (parameterNames.hasMoreElements()){
String name = parameterNames.nextElement();
System.out.println(name);
String value = req.getParameter(name);
System.out.println(value);
System.out.println("=========");
}

4. Map<String,String[]> getParameterMap():获取所有参数的map集合

/*获取所有参数的map集合*/
Map<String, String[]> parameterMap = req.getParameterMap();
Set<String> keySet = parameterMap.keySet();
for (String name : keySet) {
    //根据键获取值
    String[] values = parameterMap.get(name);
    System.out.println(name);
    for (String value : values) {
        System.out.println(value);
        System.out.println("=======");
    }
}

中文乱码问题:

get方式:tomcat 8 已经将get方式乱码问题解决了

post方式:会乱码

解决:在获取参数前,设置request的编码request.setCharacterEncoding("utf-8");

2. 请求转发:

一种在服务器内部的资源跳转方式

步骤:

1. 通过request对象获取请求转发器对象:

RequestDispatcher getRequestDispatcher(String path)

2. 使用RequestDispatcher对象来进行转发:

forward(ServletRequest request, ServletResponse response)

特点:

1. 浏览器地址栏路径不发生变化

2. 只能转发到当前服务器内部资源中。

3. 转发是一次请求

3. 共享数据:

域对象:一个有作用范围的对象,可以在范围内共享数据

request域:代表一次请求的范围,一般用于请求转发的多个资源中共享数据

方法:

1. void setAttribute(String name,Object obj):存储数据

2. Object getAttitude(String name):通过键获取值

3. void removeAttribute(String name):通过键移除键值对

4. 获取ServletContext:

ServletContext getServletContext()

案例:用户登录

用户登录案例需求:

1.编写login.html登录页面

username & password 两个输入框

2.使用Druid数据库连接池技术,操作mysql,day14数据库中user表

3.使用JdbcTemplate技术封装JDBC

4.登录成功跳转到SuccessServlet展示:登录成功!用户名,欢迎您

5.登录失败跳转到FailServlet展示:登录失败,用户名或密码错误

分析

开发步骤

1. 创建项目,导入html页面,配置文件,jar包

2. 创建数据库环境

			CREATE DATABASE day14;
			USE day14;
			CREATE TABLE USER(
			
				id INT PRIMARY KEY AUTO_INCREMENT,
				username VARCHAR(32) UNIQUE NOT NULL,
				PASSWORD VARCHAR(32) NOT NULL
			);

3. 创建包cn.itcast.domain,创建类User

package cn.itcast.domain;

/**
* 用户的实体类
*/
public class User {

    private int id;
    private String username;
    private String password;


    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPasswor
    d() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Override
    public String toString() {
        return "User{" +
            "id=" + id +
            ", username='" + username + '\'' +
            ", password='" + password + '\'' +
            '}';
    }
}

4. 创建包cn.itcast.util,编写工具类JDBCUtils

package cn.itcast.util;

import com.alibaba.druid.pool.DruidDataSourceFactory;

import javax.sql.DataSource;
import javax.xml.crypto.Data;
import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;

/**
* JDBC工具类 使用Durid连接池
*/
public class JDBCUtils {

    private static DataSource ds;

    static {

        try {
            //1.加载配置文件
            Properties pro = new Properties();
            //使用ClassLoader加载配置文件,获取字节输入流
            InputStream is = JDBCUtils.class.getClassLoader().getResourceAsStream("druid.properties");
            pro.load(is);

            //2.初始化连接池对象
            ds = DruidDataSourceFactory.createDataSource(pro);

        } catch (IOException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
* 获取连接池对象
*/
    public static DataSource getDataSource() {
        return ds;
    }


    /**
* 获取连接Connection对象
*/
    public static Connection getConnection() throws SQLException {
        return ds.getConnection();
    }
}

5. 创建包cn.itcast.dao,创建类UserDao,提供login方法

package cn.itcast.dao;

import cn.itcast.domain.User;
import cn.itcast.util.JDBCUtils;
import org.springframework.dao.DataAccessException;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.jdbc.core.JdbcTemplate;

/**
* 操作数据库中User表的类
*/
public class UserDao {

    //声明JDBCTemplate对象共用
    private JdbcTemplate template = new JdbcTemplate(JDBCUtils.getDataSource());

    /**
* 登录方法
*
* @param loginUser 只有用户名和密码
* @return user包含用户全部数据, 没有查询到,返回null
*/
    public User login(User loginUser) {
        try {
            //1.编写sql
            String sql = "select * from user where username = ? and password = ?";
            //2.调用query方法
            User user = template.queryForObject(sql,
                                                new BeanPropertyRowMapper<User>(User.class),
                                                loginUser.getUsername(), loginUser.getPassword());


            return user;
        } catch (DataAccessException e) {
            e.printStackTrace();//记录日志
            return null;
        }
    }
}

6. 编写cn.itcast.web.servlet.LoginServlet类

package cn.itcast.web.servlet;

import cn.itcast.dao.UserDao;
import cn.itcast.domain.User;

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("/loginServlet")
    public class LoginServlet extends HttpServlet {


        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            //1.设置编码
            req.setCharacterEncoding("utf-8");
            //2.获取请求参数
            String username = req.getParameter("username");
            String password = req.getParameter("password");
            //3.封装user对象
            User loginUser = new User();
            loginUser.setUsername(username);
            loginUser.setPassword(password);

            //4.调用UserDao的login方法
            UserDao dao = new UserDao();
            User user = dao.login(loginUser);

            //5.判断user
            if (user == null) {
                //登录失败
                req.getRequestDispatcher("/failServlet").forward(req, resp);
            } else {
                //登录成功
                //存储数据
                req.setAttribute("user", user);
                //转发
                req.getRequestDispatcher("/successServlet").forward(req, resp);
            }

        }

        @Override
        protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            this.doGet(req, resp);
        }
    }

7. 编写FailServlet和SuccessServlet类

@WebServlet("/successServlet")
    public class SuccessServlet extends HttpServlet {
        protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
            //获取request域中共享的user对象
            User user = (User) request.getAttribute("user");

            if (user != null) {
                //给页面写一句话

                //设置编码
                response.setContentType("text/html;charset=utf-8");
                //输出
                response.getWriter().write("登录成功!" + user.getUsername() + ",欢迎您");
            }


        }


        @WebServlet("/failServlet")
            public class FailServlet extends HttpServlet {
                protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
                    //给页面写一句话

                    //设置编码
                    response.setContentType("text/html;charset=utf-8");
                    //输出
                    response.getWriter().write("登录失败,用户名或密码错误");

                }

                protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
                    this.doPost(request, response);
                }
            }

8. login.html中form表单的action路径的写法

虚拟目录+Servlet的资源路径

9. BeanUtils工具类,简化数据封装

1.用于封装JavaBean的

1. JavaBean:标准的Java类

要求:

1. 类必须被public修饰

2. 必须提供空参的构造器

3. 成员变量必须使用private修饰

4. 提供公共setter和getter方法

功能:封装数据

2. 概念:

成员变量:

属性:setter和getter方法截取后的产物

例如:getUsername() --> Username--> username

HTTP响应消息

概述

请求消息:客户端发送给服务器端的数据

数据格式:

(1)请求行

(2)请求头

(3)请求空行

(4)请求体

响应消息:服务器端发送给客户端的数据

数据格式:

1.响应行

(1)组成:协议/版本 响应状态码 状态码描述

(2)响应状态码:服务器告诉客户端浏览器本次请求和响应的一个状态。

状态码都是3位数字

分类:

1xx:服务器就收客户端消息,但没有接受完成,等待一段时间后,发送1xx多状态码

2xx:成功。代表:200

3xx:重定向。代表:302(重定向),304(访问缓存)

4xx:客户端错误。

代表:

404(请求路径没有对应的资源)

405:请求方式没有对应的doXxx方法

5xx:服务器端错误。代表:500(服务器内部出现异常)

2. 响应头:

格式:头名称: 值

常见的响应头:

Content-Type:服务器告诉客户端本次响应体数据格式以及编码格式

Content-disposition:服务器告诉客户端以什么格式打开响应体数据

值:

in-line:默认值,在当前页面内打开

attachment;filename=xxx:以附件形式打开响应体。文件下载

3. 响应空行

4. 响应体:传输的数据

Response对象

功能:设置响应消息

1. 设置响应行

1. 格式:HTTP/1.1 200 ok

2. 设置状态码:setStatus(int sc)

2. 设置响应头:setHeader(String name, String value)

3. 设置响应体:

使用步骤:

1. 获取输出流

字符输出流:PrintWriter getWriter()

字节输出流:ServletOutputStream getOutputStream()

2. 使用输出流,将数据输出到客户端浏览器

案例

完成重定向

重定向:资源跳转的方式

代码实现:

//设置响应码
resp.setStatus(302);
//设置响应头
resp.setHeader("location","/day15/responseDemo2");

//简单的重定向方法
resp.sendRedirect("/day15/ responseDemo2");

重定向的特点:redirect

1. 地址栏发生变化

2. 重定向可以访问其他站点(服务器)的资源

3. 重定向是两次请求。不能使用request对象来共享 数据

转发的特点:forward

1. 转发地址栏路径不变

2. 转发只能访问当前服务器下的资源

3. 转发是一次请求,可以使用request对象来共享数据

路径

路径分类

1.相对路径:通过相对路径不可以确定唯一资源

如:./index.html

不以/开头,以.开头路径

规则:找到当前资源和目标资源之间的相对位置关系

./:当前目录

../:后退一级目录

2.绝对路径:通过绝对路径可以确定唯一资源

如:http://localhost/day15/responseDemo2 /day15/responseDemo2

以/开头的路径

规则:判断定义的路径是给谁用的?判断请求将来从哪儿发出

给客户端浏览器使用:需要加虚拟目录(项目的访问路径)

建议虚拟目录动态获取:request.getContextPath()

<a> , <form> 重定向...

给服务器使用:不需要加虚拟目录

转发路径

服务器输出字符数据

步骤:

1. 获取字符输出流

2. 输出数据

注意:

乱码问题:

1. PrintWriter pw = response.getWriter();获取的流的默认编码是ISO-8859-1

2. 设置该流的默认编码

3. 告诉浏览器响应体使用的编码

/*//获取流对象之前,设置流的默认编码:ISO-8859-1 设置为:GBK
resp.setCharacterEncoding("utf-8");
//告诉浏览器,服务器发送的消息体数据的编码。建议浏览器使用该编码编码
resp.setHeader("content-type","type/html;charset=utf-8");*/
//简单的形式,设置编码,是在获取流之前设置
resp.setContentType("type/html;charset=GBK");

//1.获取字符输出流
PrintWriter pw = resp.getWriter();
//2.输出数据
pw.write("<h1>你好 hello</h1>");
服务器输出字节数据到浏览器

步骤:

1. 获取字节输出流

2. 输出数据

//1.获取字节输出流
ServletOutputStream sos = resp.getOutputStream();
//2.输出数据
sos.write("你好".getBytes());
验证码
int width = 100;
int height = 50;
//1.创建一对象,在内存中图片(验证码图片对象)
BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
//2.美化图片
//2.1 填充背景色
Graphics g = image.getGraphics();//画笔对象
g.setColor(Color.PINK);//设置画笔颜色
g.fillRect(0, 0, width, height);
//2.2画边框
g.setColor(Color.BLUE);//设置画笔颜色
g.drawRect(0, 0, width - 1, height - 1);

String str = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
//生成随机角标
Random ran = new Random();
for (int i = 1; i <= 4; i++) {
    int index = ran.nextInt(str.length());
    //获取字符
    char ch = str.charAt(index);
    //2.3写验证码
    g.drawString(ch + "", width / 5 * i, height / 2);
}

//2.4画干扰线
g.setColor(Color.GREEN);
for (int i = 0; i < 10; i++) {
    int x1 = ran.nextInt(width);
    int x2 = ran.nextInt(width);
    int y1 = ran.nextInt(height);
    int y2 = ran.nextInt(height);
    g.drawLine(x1, x2, y1, y2);
}

//3.将图片输出到页面展示
ImageIO.write(image, "jpg", resp.getOutputStream());

ServletContext对象:

ServletContext

概念:代表整个web应用,可以和程序的容器(服务器)来通信

获取:

通过request对象获取:request.getServletContext();

通过HttpServlet获取:this.getServletContext();

功能:

获取MIME类型:

MIME类型:在互联网通信过程中定义的一种文件数据类型

格式: 大类型/小类型 text/html image/jpeg

获取:String getMimeType(String file)

//通过HttpServlet获取
ServletContext context = this.getServletContext();
//定义文件名称
String filename = "a.jpg";
//获取MIME类型
String mimeType = context.getMimeType(filename);
System.out.println(mimeType);//image/jpeg

域对象:共享数据

1. setAttribute(String name,Object value)

2. getAttribute(String name)

3. removeAttribute(String name)

ServletContext对象范围:所有用户所有请求的数据

获取文件的真实(服务器)路径

方法:String getRealPath(String path)

//通过HttpServlet获取
ServletContext context = this.getServletContext();
String path1 = context.getRealPath("/b.txt");//web目录下资源访问
System.out.println(path1);
String path2 = context.getRealPath("/WEF-INF/c.txt");//WEB-INF目录下资源访问
System.out.println(path2);
String path3 = context.getRealPath("/WEF-INF/classes/a.txt");//src目录下资源访问
System.out.println(path3);

案例

文件下载需求:

1. 页面显示超链接

2. 点击超链接后弹出下载提示框

3. 完成图片文件下载

分析:

1. 超链接指向的资源如果能够被浏览器解析,则在浏览器中展示,如果不能解析,则弹出下载提示框。不满足需求

2. 任何资源都必须弹出下载提示框

3. 使用响应头设置资源的打开方式:

content-disposition:attachment;filename=xxx

步骤:

1. 定义页面,编辑超链接href属性,指向Servlet,传递资源名称filename

2. 定义Servlet

1. 获取文件名称

2. 使用字节输入流加载文件进内存

3. 指定response的响应头: content-disposition:attachment;filename=xxx

4. 将数据写出到response输出流

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8">
    <title>Title</title>
  </head>
  <body>
    <a href="/day15/img/1.jpg">图片1</a>
    <a href="/day15/img/2.jpg">图片2</a>
    <hr>
    <a href="/day15/downloadServlet?filename=img/九尾.jpg">图片1</a>
    <a href="/day15/downloadServlet?filename=img/2.jpg">图片2</a>
  </body>
</html>
package cn.itcast.web.download;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.FileInputStream;
import java.io.IOException;

@WebServlet("/downloadServlet")
    public class downloadServlet extends HttpServlet {
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            //1.获取请求参数,文件名称
            String filename = req.getParameter("filename");
            //2.使用字节输入流加载文件进入内存
            //2.1找到文件服务器路径
            ServletContext servletContext = this.getServletContext();
            String realPath = servletContext.getRealPath(filename);
            //2.2用字节流关联
            FileInputStream fis = new FileInputStream(realPath);

            //3.设置响应头
            //3.1设置响应头类型
            String mimeType = servletContext.getMimeType(filename);
            resp.setHeader("content-type", mimeType);
            //3.2设置响应头打开方式
            resp.setHeader("content-disposition", "attachment;filename=" + filename);

            //4.用输入流的数据写出到输出流
            ServletOutputStream sos = resp.getOutputStream();
            byte[] buff = new byte[1024 * 8];
            int len = 0;
            while ((len = fis.read(buff)) != -1) {
                sos.write(buff, 0, len);
            }

            fis.close();

        }

        @Override
        protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            this.doGet(req, resp);
        }
    }

问题:

中文文件问题

解决思路:

1. 获取客户端使用的浏览器版本信息

2. 根据不同的版本信息,设置filename的编码方式不同

//解决中文文件名问题
//1.获取user-agent请求头
String agent=requset.getHeader("user-agent");
//2.使用工具类方法编码文件即可
filename = DownLoadUtils.getFilename(agent,filename);

resp.setHeader("content-disposition", "attachment;filename=" + filename);

工具类

package cn.itcast.utils;

import sun.misc.BASE64Encoder;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;


public class DownLoadUtils {
    public static String getFileName(String agent, String filename) throws UnsupportedEncodingException {
        if (agent.contains("MSIE")) {
            // IE浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+", " ");
        } else if (agent.contains("Firefox")) {
            // 火狐浏览器
            BASE64Encoder base64Encoder = new BASE64Encoder();
            filename = "=?utf-8?B?" + base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
        } else {
            // 其它浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        return filename;
    }
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值