JavaWeb的Servlet学习之Request03

news2025/1/26 14:37:55

目录

1.Request

1.1Request执行流程

1.2request对象和response对象的原理

1.3 request对象继承体系结构

 1.4request功能:

1.3.1获取请求消息数据

1.获取请求行数据

2.获取请求头

3.获取请求体数据

4.其他功能

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

        4.2请求转发:一种在服务器内部的资源跳转方式

        4.3共享数据

        4.4获取ServletContext:

2.综合案例-登录


1.Request

1.1Request执行流程

1.2request对象和response对象的原理

1.request和response对象是由服务器创建的。我们来使用它们
2.request对象是来获取请求消息,response对象是来设置响应消息request对象继承体系结构:

1.3 request对象继承体系结构

ServletRequest  ----接口
    「继承
HttpServletRequest ----接口


    丨实现
org.apache.catalina.connector.RequestFacade(tomcat)

 1.4request功能:

1.3.1获取请求消息数据

1.获取请求行数据

GET /day14/demo1?name=zhangsan HTTP/1.1
*方法:
1,获取请求方式:GET
(*)string getMethod()
2.获取虚拟目录:/day14
string getcontextPath()
3.获取Servlet路径:/demo1
string getservletpath()
4,获取get方式请求参数:name=zhangsan
string getQuerystring()
5,获取请求URI:/day14/demo1
(*)string getRequestURI():
/day14/demo1
stringBuffer getRequestURL()
http://localhost/day14/demo1
6,获取协议及版本:HTTP/1,1
string getprotocol()
7,获取客户机的IP地址:
string getRemoteAddr()

2.获取请求头

*方法:
(*)String getHeader(String name):通过请求头的名称获取请求头的值
Enumeration<String>getHeaderNames():获取所有的请求头名称

 @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse resp) throws ServletException, IOException {
      //演示获取请求头数据
        //1.获取所有请求头的名称
        Enumeration<String> headerNames = request.getHeaderNames();
        //2.遍历
        while (headerNames.hasMoreElements()){
            String name = headerNames.nextElement();
            //根据名称获取请求头的值
            String value = request.getHeader(name);
            System.out.println(name+"---"+value);
        }

    }

演示获取请求头数据 user-agent

    //演示获取请求头数据:user-agent
        String agent = request.getHeader("user-agent");
       if (agent.contains("Chrom")){
           System.out.println("谷歌来了...");
       }else if(agent.contains("Edge")){
                System.out.println("Edge");
            }else {
           System.out.println("不知名浏览器");
       }
        }

演示获取请求头数据 referer

防盗链

package com.hzw.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;
import java.util.Enumeration;

@WebServlet("/demo7")
public class RequestDemo4 extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        super.doPost(req, resp);
    }
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse resp) throws ServletException, IOException {
      //演示获取请求头数据 refer
        String referer = request.getHeader("referer");
        System.out.println(referer);  //http://localhost:8080/day14/login.html

        //防盗链
        if (referer!=null){
            if (referer.contains("/day14")){
                //正常访问
                System.out.println("播放电视剧");
            }else {
                System.out.println("想看电影吗?来腾讯");
            }
        }
    }
}

3.获取请求体数据

请求体:只有P0ST请求方式,才有请求体,在请求体中封装了P0ST请求的请求参数
*步骤:
1,获取流对象
,BufferedReader getReader():获取字符输入流,只能操作字符数据
ServletInputstream getInputstream():获取字节输入流,可以操作所有类型数据
在文件上传知识点后讲解

    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    //获取请求体 --请求参数
        //1.获取字节流
        BufferedReader br = request.getReader();
        //2.读取数据
        String line=null;
        while ((line=br.readLine())!=null){
            System.out.println(line);
        }
    }

4.其他功能

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

1.String getparameter(String name):根据参数名称获取参数值usernamea=zs&password=l23
2.string[] getParametervalues(String name):根据参数名称获取参数值的数组hobby=study&hobby=game
3.Enumeration<string>getParameterNames():获取所有请求的参数名称
4,Map<String,String【】>getParameterMap():获取P所有奇数的map集合

