【项目组件】第三方库——websocketpp

目录

第三方协议:websocket

websocket简介

websocket特点 

websocket协议切换

 websocket协议格式段

websocketpp库介绍

endpoint

server

 connection 

websocketpp库搭建服务器流程 

基本框架实现 

业务处理回调函数的实现

http_callback 

open_callback

close_callback

message_callback 


第三方协议:websocket

websocket简介

为什么要有websocket? 

websocket协议是应用层协议之一,既然这种协议会出现,那么也就意味着它是在某些特殊场景下弥补了一些其他应用层协议的缺点。

所以关于为什么要有websocket协议,我们首先要聊聊最常见的协议http/https,它们没有解决的一些问题!

我们都知道,http协议是如何建立起连接的呢?

  • http协议规定,首先由客户端向服务器发送一个http请求, 服务器收到后,根据客户端的http请求,返回给客户端一个http应答!
  • 补充:来回一次报文发送,http双方通信连接就会被关闭,这也意味着http协议在通常情况下是无连接或短连接的协议!

http协议上述建立连接的方式带来了一个问题:必须由客户端主动向服务器发送请求报文,服务器无法主动给客户端发送消息!这也就意味着http协议是一个单向通信的协议

而网页的即时聊天或者像我们接下来做的项目“五子棋游戏”这样的程序都是非常依赖服务器给客户端发送消息的!

若我们想要用http协议来完成这个功能,那么只能是客户端等待一段时间之后,主动向服务器询问是否有消息。也就是基于轮询的策略。

轮询策略带来了几个弊端:

  • 消息不够实时:客户端是按一定的时间间隔向服务器发送轮询的,那么从消息准备就绪到服务器收到轮询之间肯定有一定的时间间隔。这样的话就影响了整体的效率!
  • 成本过高:轮询意味着客户端需要不断发送请求到网络中,而大部分的请求都是无意义的,即服务器还未准备就绪的。

基于上述的种种,最终产生了websocket协议,很明显,websocket协议是为了解决http协议的弊端,这两种协议具有强相关。所以后面我们说websocket协议通常是由http协议切换过来的!


websocket特点 

websocket协议与http协议相比,主要具有以下特点:

  • websocket协议支持双向通信,即服务器可以主动给客户端发送消息,客户端也可以主动给服务器发送消息
  • websocket协议是一种长连接协议,通常与TCP协议相同,websocket会为通信双方建立长时间的连接,使得通信双方通信时不再需要频繁建立连接。这降低了延迟,提高了实时性,使得数据可以更快地传输到客户端。

websocket协议切换

websocket协议是由http协议切换过来的,整体切换如下图:

整体协议切换一共分为三步:

  1. 第一步:TCP三次握手建立连接         
  2. 第二步:通过http报文由http协议切换到websocket协议
  3. 第三步:websocket协议格式通信

 TCP三次握手建立连接 

不管是websocket协议还是http协议,都是基于TCP协议的应用层协议。

所以对于websocket协议来说,它需要为通信双方建立一个长连接,最好的方式就是TCP的方式


http协议切换至websocket协议

协议的切换,是通过http报文的方式实现的。即客户端向服务器发送http请求,服务器给客户端http应答,之后进行协议切换。

但不同的是,协议切换的请求和应答的格式不再与普通http报文格式相同

协议切换请求报文的格式:

  • "GET /ws HTTP/1.1":该行表示GET方法,ws是websocket的缩写
  • "Connection:Upgrade":该行表示客户端希望升级当前的HTTP连接到一个新的协议
  • "Upgrade:WebSocket":该行表示升级的协议名称为WebSocket
  • "Sec-WebSocket-Version:xxx":该行表示升级的协议版本为xxx
  • "Sec-WebSocket-Key:xxx":是一个由客户端(通常是浏览器)随机生成的Base64编码值。这个值在WebSocket握手请求中发送给服务器,用于确保客户端和服务器之间的连接是安全的,并且不是由恶意软件或未授权的第三方建立的。

