qq怎么上传文件,qq显示不能上传文件夹

转发链接:前言平常在写业务的时候常常会用的到的是GET,POST请求去请求接口,GET相关的接口会比较容易基本不会出错,而对于POST中常用的表单提交,JSON提交也比较容易,但是对于文件上传呢?大家

转发链接:

前言

平常在写业务的时候常常会用的到的是 GET,POST请求去请求接口,GET 相关的接口会比较容易基本不会出错,而对于 POST中常用的 表单提交,JSON提交也比较容易,但是对于文件上传呢?大家可能对这个步骤会比较害怕,因为可能大家对它并不是怎么熟悉,而浏览器Network对它也没有详细的进行记录,因此它成为了我们心中的一根刺,我们老是无法确定,关于文件上传到底是我写的有问题呢?还是后端有问题,当然,我们一般都比较谦虚, 总是会在自己身上找原因,可是往往实事呢?可能就出在后端身上,可能是他接受写的有问题,导致你换了各种请求库去尝试,axios,request,fetch 等等。那么我们如何避免这种情况呢?我们自身要对这一块够熟悉,才能不以猜的方式去写代码。如果你觉得我以上说的你有同感,那么你阅读完这篇文章你将收获自信,你将不会质疑自己,不会以猜的方式去写代码。

本文比较长可能需要花点时间去看,需要有耐心,我采用自顶向下的方式,所有示例会先展现出你熟悉的方式,再一层层往下,先从请求端是怎么发送文件的,再到接收端是怎么解析文件的。

前置知识

什么是 multipart/form-data?

multipart/form-data 最初由 《RFC 1867: Form-based File Upload in HTML》[1]文档提出。

Since file-upload is a feature that will benefit many applications,this proposes an extension to HTML to allow information providers to express file upload requests uniformly,and a MIME compatible representation for file upload responses.

由于文件上传功能将使许多应用程序受益,因此建议对HTML进行扩展,以允许信息提供者统一表达文件上传请求,并提供文件上传响应的MIME兼容表示。

总结就是原先的规范不满足了,我要扩充规范了。

文件上传为什么要用 multipart/form-data?

The encoding type application/x-www-form-urlencoded is inefficient for sending large quantities of binary data or text containing non-ASCII characters. Thus,a new media type,multipart/form-data,is proposed as a way of efficiently sending the values associated with a filled-out form from client to server.

1867文档中也写了为什么要新增一个类型,而不使用旧有的application/x-www-form-urlencoded:因为此类型不适合用于传输大型二进制数据或者包含非ASCII字符的数据。平常我们使用这个类型都是把表单数据使用url编码后传送给后端,二进制文件当然没办法一起编码进去了。所以multipart/form-data就诞生了,专门用于有效的传输文件。

也许你有疑问?那可以用 application/json吗?

其实我认为,无论你用什么都可以传,只不过会要综合考虑一些因素的话,multipart/form-data更好。例如我们知道了文件是以二进制的形式存在,application/json 是以文本形式进行传输,那么某种意义上我们确实可以将文件转成例如文本形式的 Base64 形式。但是呢,你转成这样的形式,后端也需要按照你这样传输的形式,做特殊的解析。并且文本在传输过程中是相比二进制效率低的,那么对于我们动辄几十M几百M的文件来说是速度是更慢的。

以上为什么文件传输要用multipart/form-data 我还可以举个例子,例如你在中国,你想要去美洲,我们的multipart/form-data相当于是选择飞机,而application/json相当于高铁,但是呢?中国和美洲之间没有高铁啊,你执意要坐高铁去,你可以花昂贵的代价(后端额外解析你的文本)坐高铁去美洲,但是你有更加廉价的方式坐飞机(使用multipart/form-data)去美洲(去传输文件)。你图啥?(如果你有钱有时间,抱歉,打扰了,老子给你道歉)

multipart/form-data规范是什么?

qq显示不能上传文件夹,摘自 《RFC 1867: Form-based File Upload in HTML》[2] 6.Example

Content-type: multipart/form-data,boundary=AaB03x--AaB03xcontent-disposition: form-data; name=&34;Joe Blow--AaB03xcontent-disposition: form-data; name=&34;; filename=&34;Content-Type: text/plain... contents of file1.txt ...--AaB03x--