常中文乱码问题:
*get方式:tomcat8已经将get方式乱码问题解决了
post方式:会码
*解决:在获取参数前,设置requesti的编码request,setcharacterEncoding("utf-8");

代码实现

package com.hzw.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.BufferedReader;
import java.io.IOException;
import java.util.Enumeration;
import java.util.Map;
import java.util.Set;

@WebServlet("/requestDemo6")
public class RequestDemo6 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  //post获取请求参数
        //根据参数名称获取参数值
        String username = request.getParameter("username");
//        System.out.println("post");
//        System.out.println(username);
        //根据参数名称获取参数值
        String[] hobbies = request.getParameterValues("hobby");
//        for (String hobby : hobbies) {
//            System.out.println(hobby);
//        }
        //根据所有请求的参数名称  keys
        Enumeration<String> parameterNames = request.getParameterNames();
        while (parameterNames.hasMoreElements()){
            String name = parameterNames.nextElement();
            System.out.println(name);
            String value = request.getParameter(name);
            System.out.println(value);
            System.out.println("-----------");
        }
        //获取所有参数的map集合
        Map<String, String[]> parameterMap = request.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("--------");
        }


    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //get 获取请求参数
        //根据参数名称获取参数值
//        String username = request.getParameter("username");
//        System.out.println("get");
//        System.out.println(username);
        this.doPost(request,response);
    }
}

register2.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<form action="/requestDemo6" method="post">
    <input type="text" name="username" placeholder="请输入名字"><br/>
    <input type="text" name="password" placeholder="请输入密码"><br/>
    <input type="checkbox" name="hobby" value="game">游戏
    <input type="checkbox" name="hobby" value="study">学习<br/>
    <input type="submit" value="注册">
</form>
</body>
</html>

4.2请求转发:一种在服务器内部的资源跳转方式

1.步骡:
1.1通过request对象获取请求转发器对象:RequestDispatcher getRequestDispatcher(string path)
1.2使用RequestDispatcher对象来进行转发:forward(ServletRequest request,ServletResponse response)
2.特点:
2.1浏览器地址栏路径不发生变化
2.2,只能转发到当前服务器内部资源中。
2.3,转发是一次请求

 4.3共享数据

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

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

方法:

1.void setAttribute(String name,object obj):存储数据
2.object getAttitude(string name):通过键获取值
3.void removeAttribute(String name):通过键移除键值对

4.4获取ServletContext:

*ServletContext getservletcontext() 

2.综合案例-登录

 代码实现

导入相应jar包

 记得lib 放入模块

1.建表user

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

INSERT INTO USER(username,PASSWORD) VALUES('hzw','123')
2.User实体类

package com.hzw.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 getPassword() {
        return password;
    }

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

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

3.UserDao 实现登录

package com.hzw.dao;

import com.hzw.domain.User;
import com.hzw.util.JDBCUtils;
import org.springframework.dao.DataAccessException;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.jdbc.core.JdbcTemplate;

import java.util.List;

/**
 * 操作数据库中User表中类
 */
public class UserDao {
    //声明JDBCTemplate对象共用
    private JdbcTemplate template=new JdbcTemplate(JDBCUtils.getDataSource());

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

            return users;
        } catch (DataAccessException e) {
            e.printStackTrace();
            return null;
        }
    }

}

4.jdbc连接代码

druid.properties

driverClassName=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/javaweb?userSSL=true&serverTimezone=Asia/Shanghai
username=root
password=root
initialSize=5
maxActive=10
maxWait=3000
timeBetweenEvictionRunsMillis=60000
minEvictableIdleTimeMillis=300000
validationQuery=SELECT 1
testWhileIdle=true
testOnBorrow=false
testOnReturn=false
poolPreparedStatements=false
maxPoolPreparedStatementPerConnectionSize=200

jdbcUtils

package com.hzw.util;
import com.alibaba.druid.pool.DruidDataSourceFactory;
import javax.sql.DataSource;
import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;
/**
 * jdbc工具类
 */