协议切换应答报文的格式:

  • "HTTP/1.1 101 xxx":其中101是响应状态码,即告诉客户端支持WebSocket协议
  • "Connection:Upgrade":与请求报文对应字段含义相同
  • "Upgrade:WebSocket":与请求报文对应字段含义相同
  • "Sec-WebSocket-Accept":Sec-WebSocket-Accept是服务器在WebSocket握手过程中,根据客户端发送的Sec-WebSocket-Key字段,通过一定的算法计算并返回给客户端的一个响应头字段。它用于验证服务器是否理解并接受客户端发起的WebSocket连接请求,同时也作为WebSocket连接安全性的一个基本保障。

接下来的内容,就是对websocket协议格式段进行介绍!


 websocket协议格式段

websocket协议格式段主要如下图:

FIN(1bit)

  • WebSocket支持将长消息切割成若干帧发送,切分后,前边的帧的FIN字段均为0,最后一个帧的FIN为1。
  • 当消息没有分段时,FIN标志位为1。

RSV1-3(各1bit)

  • 保留位,一般情况下为全0。
  • 当客户端、服务端协商采用WebSocket扩展时,这三个标志位可以非0,且值的含义由扩展进行定义。
  • 如果出现非0值但并未采用WebSocket扩展,连接会出错。

Opcode(4bit)

主要用于指定帧类型,可以指定的帧类型有以下几种:

  • %x0:表示一个延续帧。当Opcode为0时,表示本次数据传输采用了数据分片,当前收到的数据帧为其中一个数据分片。
  • %x1:表示这是一个文本帧。
  • %x2:表示这是一个二进制帧。
  • %x3-7:保留的操作代码,用于后续定义的非控制帧。
  • %x8:表示连接断开。
  • %x9:表示这是一个ping操作。
  • %xA:表示这是一个pong操作。
  • %xB-F:保留的操作代码,用于后续定义的控制帧。

注意:尽管帧类型有很多,但我们经常用的主要是文本帧与二进制帧! 


Payload数据 

  • 实际的有效数据载荷部分。
  • 如果通信双方约定使用了WebSocket扩展,则扩展数据也存放于此,并声明扩展长度。
  • 如果没有约定使用,则扩展数据为0字节。

Payload长度 

Payload长度记录的是Payload数据的长度!单位字节

在协议格式中,有四个Payload长度,它们对应着三种不同的场景:

  • 7bitsPayload长度若<126,那么该Payload长度表示的就是有效载荷的长度(0-126字节)
  • 7bitsPayload长度若=126,那么后两个字节(16bitsPayload长度)表示的就是有效载荷的长度(0-65535字节)
  • 7bitsPayload长度若=126,那么后八个字节(16+32+16bitsPayload长度)表示的就是有效载荷的长度(0-2^64-1字节)

Mask(1bit)与Mask-Key(可选) 

Mask表示Payload数据是否被编码,若为1则必有Mask-Key,⽤于解码Payload数据。仅客户端发送给服务端的消息需要设置。

  • 若Mask标志位为1,那么Mask-Key(4bits)一定被设置
  • 若Mask标志位为0,那么Mask-Key未被设置

Mask-Key:

  • 当Mask为1时存在,长度为4字节
  • 解码规则:DECODED[i] = ENCODED[i] ^ MASK[i % 4] 

websocketpp库介绍

WebSocketpp是⼀个跨平台的开源(BSD许可证)头部专⽤C++库,它实现了RFC6455(WebSocket 协议)和RFC7692(WebSocketCompression Extensions)。它允许将WebSocket客户端和服务器功能集成到C++程序中。在最常见的配置中,全功能⽹络I/O由Asio⽹络库提供。

如下为它的基本定义,混个眼熟就好~,后续用了自然就理解了

webscoketpp库


endpoint

 endpoint中提供了一些供我们使用的方法,并且endpoint就是服务器和客户端建立连接时的一个端点。endpoint屏蔽了底层网络通信的细节,依赖于boost库中的Asio对底层网络通信的具体实现

具体来说,endpoint类提供了如下类型的接口:

  • 日志相关接口
  • 回调函数相关接口
  • 通信连接相关接口
  • 其他服务器搭建的接口

日志相关接口 

设置日志输出等级:

void set_access_channels(log::level channels);   /*设置⽇志打印等级*/

 输出等级分为如下:

注意:websocketpp日志输出较为繁杂,后续我们直接设置为none,表示不输出日志即可!

其他日志接口由于不使用,不再过多介绍!


回调函数相关接口 

