【项目组件】第三方库——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);}

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

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

相关文章

超越传统:探索ONLYOFFICE的革命性办公新纪元

目录 &#x1f341;引言 &#x1f341;一、ONLYOFFICE产品简介 &#xff08;一&#xff09;、介绍 &#xff08;二&#xff09;、基本功能简介 &#x1f341;二、核心功能具体介绍 1、编辑操作 2、文本与段落&#xff1a; 3、样式与图形&#xff1a; 4、表格与图表&…

python解析网页上的json数据落地到EXCEL

安装必要的库 import requests import pandas as pd import os import sys import io import urllib3 import json测试数据 网页上的数据结构如下 {"success": true,"code": "CIFM_0000","encode": null,"message": &quo…

115页PDF | 埃森哲_XX集团信息化能力成熟度评估及能力提升方案(限免下载)

一、前言 这份报告是埃森哲_XX集团信息化能力成熟度评估及能力提升方案&#xff0c;报告首先分析了集团的战略规划&#xff0c;包括调整优化期、转型升级期和跨越发展期的目标&#xff0c;然后识别了集团面临的内部挑战和外部压力&#xff0c;如管控体系不完善、业务板块多样化…

PostgreSQL中表的数据量很大且索引过大时怎么办

在PostgreSQL中&#xff0c;当表的数据量很大且索引过大时&#xff0c;可能会导致性能问题。以下是一些优化索引和表数据的方法&#xff1a; 1. 评估和删除不必要的索引 识别未使用的索引&#xff1a;使用pg_stat_user_indexes和pg_index系统视图来查找未被使用的索引&#x…

智谱AI清影升级:引领AI视频进入音效新时代

前几天智谱推出了新清影,该版本支持4k、60帧超高清画质、任意尺寸&#xff0c;并且自带音效的10秒视频,让ai生视频告别了"哑巴时代"。 智谱AI视频腾空出世&#xff0c;可灵遭遇强劲挑战&#xff01;究竟谁是行业翘楚&#xff1f;(附测评案例)之前智谱出世那时体验了一…

7. djang5 官网 polls app (关于管理员站点)

本教程从 第 6 个教程 结束的地方开始。我们将继续开发 web-poll 应用程序&#xff0c;并专注于自定义 Django 自动生成的管理网站&#xff0c;这一点我们在 第 2 个教程 中首次进行了探索。 自定义管理表单 在 polls/admin.py 中修改&#xff1a; from django.contrib impo…

Elasticsearch基本概念及使用

Elasticsearch 是一个开源的、分布式的全文搜索和分析引擎&#xff0c;基于 Apache Lucene 构建。它提供了快速的搜索能力&#xff0c;支持大规模的数据分析&#xff0c;广泛应用于日志分析、全文搜索、监控系统和商业智能等领域。ES操作指令是基于restAPI构建&#xff0c;也就…

无插件H5播放器EasyPlayer.js网页web无插件播放器vue和react详细介绍

EasyPlayer.js H5播放器&#xff0c;是一款能够同时支持HTTP、HTTP-FLV、HLS&#xff08;m3u8&#xff09;、WS、WEBRTC、FMP4视频直播与视频点播等多种协议&#xff0c;支持H.264、H.265、AAC、G711A、Mp3等多种音视频编码格式&#xff0c;支持MSE、WASM、WebCodec等多种解码方…

Ubuntu 的 ROS 操作系统安装与测试

引言 机器人操作系统&#xff08;ROS, Robot Operating System&#xff09;是一个用于开发机器人应用的开源框架&#xff0c;它提供了一系列功能丰富的库和工具&#xff0c;能够帮助开发者构建和控制机器人。 当前&#xff0c;ROS1的最新版本为Noetic Ninjemys&#xff0c;专为…

入门网络安全工程师要学习哪些内容(详细教程)

&#x1f91f; 基于入门网络安全/黑客打造的&#xff1a;&#x1f449;黑客&网络安全入门&进阶学习资源包 大家都知道网络安全行业很火&#xff0c;这个行业因为国家政策趋势正在大力发展&#xff0c;大有可为!但很多人对网络安全工程师还是不了解&#xff0c;不知道网…

IP数据云 识别和分析tor、proxy等各类型代理

在网络上使用代理&#xff08;tor、proxy、relay等&#xff09;进行访问的目的是为了规避网络的限制、隐藏真实身份或进行其他的不正当行为。 对代理进行识别和分析可以防止恶意攻击、监控和防御僵尸网络和提高防火墙效率等&#xff0c;同时也可以对用户行为进行分析&#xff…

【C#设计模式(10)——装饰器模式(Decorator Pattern)】

前言 装饰器模式可以在运行时为对象添加额外的功&#xff0c;而无需修改原始对象的代码。这种方式比继承更加灵活。 代码 //蛋糕类&#xff08;抽象类&#xff09; public abstract class Cake {public abstract void Create(); } //奶油蛋糕类 public class CreamCake : Cak…

【论文速读】| 注意力是实现基于大语言模型的代码漏洞定位的关键

基本信息 原文标题&#xff1a;Attention Is All You Need for LLM-based Code Vulnerability Localization 原文作者&#xff1a;Yue Li, Xiao Li, Hao Wu, Yue Zhang, Xiuzhen Cheng, Sheng Zhong, Fengyuan Xu 作者单位&#xff1a;National Key Laboratory for Novel So…

python包管理工具pip和conda的使用对比

python包管理工具pip和conda的使用对比 总述1. pip使用2. conda注意虚拟环境之间的嵌套&#xff0c;这个会导致安装包后看不到包&#xff0c;实际是安装到了base环境里 未完待续 总述 pip相对于conda,对应包的依赖关系管理不强&#xff0c;坏处是容易造成包冲突&#xff0c;好…

Diff 算法的误判

起源&#xff1a; for循环的:key的值使用index绑定&#xff0c;当循环列表条目变化更新&#xff0c;导致虚拟 DOM Diff 算法认为原有项被替换&#xff0c;而不是更新。 // vue2写法 错误例子 <template><div><button click"addItem">添加项目<…

Python与其他语言比较·练习题 --- 《跟着小王学Python》

Python与其他语言比较练习题 — 《跟着小王学Python》 《跟着小王学Python》 是一套精心设计的Python学习教程&#xff0c;适合各个层次的学习者。本教程从基础语法入手&#xff0c;逐步深入到高级应用&#xff0c;以实例驱动的方式&#xff0c;帮助学习者逐步掌握Python的核心…

如何修改npm包

前言 开发中遇到一个问题&#xff0c;配置 Element Plus 自定义主题时&#xff0c;添加了 ElementPlusResolver({ importStyle: "sass" }) 后&#xff0c;控制台出现报错&#xff0c;这是因为 Dart Sass 2.0 不再支持使用 !global 来声明新变量&#xff0c;虽然当前…

[CKS] K8S RuntimeClass SetUp

最近准备花一周的时间准备CKS考试&#xff0c;在准备考试中发现有一个题目关于RuntimeClass创建和挂载的题目。 ​ 专栏其他文章: [CKS] Create/Read/Mount a Secret in K8S-CSDN博客[CKS] Audit Log Policy-CSDN博客 -[CKS] 利用falco进行容器日志捕捉和安全监控-CSDN博客[CKS…

第三百二十五节 Java线程教程 - Java Fork/Join框架

Java线程教程 - Java Fork/Join框架 fork/join框架通过利用机器上的多个处理器或多个内核来解决问题。 该框架有助于解决涉及并行性的问题。 fork/join框架创建一个线程池来执行子任务。 当线程在子任务上等待完成时&#xff0c;框架使用该线程来执行其他线程的其他未决子任…

Vue3集成搜索引擎智能提示API

需求&#xff1a; 如何在项目中实现像百度搜索框一样的智能提示效果&#xff0c;如下图所示&#xff1a; 相关知识&#xff1a; 下面是各厂商提供的免费API 厂商请求百度http://suggestion.baidu.com/su?wd中国&cbwindow.baidu.sug必应http://api.bing.com/qsonhs.as…