public class JDBCUtils {
private static DataSource ds;
static {
    //1加载配置文件
    Properties properties = new Properties();
    try {
        //使用ClassLoader加载配置文件,获取字节输入流
        InputStream is = JDBCUtils.class.getClassLoader().getResourceAsStream("druid.properties");
        properties.load(is);
        //2.初始化连接池对象
        ds=DruidDataSourceFactory.createDataSource(properties);
    }catch (IOException e){
        e.printStackTrace();
    } catch (Exception e) {
        e.printStackTrace();
    }
}
    /**
     * 获取连接池对象
     */
    public static DataSource getDataSource(){
        return ds;
    }
    /**
     * 获取连接池对象
     * @return
     */
    public static Connection getConnection() throws SQLException {
        return ds.getConnection();

    }
}

Servlet

loginServlet 如果登录成功跳转到SuccessServlet  失败跳转FailServlet

package com.hzw.web.servlet;

import com.hzw.dao.UserDao;
import com.hzw.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("/login1")
public class LoginServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
      this.doPost(request,response);
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1.设置编码
        request.setCharacterEncoding("utf-8");
        //2.获取请求参数
        String username = request.getParameter("username");
        String password = request.getParameter("password");
        //3.封装user对象
        User loginUser = new User();
        loginUser.setUsername(username);
        loginUser.setPassword(password);
        //4.调用UserDao的login
        UserDao userDao = new UserDao();
        User user = userDao.login(loginUser);
        //5.判断user
        if (user ==null){
            //登录失败
            request.getRequestDispatcher("/failServlet").forward(request,response);
        }else {
            //登录成功
            //存储数据
            request.setAttribute("user",user);
            //转发
            request.getRequestDispatcher("/successServlet").forward(request,response);
        }
    }
}
package com.hzw.web.servlet;

import com.hzw.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("/successServlet")
public class SuccessServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //获取request域中共享对象
        User user = (User)request.getAttribute("user");
        if (user!=null){
            //设置编码
            response.setContentType("text/html;charset=utf-8");
            //输出响应
            response.getWriter().write("登录成功,"+user.getUsername()+"欢迎您大佬");
        }
    }

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

    }
}
package com.hzw.web.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("/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);

    }
}

设置虚拟路径

登录html

<!DOCTYPE html>
<html >
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <form action="/hzw/login1" method="post">
        用户名:<input type="text" name="username" placeholder="请输入用户名"> <br>
        密码:<input type="password" name="password" placeholder="请输入密码"> <br>
        <input type="submit" value="登录">
    </form>
</body>
</html>

运行截图:登陆成功 跳转SuccessServlet 响应数据

使用BeanUtils封装数据

导入对应jar包

   @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1.设置编码
        request.setCharacterEncoding("utf-8");
        //2.获取所有请求参数
        Map<String, String[]> map = request.getParameterMap();
        //3.创建user对象
        User loginUser = new User();
        //3,1使用BeanUtils封装
        try {
            BeanUtils.populate(loginUser,map);
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

        //4.调用UserDao的login
        UserDao userDao = new UserDao();
        User user = userDao.login(loginUser);
        //5.判断user
        if (user ==null){
            //登录失败
            request.getRequestDispatcher("/failServlet").forward(request,response);
        }else {
            //登录成功
            //存储数据
            request.setAttribute("user",user);
            //转发
            request.getRequestDispatcher("/successServlet").forward(request,response);
        }
    }

BeanUtils.工具类,简化数据封装
*用于封装JavaBean的
1,]avaBean:标准的Java类
1,要求:
1,类必须被public修饰
2,必须提供空参的构造器
3,成员变量必须使用private修饰
4.提供公共setter和getter方法
2,功能:封装数据
2,概念:
成员变量:
属性:setter和getter方法截取后的产物
例如:getusername()-->Username-->username
3,方法:
1.setProperty()
2.getProperty()
3.populate(Object obj ,Map map) 将map集合的键值对信息,封装到对应的JavaBean对象中。

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

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

相关文章

开源CA搭建-基于openssl实现数字证书的生成与分发