websocketpp的回调思想:针对特定的事件可以进行设置它的处理函数指针。

websocketpp搭建了服务器之后,给不同的事件设置了不同的处理函数指针,这些指针可以指向指定的函数,当服务器收到了指定的数据,触发了指定的事件后就会通过函数指针去调用这些函数,这时候,我们程序员就可以编写一些业务处理函数,将其设置为对应事件的业务处理函数!

例如:五子棋游戏中,当一名用户想进入到某个游戏房间时,该用户会向服务器发送websocket连接请求,websocket握手连接建立成功,该用户进入了房间的消息应该转发给房间内的所有成员!对于这种情况,我们修改握手成功的回调即可!

websocketpp提供了以下事件的回调:

  • set_open_handler:设置websocket协议握手成功的回调函数
  • set_close_handler:设置websocket连接断开的回调函数
  • set_message_handler:设置websocket消息处理函数
  • set_http_handler:设置http请求的处理函数

通信连接相关接口 

send:给客户端发送消息

close:关闭连接

get_con_from_hdl:通过connection_hdl获取对应的connection_ptr

  • connection_hdl:是一个用于引用和操作WebSocket连接的句柄,而不是连接实例本身。它是WebSocket++库提供的一种机制,允许用户在不直接访问连接实例的情况下与连接进行交互。
  • connection_ptr:具体的连接对象,是一个智能指针类型,当连接被关闭时会自动释放该连接。除此之外,也能通过该类型,直接访问连接的执行方法

其他服务器搭建接口 

init_asio:初始化asio框架,websocketpp网络通信底层依赖的就是这个框架

set_reuse_addr:设置是否启动地址重用

listen:设置绑定监听套接字

run:启动服务器

set_timer:设置定时任务


server

server继承自endpoint,而它自己的接口仅有一个start_accept

start_accept:初始化并启动服务端监听连接的accept事件处理


 connection 

connection是连接管理类,它是对asio中的底层连接进行再封装


以上就是websocketpp库中的基本介绍! 


websocketpp库搭建服务器流程 

使用websocketpp库搭建一个服务器最主要的逻辑如下:

  1. 实例化server对象
  2. 设置日志输出等级
  3. 初始化asio框架中的调度器
  4. 设置业务处理回调函数(具体业务处理的函数由我们自己实现)
  5. 设置服务器监听端口
  6. 开始获取新建连接
  7. 启动服务器

基本框架实现 


搭建服务器的前置准备

首先是包含websocketpp服务器对应的头文件和asio框架的头文件:

#include <websocketpp/config/asio_no_tls.hpp> //asio框架头文件
#include <websocketpp/server.hpp> //

注意:asio框架头文件我们采用asio_no_tls.hpp,不采用asio.hpp。两者的区别是asio_no_tls.hpp不支持TLS功能,TLS是一种用于在两个通信应用程序之间提供保密性和数据完整性的协议。


使用websocketpp中的server实例化对象时,需要传入一个底层网络通信模板参数。我们采用的时asio作为模板参数传入,同时为了使得代码简短,可以对它进行typedef

typedef websocketpp::server<websocketpp::config::asio> wsserver_t;

实例化server对象 

//1、实例化server对象
wsserver_t svr;

设置日志输出等级

由于websocketpp自带的日志输出内容非常多,不便于观察,我们不使用它的日志,把日志输出等级设置为none即可

//2、设置日志输出等级
svr.set_access_channels(websocketpp::log::alevel::none);

初始化asio框架中的调度器

//3、初始化asio框架
svr.init_asio();

设置业务处理回调函数

业务处理回调函数一共有4个 

  • set_open_handler:设置websocket协议握手成功的回调函数
  • set_close_handler:设置websocket连接断开的回调函数
  • set_message_handler:设置websocket消息处理函数
  • set_http_handler:设置http请求的处理函数

这四个函数的函数原型如下:

typedef lib::function<void(connection_hdl)> open_handler;
typedef lib::function<void(connection_hdl)> close_handler;
typedef lib::function<void(connection_hdl)> http_handler;
typedef lib::function<void(connection_hdl, message_ptr)> message_handler;

void set_open_handler(open_handler h);       /*websocket握⼿成功回调处理函数*/
void set_close_handler(close_handler h);     /*websocket连接关闭回调处理函数*/
void set_message_handler(message_handler h); /*websocket消息回调处理函数*/
void set_http_handler(http_handler h);       /*http请求回调处理函数*/