可以简单解释一些,首先是请求类型,然后是一个 boundary (分割符),这个东西是干啥的呢?其实看名字就知道,分隔符,当时分割作用,因为可能有多文件多字段,每个字段文件之间,我们无法准确地去判断这个文件哪里到哪里为截止状态。因此需要有分隔符来进行划分。然后再接下来就是

好了讲完了这些前置知识,我们接下来要进入我们的主题了。面对File,formData,Blob,Base64,ArrayBuffer,到底怎么做?还有文件上传不仅仅是前端的事。服务端也可以文件上传(例如我们利用某云,把静态资源上传到 OSS 对象存储)。服务端和客户端也有各种类型,Buffer,Stream,Base64....头秃,怎么搞?不急,就是因为上传文件不单单是前端的事,所以我将以下上传文件的一方称为请求端,接受文件一方称为接收方。我会以请求端各种上传方式,接收端是怎么解析我们的文件以及我们最终的杀手锏调试工具-wireshark来进行讲解。以下是讲解的大纲,我们先从浏览器端上传文件,再到服务端上传文件,然后我们再来解析文件是如何被解析的。

请求端

浏览端

File

首先我们先写下最简单的一个表单提交方式。

<form action=&34; method=&34;> <input name=&34; type=&34; id=&34;> <input type=&34; value=&34;></form>

我们选择文件后上传,发现后端返回了文件不存在。

不用着急,熟悉的同学可能立马知道是啥原因了。嘘,知道了也听我慢慢叨叨。

我们打开控制台,由于表单提交会进行网页跳转,因此我们勾选preserve log 来进行日志追踪。

我们可以发现其实 FormData 中 file 字段显示的是文件名,并没有将真正的内容进行传输。再看请求头。

发现是请求头和预期不符,也印证了 application/x-www-form-urlencoded 无法进行文件上传。

我们加上请求头,再次请求。

发现文件上传成功,简单的表单上传就是像以上一样简单。但是你得熟记文件上传的格式以及类型。

FormData

formData 的方式我随便写了以下几种方式。

<input type=&34; id=&34;><button id=&34;>上传</button><script src=&39;file&39;file&39;http://localhost:7787/files&39;http://localhost:7787/files&39;POST&39;POST&39;http://localhost:7787/files',true);xhr.onload = function () { console.log(xhr.responseText);};xhr.send(form);}</script>

以上几种方式都是可以的。但是呢,请求库这么多,我随便在 npm 上一搜就有几百个请求相关的库。

因此,掌握请求库的写法并不是我们的目标,目标只有一个还是掌握文件上传的请求头和请求内容。

Blob

因此如果我们遇到 Blob 方式的文件上方式不用害怕,可以用以下两种方式:

1.直接使用 blob 上传

const json = { hello: &34; };const blob = new Blob([JSON.stringify(json,null,2)],{ type: &39; });const form = new FormData();form.append(&39;,blob,&39;);axios.post(&39;,form);

2.使用 File 对象,再进行一次包装(File 兼容性可能会差一些 )

const json = { hello: &34; };const blob = new Blob([JSON.stringify(json,null,2)],{ type: &39; });const file = new File([blob],&39;);form.append(&39;,file);axios.post(&39;,form)

ArrayBuffer

ArrayBuffer 对象用来表示通用的、固定长度的原始二进制数据缓冲区。

虽然它用的比较少,但是他是最贴近文件流的方式了。

在浏览器中,他每个字节以十进制的方式存在。我提前准备了一张图片。

这里需要注意的是 new Blob([typedArray.buffer],{type: &39;}),第一个参数是由一个数组包裹。里面是 typedArray 类型的 buffer。

Base64

const base64 = &39;;const byteCharacters = atob(base64);const byteNumbers = new Array(byteCharacters.length);for (let i = 0; i < byteCharacters.length; i++) { byteNumbers[i] = byteCharacters.charCodeAt(i);}const array = Uint8Array.from(byteNumbers);const blob = new Blob([array],{type: &39;});const form = new FormData();form.append(&39;,blob,&39;);axios.post(&39;,form);

关于 base64 的转化和原理可以看这两篇 base64 原理[4] 和

原来浏览器原生支持JS Base64编码解码[5]