目录 一、前言 二、openssl介绍 三、openssl的常用用法 &#xff08;一&#xff09;单向加密 &#xff08;二&#xff09;生成随机数 &#xff08;三&#xff09;生成公钥&#xff0c;私钥 1.生成私钥 2.提取公钥 四、搭建CA &#xff08;一&#xff09;创建根CA私钥…

Linux的camera驱动 摄像头调试方法

CameraInfo类用来描述相机信息&#xff0c;通过Camera类中getCameraInfo(int cameraId, CameraInfo cameraInfo)方法获得&#xff0c; 主要包括以下两个成员变量facing&#xff0c;facing 代表相机的方向&#xff0c; 它的值只能是CAMERA_FACING_BACK&#xff08;后置摄像头&am…

Golang 【basic_leaming】1 基本语法

阅读目录Go 语言变量Go 语言 - 声明变量1. 标准格式2. 批量格式Go 语言 - 初始化与定义变量1. 标准格式2. 编译器推导类型格式3. 短变量声明与初始化Go语言 - 多变量同时赋值Go 语言 - 匿名变量参考资料Go 语言整型&#xff08;整数类型&#xff09;1 自动匹配平台的 int 和 un…

新项目为什么决定用 JDK 17了

大家好&#xff0c;我是风筝。公众号「古时的风筝」&#xff0c;专注于后端技术&#xff0c;尤其是 Java 及周边生态。文章会收录在 JavaNewBee 中&#xff0c;更有 Java 后端知识图谱&#xff0c;从小白到大牛要走的路都在里面。 最近在调研 JDK 17&#xff0c;并且试着将之前…

阴差阳错,阴阳之变

北京的第一批“杨康”们已经返回到工作岗位&#xff0c;这其中就包括我。简单总结一下我的感染和康复过程&#xff0c;给大家做个样本吧。我属于北京放开的第一波感染者&#xff0c;12.9日当天感觉嗓子干&#xff0c;毫不犹豫&#xff0c;果然是中招了&#xff1b;周末开始发烧…

特朗普发行NFT惹群嘲,上线售罄现“真香定律”

文/章鱼哥出品/陀螺财经特朗普14日在其创建的社交平台truth social上发帖称&#xff0c;“美国需要一个超级英雄”。他还预告自己将于当地时间15日宣布“重大消息”。据《新闻周刊》报道&#xff0c;特朗普当日在其社交平台上发了一段十几秒的视频&#xff0c;里面有一个他站在…

Windows实时运动控制软核(三):LOCAL高速接口测试之C++

今天&#xff0c;正运动小助手给大家分享一下MotionRT7的安装和使用&#xff0c;以及使用C对MotionRT7开发的前期准备。 01 MotionRT7简介 MotionRT7是深圳市正运动技术推出的跨平台运动控制实时内核&#xff0c;也是国内首家完全自主自研&#xff0c;自主可控的Windows运动控…

Linux搭建测试环境详细步骤

本文讲解如何在Linux CentOS下部署Java Web项目的步骤 环境准备 &#xff08;1&#xff09;Linux系统&#xff08;2&#xff09;JDK&#xff08;3&#xff09;Tomcat &#xff08;4&#xff09;MySQL工具下载 一、Linux系统 本文主要是Linux CentOS7为例 自己在家练习小项…

[拆轮子] PaddleDetection 中的 COCODataSet 是怎么写的

今日&#xff0c;辗转反侧&#xff0c;该&#x1f4a9;的代码就是跑不成功&#xff0c;来看看 COCODataSet 到底是怎么写的&#xff0c;本文只参考当前版本的代码&#xff0c;当前版本 PaddleDetection2.5 COCODataSet 源码见附录 COCODataSet 类内部就三个函数&#xff1a; …

词义和词义消歧

Synsets(“synonym sets”, effectively senses) are the basic unit of organization in WordNet.同义词集 对于许多应用程序&#xff0c;我们希望消除歧义 • 我们可能只对一种含义感兴趣 • 在网络上搜索chemical plant 化工厂&#xff0c;我们不想搜到香蕉中的化学物质 所以…