为了后续操作方便,我们回调函数的参数中传入一个server对象,使用bind把这个server对象绑定回调函数,生成一个新的可调用对象传入给这四个函数的参数即可!

void http_callback(wsserver_t* svr,websocketpp::connection_hdl)
{}
void open_callback(wsserver_t* svr,websocketpp::connection_hdl)
{}
void close_callback(wsserver_t* svr,websocketpp::connection_hdl)
{}
void message_callback(wsserver_t* svr,websocketpp::connection_hdl hdl,
                    wsserver_t::message_ptr msg)
{}


//4、设置业务处理回调函数
svr.set_close_handler(std::bind(close_callback,&svr,std::placeholders::_1));
svr.set_open_handler(std::bind(open_callback,&svr,std::placeholders::_1));
svr.set_http_handler(std::bind(http_callback,&svr,std::placeholders::_1));
svr.set_message_handler(std::bind(message_callback,&svr,std::placeholders::_1,std::placeholders::_2));

 设置服务器监听端口

//5、设置服务器监听端口
svr.listen(8888);

 开始获取新建连接与启动服务器

//6、开始获取新建连接
svr.start_accept();
svr.run();

业务处理回调函数的实现

通过上述的几个步骤,我们的服务器的框架已经被搭建好了,接下来处理4个回调方法中的实现即可


http_callback 

 对于http_callback,我们要实现的是给客户端返回一个Hello World的页面

大体上,一共分为两步:

  1. 处理来自客户端的http请求
  2. 构建并发送http应答给客户端

1、处理来自客户端的http请求 

http请求中,最为关键的几个要素:请求方法、请求正文、uri

接下来处理http请求就是我们把这几个关键要素获取下来,并打印!

我们首先获取http请求中的body,在websocketpp库中,connection类提供了获取body的这个方法!

接下来的问题是如何获取connection类对象呢?

实际上,websocketpp库中endpoint类提供了一个方法get_con_from_hdl,即通过一个connection_hdl对象获取一个connection_ptr对象,connection_ptr指向的内容就是connection对象

在websocketpp库中,http命名空间下的parser命名空间下的request类提供了获取请求方法与uri的方法

接下来的问题是如何获取request类对象?

实际上,connection对象中提供了get_request方法用于获取一个request对象


2、构建并发送http应答给客户端

构建http应答一共经历如下几个步骤:

  1. 构建应答正文(Hello World 页面)
  2. 设置应答正文
  3. 添加Content-Type为"text/html"
  4. 设置状态码为ok

其中,我们可以使用string类型构建应答正文 。剩余的方法connection类中都提供了


代码

void http_callback(wsserver_t* svr,websocketpp::connection_hdl hdl)
{
    //1、处理http请求
    wsserver_t::connection_ptr conn = svr->get_con_from_hdl(hdl);
    std::cout << "body:" << conn->get_request_body() << std::endl;
    websocketpp::http::parser::request req = conn->get_request();
    std::cout << "method:" << req.get_method() << std::endl;
    std::cout << "uri:" << req.get_uri() << std::endl;

    //2、构建并发送http应答给客户端
    std::string body = "<html><body><h1>Hello World</h1></body></html>";
    conn->set_body(body);
    conn->append_header("Content-Type","text/html");
    conn->set_status(websocketpp::http::status_code::ok);
}

open_callback

对于该回调,无其他特殊需求,直接打印一行用于观察即可!

void open_callback(wsserver_t* svr,websocketpp::connection_hdl)
{
    std::cout << "websocket握手成功!" << std::endl;
}

close_callback

与open_callback同理

void close_callback(wsserver_t* svr,websocketpp::connection_hdl)
{
    std::cout << "连接关闭!" << std::endl;
}

message_callback 

message_callback被回调时,一定是服务器收到了来自客户端的websocket格式的消息。也就是message_callback的msg参数

为方便测试,我们实现的是服务器把客户端发来的消息原封不动返回

主要完成两个工作:

  • 构建回复消息
  • 发送消息

构建回复消息我们可以采用message类中的get_payload接口

发送消息我们使用connection对象中的send接口

send接口需要传入一个字符串与帧格式