小结

对于浏览器端的文件上传,可以归结出一个套路,所有东西核心思路就是构造出 File 对象。然后观察请求 Content-Type,再看请求体是否有信息缺失。而以上这些二进制数据类型的转化可以看以下表。

图片来源 ([6])

服务端

讲完了浏览器端,现在我们来讲服务器端,和浏览器不同的是,服务端上传有两个难点。

1.浏览器没有原生 formData,也不会想浏览器一样帮我们转成二进制形式。

2.服务端没有可视化的 Network 调试器。

Buffer

Request

首先我们通过最简单的示例来进行演示,然后一步一步深入。相信文档可以查看

/ request-error.jsconst fs = require(&39;);const path = require(&39;);const request = require(&39;);const stream = fs.readFileSync(path.join(__dirname,&39;));request.post({url: &39;, formData: {file: stream, }},(err,res,body) => {console.log(body);})

发现报了一个错误,正像上面所说,浏览器端报错,可以用NetWork。那么服务端怎么办?这个时候我们拿出我们的利器 -- wireshark

我们打开 wireshark (如果没有或者不会的可以查看教程 )

设置配置 tcp.port == 7787,这个是我们后端的端口。

运行上述文件 node request-error.js

我们来找到我们发送的这条http的请求报文。中间那堆乱七八糟的就是我们的文件内容。

POST /files HTTP/1.1host: localhost:7787content-type: multipart/form-data; boundary=--------------------------437240374415content-length: 305Connection: close----------------------------437240374415Content-Disposition: form-data; name=&34;Content-Type: application/octet-stream.PNG....IHDR.............%.V.....PLTE......Ll..... pHYs..........+.....IDAT..c`.......qd.....IEND.B`.----------------------------437240374415--

可以看到上述报文。发现我们的内容请求头 Content-Type: application/octet-stream有错误,我们上传的是图片请求头应该是image/png,并且也少了 filename=&34;。

我们来思考一下,我们刚才用的是fs.readFileSync(path.join(__dirname,&39;)) 这个函数返回的是 Buffer,Buffer是什么样的呢?就是下面的形式,不会包含任何文件相关的信息,只有二进制流。

QQ上传群文件,其实和上传群图片是一样的,下面我就来介绍一下具体的步骤:1、打开手机QQ,找到想上传群文件的一个群聊 2、点击进入该群聊的聊天界面,点击右下角的加号 3、在新弹出的页面中,点击文件 4、去选择文档。

<Buffer 01 02>

所以我想到的是,需要指定文件名以及文件格式,幸好 request 也给我们提供了这个选项。

key: {value:fs.createReadStream(&39;), options: {filename: &39;, contentType: &39;}}

可以指定options,因此正确的代码应该如下(省略不重要的代码)

...request.post({url: &39;, formData: {file: {value: stream, options: {filename: &39;}}, }});

Form-data

我们再深入一些,来看看 request 的源码,他是怎么实现Node端的数据传输的。

打开源码我们很容易地就可以找到关于 formData 这块相关的内容

就是利用form-data,我们先来看看 formData 的方式。

const path = require(&39;);const FormData = require(&39;);const fs = require(&39;);const http = require(&39;);const form = new FormData();form.append(&39;,fs.readFileSync(path.join(__dirname,&39;)),{filename: &39;, contentType: &39;,});const request = http.request({method: &39;, host: &39;, port: &39;, path: &39;, headers: form.getHeaders()});form.pipe(request);request.on(&39;,function(res) {console.log(res.statusCode);});

原生 Node

看完formData,可能感觉这个封装还是太高层了,于是我打算对照规范手动来构造multipart/form-data请求方式来进行讲解。我们再来回顾一下规范。

我模拟上方,我用原生 Node 写出了一个multipart/form-data 请求的方式。

主要分为4个部分

构造请求header

构造内容header

写入内容

写入结束分隔符