【SpringBoot扩展点】 容器刷新前回调ApplicationContextInitializer

本文将作为Spring系列教程中源码版块的第一篇&#xff0c;整个源码系列将分为两部分进行介绍&#xff1b;单纯的源码解析&#xff0c;大概率是个吃力没人看的事情&#xff0c;因此我们将结合源码解析&#xff0c;一个是学习下别人的优秀设计&#xff0c;一个是站在源码的角度看…

【MySQL】索引和事务重点知识汇总

目录1.索引:1.1 索引的使用:1.2 索引背后的核心数据结构:1.2.1 先认识 B 树(N叉搜索树):1.2.2 再认识 B 树(N叉搜索树):2.事务:2.1 隔离性:2.1.1 脏读问题:2.1.2 不可重复读问题:2.1.3 幻读问题:2.1.4 总结:2.1.5 隔离级别:1.索引: 索引存在的意义就是为了提高查询到效率.索引…

【AI理论学习】Python机器学习中的特征选择

Python机器学习中的特征选择特征选择方法特征选择的Python库使用Scikit-learn实现特征选择方差卡方检验ANOVALasso正则化递归特征消除使用Feature-engine进行特征选择单变量特征选择相关性Python 中的更多特性选择方法参考资料任何数据科学项目的一个重要步骤是选择最具预测性的…

vue实现文件下载

引言 最近在自己做项目的需求的过程中&#xff0c;需要vuespringboot实现文件的下载功能&#xff08;导出博客文件&#xff09;。 问题重现 在我后端文件下载接口开发完成后&#xff0c;使用vue前端去进行对接时出现了问题。 我是直接使用的axios去进行请求接口&#xff0c…

Python 炫技操作:条件语句的七种写法

原代码 这是一段非常简单的通过年龄判断一个人是否成年的代码&#xff0c;由于代码行数过多&#xff0c;有些人就不太愿意这样写&#xff0c;因为这体现不出自己多年的 Python 功力。 if age > 18:return "已成年" else:return "未成年"下面我列举了六…

SwiftUI 中创建谷歌字体浏览器

Google Fonts是设计用户界面时使用的免费字体的转到站点。本教程将展示如何编写一个简单的工具来预览这些字体,而无需在系统中注册每种字体。 该应用程序包含一个拆分视图,该视图在左侧面板中包含字体列表。右侧面板将显示字体样式选项的预览。 项目设置 创建一个名为 Googl…

Vue2之webpack篇(一)

目录 前言 1、什么是webpack&#xff1f; 2、传统开发模式 一、传统开发模式 1、场景 2、问题 3、原因 4、解决方案 二、ES6模块化 1、ES6的解决方案 3、拓展 4、取别名 5、*搭配取别名 6、导出default{} 三、CommonJS规范 1、推荐文档 2、使用CommonJS规范解决方…

十二、DockerFile构建过程解析

1、概述 Dockerfile是用来构建Docker镜像的文本文件&#xff0c;是由一条条构建镜像所需的指令和参数构成的脚本。 在Docker 常用命令篇中&#xff0c;我们已经知道了2中构建镜像的方式 export\import 和 commit方式。这两种方式都需要先运行并创建容器&#xff0c;然后在容器…

python自学之《21天学通Python》(5)

第8章 复杂程序组织 当一个应用程序简单时&#xff0c;将程序代码写入一个文件即可。但随着应用程序或项目复杂度增加时&#xff0c;如果将所有代码都写入同一个文件中时&#xff0c;会出现文件过长或过大&#xff0c;即不方便代码浏览&#xff0c;也不方便代码的管理、使用与维…

人工智能人才缺口暴增,想转行的你赶紧把Python学起来...

当前AI人才极度紧缺&#xff0c;据《中国ICT人才生态白皮书》研究分析&#xff0c;到2018年底&#xff0c;我国人工智能人才缺口将突破100万&#xff0c;到2020年&#xff0c;这一数字将攀升到226万。 在过去的几年中&#xff0c;Python已经成为现代软件开发&#xff0c;基础设…