帧格式在websocketpp::frame::opcode中,我们采用的是为text(文本)帧,缺省参数也为文本帧!

void message_callback(wsserver_t* svr,websocketpp::connection_hdl hdl,
                    wsserver_t::message_ptr msg)
{
    wsserver_t::connection_ptr conn = svr->get_con_from_hdl(hdl);
    std::cout << "client say: " << msg->get_payload() << std::endl;
    std::string rep = "server say: " + msg->get_payload();
    conn->send(rep);
    
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【手撕 Spring】 -- Bean 的创建以及获取

&#x1f308;手写简化版 Spring 框架&#xff1a;通过构建一个精简版的 Spring 框架&#xff0c;深入理解 Spring 的核心机制&#xff0c;掌握其设计思想&#xff0c;进一步提升编程能力 &#x1f308;项目代码地址&#xff1a;https://github.com/YYYUUU42/mini-Spring 如果该…

Jdbc学习笔记(四)--PreparedStatement对象、sql攻击(安全问题)

目录 &#xff08;一&#xff09;使用PreparedStatement对象的原因&#xff1a; 使用Statement对象编写sql语句会遇到的问题 ​编辑 &#xff08;二&#xff09;sql攻击 1.什么是sql攻击 2.演示sql攻击 &#xff08;三&#xff09;防止SQL攻击 1.PreparedStatement是什么 …

Jmeter中的定时器(二)

5--JSR223 Timmer 功能特点 自定义延迟逻辑&#xff1a;使用脚本语言动态计算请求之间的延迟时间。灵活控制&#xff1a;可以根据测试数据和条件动态调整延迟时间。支持多种脚本语言&#xff1a;支持 Groovy、JavaScript、BeanShell 等多种脚本语言。 支持的脚本语言 Groov…

【Istio】Istio原理

第一章 Istio原理 一、服务网格(servicemesh)1、六个时代2、服务网格定义及优缺点二、Istio1、Istio定义2、Istio安装3、Istio架构1.5版本之前1.5版本之后4、bookinfo案例架构部署5、CRD一、服务网格(servicemesh) 微服务:架构风格,职责单一,api通信 服务网格:微服务时代的…

4.远程访问及控制

SSH 简介&#xff1a; SSH&#xff08;Secure Shell&#xff09;协议是一种安全通道协议&#xff0c;对通信数据进行了加密处理&#xff0c;用于远程管理。 OpenSSH简介 OpenSSH 服务名称&#xff1a;sshd 服务端主程序&#xff1a;/usr/sbin/sshd 服务端配置文件&#xf…

精通rust宏系列教程-入门篇

Rust最令人敬畏和强大的特性之一是它使用和创建宏的能力。不幸的是&#xff0c;用于创建宏的语法可能相当令人生畏&#xff0c;并且对于新开发人员来说&#xff0c;这些示例可能会令人不知所措。我向你保证Rust宏非常容易理解&#xff0c;本文将为你介绍如何创建自己的宏。 什么…

设计模式之装饰器模式(SSO单点登录功能扩展,增加拦截用户访问方法范围场景)

前言&#xff1a; 两个本想描述一样的意思的词&#xff0c;只因一字只差就让人觉得一个是好牛&#xff0c;一个好搞笑。往往我们去开发编程写代码时也经常将一些不恰当的用法用于业务需求实现中&#xff0c;但却不能意识到。一方面是由于编码不多缺少较大型项目的实践&#xff…

kubernetes简单入门实战

本章将介绍如何在kubernetes集群中部署一个nginx服务&#xff0c;并且能够对其访问 Namespace Namespace是k8s系统中一个非常重要的资源&#xff0c;它的主要作用是用来实现多套环境的资源隔离或者多租户的资源隔离。 默认情况下&#xff0c;k8s集群中的所有的Pod都是可以相…

webpack5 + vue3 从零配置项目

前言 虽然在实际项目当中很少会从 0 到 1 配置一个项目&#xff0c;毕竟很多重复工作是没有必要的&#xff0c;脚手架将这些重复性的工作进行了整合&#xff0c;方便开发者使用。也正因如此&#xff0c;导致部分开发者过于依赖脚手架&#xff0c;却不清楚其内部的实现流程&…

Linux git-bash配置

参考资料 命令提示符Windows下的Git Bash配置&#xff0c;提升你的终端操作体验WindowsTerminal添加git-bash 目录 一. git-bash配置1.1 解决中文乱码1.2 修改命令提示符 二. WindowsTerminal配置git-bash2.1 添加git-bash到WindowsTerminal2.2 解决删除时窗口闪烁问题 三. VS…

(RK3566驱动开发 - 1).pinctrl和gpio子系统

一.设备树 pinctrl部分可以参考 rockchip 官方的绑定文档 &#xff1a;kernel/Documentation/devicetree/bindings/pinctrl PIN_BANK&#xff1a;引脚所属的组 - 本次例程使用的是 GPIO3_A1 这个引脚&#xff0c;所以所属的组为 3&#xff1b; PIN_BANK_IDX&#xff1a;引脚的…

基于OpenFOAM和深度学习驱动的流体力学计算与应用

在深度学习与流体力学深度融合的背景下&#xff0c;科研边界不断拓展&#xff0c;创新成果层出不穷。从物理模型融合到复杂流动模拟&#xff0c;从数据驱动研究到流场智能分析&#xff0c;深度学习正以前所未有的力量重塑流体力学领域。近期在Nature和Science杂志上发表的深度学…

uniapp设置tabBar高斯模糊并设置tabBar高度占位

1、设置tabBar高斯模糊 2、设置tabBar高度占位 &#xff08;1&#xff09;需要先在App.vue中获取一下 uni.getSystemInfoSync().windowBottom; //返回值是tabBar的高度&#xff08;2&#xff09;在全局样式文件/uview-ui/libs/css/style.vue.scss中定义一个全局样式 3、在需…

嵌入式Linux输入系统应用编程学习总结

嵌入式Linux输入系统应用编程学习总结 目录 嵌入式Linux输入系统应用编程学习总结1. 嵌入式Linux输入系统介绍2. Linux设备输入数据的几个结构体2.1 内核中表示一个输入设备的结构体2.2 APP从输入设备获取的数据类型结构体 3. 查看LCD设备信息和输入数据3.1 查看设备信息3.2 使…

力扣=Mysql-3322- 英超积分榜排名 III(中等)

一、题目来源 3322. 英超积分榜排名 III - 力扣&#xff08;LeetCode&#xff09; 二、数据表结构 表&#xff1a;SeasonStats --------------------------- | Column Name | Type | --------------------------- | season_id | int | | team_id …

HTML之列表学习记录

练习题&#xff1a; 图所示为一个问卷调查网页&#xff0c;请制作出来。要求&#xff1a;大标题用h1标签&#xff1b;小题目用h3标签&#xff1b;前两个问题使用有序列表&#xff1b;最后一个问题使用无序列表。 代码&#xff1a; <!DOCTYPE html> <html> <he…

【ElasticSearch】定位分片不分配

记录遇到的es集群分片不分配的情况--待补全 定位分片不分配的原因 定位分片不分配的原因 在shell客户端执行如下的语句&#xff1a; curl -X GET "http://192.168.0.209:9200/_cat/shards?v&hindex,shard,prirep,state,unassigned.reason"集群中各分片的状态都…

10款PDF合并工具讲解与推荐!!!

在现在的大环境下&#xff0c;PDF文件因其跨平台、格式固定等优势&#xff0c;成为了我们工作和学习中不可或缺的一部分。是最常用的文档格式之一。然而&#xff0c;面对多个PDF文件需要合并成一个的场景&#xff0c;如何选择一款高效、易用的PDF合并工具就显得尤为重要。今天&…

「QT」窗口类 之 QWidget 窗口基类

✨博客主页何曾参静谧的博客&#x1f4cc;文章专栏「QT」QT5程序设计&#x1f4da;全部专栏「Win」Windows程序设计「IDE」集成开发环境「UG/NX」BlockUI集合「C/C」C/C程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「UG/NX」NX定制…

【机器学习】如何配置anaconda环境(无脑版)

马上就要上机器学习的实验&#xff0c;这里想写一下我配置机器学习的anaconda环境的二三事 一、首先&#xff0c;下载安装包&#xff1a; Download Now | Anaconda 二、打开安装包&#xff0c;一直点NEXT进行安装 这里要记住你要下载安装的路径在哪&#xff0c;后续配置环境…