const path = require(&39;);const fs = require(&39;);const http = require(&39;);// 定义一个分隔符,要确保唯一性const boundaryKey = &39;;const request = http.request({method: &39;, host: &39;, port: &39;, path: &39;, headers: {&39;: &39; + boundaryKey,// 在请求头上加上分隔符&39;: &39;}});// 写入内容头部request.write(`--${boundaryKey}\r\nContent-Disposition: form-data; name=&34;; filename=&34;\r\nContent-Type: image/jpeg\r\n\r\n`);// 写入内容const fileStream = fs.createReadStream(path.join(__dirname,&39;));fileStream.pipe(request,{ end: false });fileStream.on(&39;,function () {// 写入尾部request.end(&39; + boundaryKey + &39; + &39;);});request.on(&39;,function(res) {console.log(res.statusCode);});

至此,已经实现服务端上传文件的方式。

Stream、Base64

由于这两块就是和Buffer的转化,比较简单,我就不再重复描述了。可以作为留给大家的作业,感兴趣的可以给我这个示例代码仓库贡献这两个示例。

/ base64 to bufferconst b64string = /* whatever */;const buf = Buffer.from(b64string,&39;);

/ stream to bufferfunction streamToBuffer(stream) {return new Promise((resolve,reject) => {const buffers = [];stream.on(&39;,reject);stream.on(&39;,(data) => buffers.push(data))stream.on(&39;,() => resolve(Buffer.concat(buffers))});}

小结

由于服务端没有像浏览器那样 formData 的原生对象,因此服务端核心思路为构造出文件上传的格式(header,filename等),然后写入 buffer 。然后千万别忘了用 wireshark进行验证。

接收端

这一部分是针对 Node 端进行讲解,对于那些 koa-body 等用惯了的同学,可能一样不太清楚整个过程发生了什么?可能唯一比较清楚的是 ctx.request.files ??? 如果ctx.request.files 不存在,就会懵逼了,可能也不太清楚它到底做了什么,文件流又是怎么解析的。

我还是要说到规范...请求端是按照规范来构造请求..那么我们接收端自然是按照规范来解析请求了。

Koa-body

const koaBody = require(&39;);app.use(koaBody({ multipart: true }));

我们来看看最常用的 koa-body,它的使用方式非常简单,短短几行,就能让我们享受到文件上传的简单与快乐(其他源码库一样的思路去寻找问题的本源) 可以带着一个问题去阅读,为什么用了它就能解析出文件?

寻求问题的本源,我们当然要打开 koa-body的源码,koa-body 源码很少只有211行, 很容易地发现它其实是用了一个叫做formidable的库来解析files 的。并且把解析好的files 对象赋值到了 ctx.req.files。(所以说大家不要一味死记 ctx.request.files,注意查看文档,因为今天用 koa-body是 ctx.request.files 明天换个库可能就是 ctx.request.body 了)

因此看完koa-body我们得出的结论是,koa-body的核心方法是formidable

Formidable

那么让我们继续深入,来看看formidable做了什么,我们首先来看它的目录结构。

.├── lib│ ├── file.js│ ├── incoming_form.js│ ├── index.js│ ├── json_parser.js│ ├── multipart_parser.js│ ├── octet_parser.js│ └── querystring_parser.js

看到这个目录,我们大致可以梳理出这样的关系。

index.js|incoming_form.js|type?|1.json_parser2.multipart_parser3.octet_parser4.querystring_parser

由于源码分析比较枯燥。因此我只摘录比较重要的片段。由于我们是分析文件上传,所以我们只需要关心multipart_parser 这个文件。

...MultipartParser.prototype.write = function(buffer) { console.log(buffer);var self = this, i = 0, len = buffer.length, prevIndex = this.index, index = this.index, state = this.state,...

我们将它的 buffer 打印看看.

<Buffer 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 34 36 31 35 39 31 30 38 30 39 34 31 36 32 32 35 31 31 33 33 36 36 36 ... >144<Buffer 89 50 4e 47 0d 0a 1a 0a 00 00 00 0d 49 48 44 52 00 00 00 01 00 00 00 01 01 03 00 00 00 25 db 56 ca 00 00 00 06 50 4c 54 45 00 00 ff 80 80 80 4c 6c bf ... >106<Buffer 0d 0a 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 34 36 31 35 39 31 30 38 30 39 34 31 36 32 32 35 31 31 33 33 36 ... >

我们来看wireshark 抓到的包

我用红色进行了分割标记,对应的就是formidable所分割的片段 ,所以说这个包主要是将大段的 buffer 进行分割,然后循环处理。

这里我还可以补充一下,可能你对以上表非常陌生。左侧是二进制流,每1个代表1个字节,1字节=8位,上面的 2d 其实就是16进制的表示形式,用二进制表示就是 0010 1101,右侧是ascii 码用来可视化,但是 assii 分可显和非可显示。有部分是无法可视的。比如你所看到文件中有需要小点,就是不可见字符。

你可以对照,ascii表对照表[7]来看。

我来总结一下formidable对于文件的处理流程。

原生 Node

好了,我们已经知道了文件处理的流程,那么我们自己来写一个吧。

const fs = require(&39;);const http = require(&39;);const querystring = require(&39;);const server = http.createServer((req,res) => {if (req.url === &34; && req.method.toLowerCase() === &34;) {parseFile(req,res)}})function parseFile(req,res) {req.setEncoding(&34;);let body = &34;;let fileName = &34;;// 边界字符let boundary = req.headers[&39;].split(&39;)[1].replace(&34;,&34;)req.on(&34;,function(chunk) {body += chunk;});req.on(&34;,function() {// 按照分解符切分const list = body.split(boundary);let contentType = &39;;let fileName = &39;;for (let i = 0; i < list.length; i++) {if (list[i].includes(&39;)) {const data = list[i].split(&39;);for (let j = 0; j < data.length; j++) {// 从头部拆分出名字和类型if (data[j].includes(&39;)) {const info = data[j].split(&39;)[1].split(&39;);fileName = info[info.length - 1].split(&39;)[1].replace(/&39;&39;Content-Type&39;:&34;--&34;--&34;binary&34;sucess");});;})}server.listen(7787)

总结

相信有了以上的介绍,你不再对文件上传有所惧怕,对文件上传整个过程都会比较清晰了,还不懂。。。。找我。

再次回顾下我们的重点:

请求端出问题,浏览器端打开 network 查看格式是否正确(请求头,请求体),如果数据不够详细,打开wireshark,对照我们的规范标准,看下格式(请求头,请求体)。

推荐JavaScript经典实例学习资料文章

《》

《》

《》

《》

《》

《》

《》

《》

《》

《》

《》

《》

qq怎么上传文件

《》

《》

《》

《》

《》

《》

《》

《》

《》

《》

《》

《》

1、打开电脑,找到QQ,登陆账号密码并进入。2、进入到QQ界面,点击右下方的“应用管理器”进入。3、进入到应用管理器界面后,在点击“微云”进入。4、进入到微云界面后,点击“上传”进入 。5、然后选择文件,将电脑文件上。

《》

《》

《》

《》

《》

《》

1、首先,我们需要在手机上下载一个QQ,点击打开。2、打开后,我们随便选择一个好友,点开进行聊天,然后选择右下角的那个符号。3、选择文件。4、弹出的界面,我们选择文档。5、选中一个文档,并点击下方的发送。6、显示文。

《》

《》

《》

《》

《》

《》

《》

《》

qq上传群文件的方法如下:工具/原料:iphone 12,iOS14.8,qq8.8.28。1、打开qq,进入群聊天界面,点击右上角的菜单图标。2、找到并点击文件选项。3、点击右上角的双向单箭头,弹出一个小窗口,选择上传文件。点击需要。

《》

《》

《》

《》

《》

1、首先将我们想要传送的文件进行一下压缩,如果直接传文件的话很有可能会丢失某些数据,压缩一下再传会安全很多。2、压缩完毕后,打开QQ,找到想要发送的好友,接着打开与他的聊天界面。3、在聊天界面的上方可以看到有一个。

《》

《》

《》

《》

《》

《》

转发链接:

上一篇 2023年02月23 18:22
下一篇 2023年02月09 07:23

相关推荐

  • m12螺距是多少,M12标准螺距是多少

    内六角平圆头螺钉是螺丝人经常打交道的螺丝之一,也是普通人在日常生活中接触较多的螺丝之一。所以这篇文章就为大家整理了内六角圆头螺钉的规格以及其他的一些知识。以上为GB/T70.2-2008的内六角平圆头

    2023年02月22 262
  • 呼呼影音为什么用不了,西瓜影音播放器怎么用

    据@新浪科技报道,今天(11月25日)暴风影音的官方网站以及其APP都出现了无法正常打开的情况,陪伴了许多80后90后的经典视频播放器就这么挂了吗?暴风影音官网:可能是安装有问题或运行不了软件。目前股

    2023年04月11 234
  • iphone6银色多少钱,iphone11pro银色

    Phone6现在已经非常便宜了,价格早已经跌入几百钱。很多人都买iPhone6当备用机使用,网友也不例外,自己想买一款iPhone6当备用机,看到一款iPhone6成色好,系统好,价格便宜,只要580

    2023年03月23 205
  • 2个g流量多少钱,2G流量等于多少元

    今天跟朋友闲聊,说起来天津的孩子真可怜,开学一个月了,就家长去了一次学校把课本给领回来了,学校一次也没去,有的课程也换老师了,连面都没见着,现在一些主课一般以直播腾讯会议的形式授课,2G流量等于多少元

    2023年03月23 289
  • 网警电话号码是多少

    12321是啥?一、工作职责(1)接收社会各界关于网络不良与垃圾信息的投诉;(2)对投诉所反映的问题进行核查、统计和分析,并报送有关政府部门;(3)监督基础运营商等相关电信企业的网络不良与垃圾信息用户

    2023年02月21 279
  • 显示器怎么用,没有主机的显示器可以投屏吗

    屏幕该怎么录像?我们经常使用到的往往都是手机录屏,在电脑上录制视频相对较少。但对于经常要使用电脑工作的小伙伴来说,屏幕录像那可就是一件经常要做的事了,像是录制课程、会议的演讲,没有主机的显示器可以投屏

    2023年02月10 249
  • 影驰主板怎么样,影驰主板能买吗

    IT之家10月11日消息,影驰现已发布B650EGAMER主板,预计将在近期上市。据介绍,该主板支持AMD新一代7000系列处理器,将同时兼容DDR5内存以及即将推出的PCI-E5.0SSD。据介绍,

    2023年02月06 285
  • 键盘怎么,键盘应该怎样使用

    哈喽,各位小伙伴好呀!今天给大家分享当我们的键盘无法正常使用的情景电脑键盘怎么切换中文输入法的操作:方法一、1.首先打开输入法,此时输入法为英文输入模式,在页面中可以直接输入英文字母;2.然后按下键盘

    2023年02月08 255
  • 抖音怎么开通,抖音出售交易网

    前段时间有人就问我关于抖音的内容,让我讲讲抖音咋玩,抖音出售交易网,奈何抖音我一直没有涨多少粉丝,也没有花更多精力在上面,于是就一直没有写关于这方面的内容。不过不写也不行啊,出版社一直在催呢,所以咱们

    2023年02月09 225
  • 为什么仅限紧急呼叫,手机莫名其妙仅限紧急呼叫

    老人脑出血倒地,家人用手机多次拨打120急救电话,却无法拨出。近日,四川省南充市的何女士反映,6月14日,其母发现父亲晕倒后,用一款魅族品牌手机拨打120电话近半个小时,都无法拨出,手机提示“您的号码

    2023年04月13 238
  • 鼠标怎么调,鼠标正常设置怎么设置

    鼠标正常设置怎么设置,不论你是看再多的比赛,或是研究在精细烟闪道具技巧,但CS:GO终究是一个依靠枪法的游戏。实际上限制萌新升段的原因就是枪法,今天就来介绍下如何调整鼠标设置。鼠标握持方式1、首先,我

    2023年02月09 228
  • 10倍音速是多少公里,中国30倍音速飞行器

    本期川陀太空仅从助推-滑翔弹道上解读这个载具的基本性能。从弹道抛物面上看,助推-滑翔弹道需要有一个助推段,因此我们看到东风17有两个部分构成,第一个助推段,第二个是滑翔飞行器,前端长得像飞机的载具就是

    2023年04月10 284
  • 修苹果白苹果多少钱,苹果7白苹果修要多少钱

    苹果7白苹果修要多少钱,这几天苹果7手机突然开不了机了,充电的时候按电源键能显示充电电池图像,同时按音量键±和电源键能进入白苹果,不能进入系统,黑屏状态。去苹果深圳专卖店去进行售后,换屏不能开机,需要

    2023年03月15 220
关注微信