如何在Docker中构建多阶段PHP镜像 PHP环境优化镜像体积的方法

多阶段构建是优化php docker镜像体积的首选,因为它能将构建时依赖与运行时依赖分离。1. 使用from … as …语法划分构建器和运行时两个阶段;2. 构建器阶段负责安装composer依赖、编译前端资源或pecl扩展,只将必要文件复制到运行时阶段;3. 运行时阶段基于轻量级镜像,仅保留应用运行所需的最小环境;4. 通过copy –from=builder指令精确控制文件复制;5. 镜像瘦身技巧包括选择alpine基础镜像、利用.dockerignore文件、合并并清理run指令、只安装必要依赖、使用明确镜像标签;6. 实际挑战包括构建依赖调试困难、文件复制粒度控制、缓存失效影响构建速度、多架构适配问题,可通过分步构建、路径管理、层缓存优化、docker buildx等策略应对。

如何在Docker中构建多阶段PHP镜像 PHP环境优化镜像体积的方法

Docker中构建精简高效的PHP镜像,核心思路在于“分阶段”处理,也就是利用多阶段构建(Multi-stage build)把构建时的“垃圾”隔离掉,只把运行时真正需要的东西带到最终镜像里。这是一个非常强大的功能,能够显著减小镜像体积,提升部署效率和安全性。当然,除了多阶段,还有一些“抠门”的小技巧,也能让你的镜像瘦身成功。

如何在Docker中构建多阶段PHP镜像 PHP环境优化镜像体积的方法

解决方案

要在Docker中构建多阶段PHP镜像并优化其体积,最直接且有效的方法就是利用Docker的FROM ... AS ...语法,将编译和构建过程与最终运行环境分离。

一个典型的多阶段PHP Dockerfile会包含至少两个阶段:一个“构建器”阶段(builder stage)负责处理Composer依赖、前端资源编译(如果PHP项目有前端部分),以及可能需要的PECL扩展编译等;另一个“运行时”阶段(runtime stage)则基于一个轻量级的PHP基础镜像,只复制构建器阶段产生的必要文件和产物。

立即学习“PHP免费学习笔记(深入)”;

如何在Docker中构建多阶段PHP镜像 PHP环境优化镜像体积的方法

以下是一个简化但能体现核心思想的Dockerfile示例:

# --- 第一阶段:构建器 (Builder Stage) ---# 使用一个包含Node.js和Composer的镜像,或者一个PHP CLI镜像来处理依赖FROM php:8.2-cli-alpine AS builder# 安装Git,如果你的Composer依赖需要从Git仓库拉取RUN apk add --no-cache git# 设置工作目录WORKDIR /app# 复制composer.json和composer.lock,优先处理依赖,利用Docker层缓存COPY composer.json composer.lock ./# 安装Composer依赖,只安装生产环境依赖,并优化自动加载# 注意:这里如果你的应用需要PECL扩展,比如redis,可以在这里安装# RUN docker-php-ext-install pdo_mysql opcache #     && pecl install redis #     && docker-php-ext-enable redis# 实际项目中,这些扩展通常在FPM镜像中安装,但如果构建时需要,可以在此阶段安装并复制.so文件RUN composer install --no-dev --optimize-autoloader --no-scripts# 复制整个应用代码COPY . .# 如果你的PHP项目包含前端构建(如Vue/React),可以在此阶段进行NPM/Yarn构建# FROM node:lts-alpine AS frontend_builder# WORKDIR /app# COPY package.json package-lock.json ./# RUN npm install# COPY . .# RUN npm run build# --- 第二阶段:运行时 (Runtime Stage) ---# 使用一个轻量级的PHP-FPM基础镜像,例如Alpine版本FROM php:8.2-fpm-alpine# 安装运行时可能需要的系统依赖,比如GD库依赖、MySQL客户端等RUN apk add --no-cache libpng libjpeg-turbo freetype     libpq-dev     && docker-php-ext-configure gd --with-freetype --with-jpeg     && docker-php-ext-install -j$(nproc) gd pdo_mysql pdo_pgsql opcache# 如果需要Redis扩展,通常在运行时阶段安装,因为它是个运行时依赖RUN pecl install redis     && docker-php-ext-enable redis# 设置工作目录为Nginx/Apache通常指向的应用根目录WORKDIR /var/www/html# 从构建器阶段复制Composer安装的依赖COPY --from=builder /app/vendor ./vendor# 从构建器阶段复制应用代码(不包含构建时的临时文件和开发依赖)COPY --from=builder /app ./# 如果有前端构建阶段,从前端构建器复制编译后的前端资源# COPY --from=frontend_builder /app/dist ./public/dist# 清理不必要的缓存和文件,进一步减小镜像RUN rm -rf /tmp/* /var/cache/apk/* /usr/share/doc/*# 暴露PHP-FPM端口EXPOSE 9000# 启动PHP-FPMCMD ["php-fpm"]

这个Dockerfile通过AS builderCOPY --from=builder实现了阶段分离。构建器阶段负责所有繁重的编译和依赖安装工作,最终运行时阶段只接收处理好的“成品”。

如何在Docker中构建多阶段PHP镜像 PHP环境优化镜像体积的方法

为什么多阶段构建是优化PHP Docker镜像体积的首选?

说实话,我个人觉得多阶段构建简直是Docker镜像优化的“杀手锏”,尤其对于PHP应用这种经常需要Composer、前端编译甚至PECL扩展编译的场景。它之所以成为首选,核心在于它彻底解决了“构建时依赖”和“运行时依赖”的混淆问题。

你想想看,一个典型的PHP项目,在本地开发时,你需要composer install来下载一大堆开发依赖(比如PHPUnit、PHP_CodeSniffer),可能还需要npm installnpm run build来编译前端资源。这些工具链和它们的产物,比如node_modules目录,或者编译C扩展时需要的gcc、各种*-dev包,它们在你的应用最终运行时,根本就是多余的,甚至可以说是一种“负担”。它们占据了宝贵的镜像空间,增加了镜像的下载时间,更糟糕的是,它们还扩大了潜在的攻击面——毕竟,你运行的镜像里东西越少,被利用的漏洞可能性就越低。

多阶段构建就像是把一个复杂的生产线分成了几个独立的工坊。第一个工坊(builder stage)里堆满了各种工具和原材料,你可以在这里尽情地编译、打包、测试。一旦成品出来了,你只需要把这个“成品”小心翼翼地拿出来,放到第二个、干净整洁的工坊(runtime stage)里,这里只有运行产品所必需的最小化环境。那些在第一个工坊里用过的工具、废弃的材料,通通都不会被带到最终的产品里。

具体到PHP项目:

Composer依赖管理: composer install --no-dev虽然可以减少开发依赖,但vendor目录本身还是挺大的。更重要的是,运行composer命令本身需要PHP CLI环境、Git(如果依赖私有库)、甚至一些系统级别的压缩工具。这些在最终的FPM运行时并不需要。多阶段构建可以把Composer的安装过程放在一个专门的CLI镜像里,然后只把vendor目录复制到FPM镜像中。前端资源编译: 这是个大头!node_modules目录动辄几百兆甚至上G,Webpack、Babel等构建工具也都是重量级的。如果你把这些都打包进一个镜像,那镜像体积会非常惊人。多阶段构建允许你在一个Node.js镜像里完成前端编译,然后只把最终生成的distpublic/build目录复制到PHP-FPM镜像中,完美避开所有前端构建的“垃圾”。PECL扩展编译: 某些PHP扩展(如Redis、Imagick等)需要通过pecl installdocker-php-ext-install来编译安装。这个过程往往需要安装build-essentiallib*dev等开发库和编译器。这些编译工具在扩展安装完成后就没用了。在构建阶段安装并编译好扩展,然后只把编译好的.so文件或者直接使用一个预装了这些扩展的轻量级FPM镜像,是更明智的选择。

这种隔离机制带来的结果是显而易见的:镜像体积大大减小,部署速度更快,资源占用更少,并且由于减少了不必要的组件,安全性也得到了提升。这不仅仅是技术上的优化,更是对资源的一种精打细算。

除了多阶段构建,还有哪些实用的PHP Docker镜像瘦身技巧?

当然,多阶段构建是“大招”,但就像做饭一样,除了主菜,配菜和调料也很重要。在Docker构建PHP镜像时,还有一些虽然看起来是小细节,但往往能带来大惊喜的“抠门”技巧,能让你的镜像更精简。

选择最精简的基础镜像: 这是最基础也是最有效的一步。对于PHP应用,通常推荐使用php:X.Y-fpm-alpine系列镜像。Alpine Linux以其极小的体积(通常只有几MB)而闻名,因为它使用的是musl libc而不是glibc,并且只包含最基本的工具。相比于基于Debian或Ubuntu的PHP镜像,Alpine版本能直接帮你省下几百兆的空间。当然,选择Alpine可能意味着某些特定的C语言扩展在编译时需要寻找对应的*-dev包,但这通常是值得的。充分利用.dockerignore文件: 这是一个经常被忽视,但极其重要的文件。它就像Git的.gitignore一样,告诉Docker在构建上下文时要忽略哪些文件和目录。比如你的.git目录、node_modules(如果你在容器外构建前端)、tests目录、docs.env文件、本地IDE的配置文件(如.idea/)等等,这些都不应该被复制到最终镜像中。一个小小的.dockerignore文件能帮你避免复制大量无用文件,从而直接减小镜像大小。合并RUN指令并清理: Docker的每一条RUN指令都会创建一个新的镜像层。层越多,镜像体积可能越大,而且在构建过程中,如果某个层的内容发生了变化,后续所有依赖于它的层都需要重新构建,导致缓存失效。所以,尽可能地将多个RUN命令合并成一个,使用&& 连接。更重要的是,在每个RUN命令的末尾,立即清理掉临时文件和缓存。例如,对于基于Debian/Ubuntu的镜像:apt-get update && apt-get install -y some-package && rm -rf /var/lib/apt/lists/*。对于Alpine:apk add --no-cache some-package && rm -rf /var/cache/apk/*。这个清理步骤是关键,它能确保安装过程中产生的临时文件不会被保留在镜像层中。只安装必要的运行时依赖: 仔细审视你的应用程序,它在运行时到底需要哪些系统包和PHP扩展?是不是真的需要vimcurlwgetgit这些工具?如果不需要,就不要安装。每多安装一个包,就多占用一份空间,多一份潜在的安全风险。这种“极简主义”原则在构建生产环境镜像时尤其重要。使用明确的镜像标签: 避免使用latestphp:8.2这样的宽泛标签作为基础镜像。它们可能会在未来发生变化,导致你的构建行为不一致。使用php:8.2-fpm-alpinephp:8.2.10-fpm-alpine这样具体的标签,可以确保你的构建是可复现的,并且不会因为上游镜像的更新而意外增加体积或引入问题。

这些技巧与多阶段构建结合使用,能让你的PHP Docker镜像达到一个非常精简和高效的状态。这不仅仅是为了节省磁盘空间,更是为了提升CI/CD流程的效率,缩短部署时间,并最终降低运营成本。

在实际项目中应用多阶段构建时可能遇到的挑战与应对策略

在实际项目中,尽管多阶段构建优势明显,但落地过程中也确实会遇到一些小麻烦,甚至让人挠头。这不像教程里那么一帆风顺,总有些边边角角的问题需要我们去琢磨和解决。

挑战1:构建依赖的“黑盒”问题。

问题描述: 有时候,你的PHP应用可能依赖某个特定的系统库(比如libxml2-dev)或者需要编译一个不常见的PECL扩展。在构建器阶段,你可能忘记安装这些依赖,导致编译失败。而由于是多阶段构建,这个失败的构建器阶段可能不会直接生成一个可运行的镜像让你进去调试,排查起来就有点像“盲人摸象”。应对策略:分步构建与调试: 在开发Dockerfile时,可以先只构建到第一个阶段。例如:docker build --target builder -t my-app-builder:debug .。构建成功后,你可以通过docker run -it my-app-builder:debug /bin/bash进入这个构建器镜像内部,手动执行命令,查看文件,就像在一个普通的容器里一样进行调试,直到所有构建步骤都顺利通过。详细的日志: 确保你的RUN命令输出足够详细的日志。如果构建失败,这些日志是排查问题的关键线索。逐步添加依赖: 不要一次性把所有依赖都写进去。从最少的基础开始,每当构建失败,就根据错误信息添加缺失的依赖,直到成功。

挑战2:文件复制的粒度控制。

问题描述: COPY --from=builder /path/to/source /path/to/dest这条命令看起来简单,但实际操作中,你可能需要精确地复制哪些文件,哪些目录。比如,Composer的vendor目录可能很大,你可能只想复制特定的子目录;或者前端构建后,你只想要dist文件夹里的内容。复制多了浪费空间,复制少了应用跑不起来。应对策略:明确路径: 在构建器阶段,把所有需要复制到最终镜像的文件和目录都放在一个明确的、易于识别的路径下,比如/app/dist//app/final-vendor/。这样在最终阶段复制时,路径会非常清晰。使用ls -la确认: 在调试构建器阶段时,进入容器内部,用ls -la命令确认你期望复制的文件和目录是否存在于正确的路径。细致的COPY指令: 不要怕多写几条COPY指令。与其COPY /app .一股脑地复制所有,不如COPY /app/vendor ./vendorCOPY /app/src ./srcCOPY /app/public ./public这样分开复制,粒度更细,也更安全。

挑战3:缓存失效与构建速度。

问题描述: Docker构建是基于层缓存的。一旦某个层发生变化,其后的所有层都会失效,需要重新构建。在多阶段构建中,如果你的应用代码频繁变动,可能会导致Composer依赖的安装层频繁失效,每次构建都要重新下载和安装vendor,这会非常耗时。应对策略:利用层缓存优化composer install 这是非常经典的优化技巧。将composer.jsoncomposer.lock文件单独复制到工作目录,然后运行composer install。只有当这两个文件发生变化时,这一层才会失效。之后再复制应用的其他代码。

# Builder StageFROM php:8.2-cli-alpine AS builderWORKDIR /appCOPY composer.json composer.lock ./  # 这一层只依赖composer文件RUN composer install --no-dev --optimize-autoloader --no-scriptsCOPY . . # 这一层依赖所有代码,但如果只有代码变动,composer层不会失效

使用--cache-from 在CI/CD环境中,可以利用--cache-from参数来指定一个已有的镜像作为缓存源,这样可以复用之前构建的层,即使本地没有。优化基础镜像选择: 选择一个更新频率相对稳定,且包含大部分常用PHP扩展的基础镜像,可以减少你在Dockerfile中安装和编译扩展的次数,从而减少层变动的可能性。

挑战4:多架构构建(例如M1 Mac)。

问题描述: 如果你的开发团队成员使用不同的CPU架构(例如Intel和M1 Mac),直接构建可能遇到基础镜像不支持或性能问题。应对策略:使用docker buildx buildx是Docker官方提供的构建工具,可以方便地进行多平台构建。通过它,你可以为不同的CPU架构构建兼容的镜像,确保团队成员无论使用何种架构,都能顺利构建

以上就是如何在Docker中构建多阶段PHP镜像 PHP环境优化镜像体积的方法的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1288711.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
使用 JavaScript 实现表格点击显示/隐藏效果
上一篇 2025年12月11日 05:14:12
如何配置PHP环境支持MQ服务对接 Docker容器连接RabbitMQ方法
下一篇 2025年12月11日 05:14:20

相关推荐

  • 开源免费PHP工具 PHP开发效率提升利器

    推荐开源免费PHP开发工具以提升效率:VS Code、Sublime Text轻量高效,PhpStorm专业强大;调试用Xdebug、Kint、Ray;依赖管理选Composer;代码质量工具包括PHPStan、Psalm、PHP_CodeSniffer;数据库管理可用%ignore_a_1%MyA…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000
  • c++如何实现UDP通信_c++基于UDP的网络通信示例

    UDP通信基于套接字实现,适用于实时性要求高的场景。1. 流程包括创建套接字、绑定地址(接收方)、发送(sendto)与接收(recvfrom)数据、关闭套接字;2. 服务端监听指定端口,接收客户端消息并回传;3. 客户端发送消息至服务端并接收响应;4. 跨平台需处理Winsock初始化与库链接,编…

    2026年5月10日
    100
  • 谷歌浏览器如何截图 谷歌浏览器页面截图技巧

    谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧

    使用谷歌浏览器的开发者工具截图步骤:1. 按ctrl+shift+i(windows/linux)或cmd+option+i(mac)打开开发者工具。2. 点击右上角三个点,选择”更多工具”,再选择”截图”。3. 选择截取整个页面。推荐的谷歌浏览器扩展…

    2026年5月10日 用户投稿
    100
  • MySQL数据库不支持中文的解决办法

    接上一篇文章,在解决了mysql+flask环境配置问题之后,往数据库存中文字符串会报1366错误,提示不正确的字符。继而发现默认的mysql采用了latin1字符集,这种编码是不支持中文的。 如果想支持中文的话,需要设置一下mysql字符集。 众所周知utf-8是可以的,gbk也没问题,为了可扩展…

    用户投稿 2026年5月10日
    000
  • JavaScript计算器开发:解决数值显示与初始化问题

    本教程深入探讨了使用JavaScript构建计算器时常见的数值显示异常问题,特别是由于类属性未初始化导致的`Cannot read properties of undefined`错误。我们将详细分析问题根源,并通过在构造函数中调用初始化方法来解决该问题,同时优化显示逻辑,确保计算器功能稳定且界面显…

    2026年5月10日
    000
  • NextAuth getToken 在服务端返回 null 的问题排查与解决

    问题描述 在使用 Next.js 和 NextAuth 构建应用程序时,有时需要在服务端获取用户的身份验证信息。getToken 函数是 NextAuth 提供的一个便捷方法,用于从请求中提取 JWT (JSON Web Token)。然而,在某些情况下,尤其是在使用 getServerSidePr…

    2026年5月10日
    000
  • pycharm解析器怎么添加 解析器添加详细流程

    在pycharm中添加解析器的步骤包括:1) 打开pycharm并进入设置,2) 选择project interpreter,3) 点击齿轮图标并选择add,4) 选择解析器类型并配置路径,5) 点击ok完成添加。添加解析器后,选择合适的类型和版本,配置环境变量,并利用解析器的功能提高开发效率。 在…

    2026年5月10日
    000
  • HTML文档如何工作?如何编辑HTML格式文件?

    HTML文档如何工作?如何编辑HTML格式文件?HTML文档如何工作?如何编辑HTML格式文件?HTML文档如何工作?如何编辑HTML格式文件?HTML文档如何工作?如何编辑HTML格式文件?

    浏览器解析和渲染html的过程包括:1. 解析html构建dom树;2. 结合css构建渲染树;3. 布局计算元素位置;4. 绘制像素到屏幕。编辑html可使用记事本、vs code、sublime text等文本或代码编辑器,其中vs code因语法高亮、自动补全和插件生态成为主流选择。标准htm…

    2026年5月10日 用户投稿
    000
  • 一台服务器上如何同时运行多个UWSGI服务避免冲突?

    多UWSGI服务部署方案:利用Docker实现服务器资源隔离 本文探讨如何在单台服务器上安全运行多个UWSGI服务,避免服务冲突。 问题在于,即使端口不同,两个UWSGI服务(例如:san和san_test)也可能发生冲突,后启动的服务覆盖之前的服务。 理想情况下,san_test应该持续运行,而s…

    2026年5月10日
    000
  • GolangWeb项目异常捕获与日志记录

    答案:通过中间件使用defer和recover捕获panic,结合zap等结构化日志库记录请求链路信息,为每个请求生成trace ID,实现异常捕获与可追踪日志,提升系统稳定性与可观测性。 在Go语言Web项目中,异常捕获与日志记录是保障系统稳定性和可维护性的关键环节。Go本身没有像其他语言那样的t…

    2026年5月10日
    000
  • Python官网用户调查的参与方式_Python官网反馈提交详细教程

    答案是通过访问Python官网新闻页面、邮件邀请链接或GitHub仓库提交反馈。具体为:访问官网查找用户调查公告,或点击邮件中的专属链接参与,在GitHub的cpython仓库提交技术建议,并注意如实填写问卷与保护隐私。 如果您希望参与Python官网的用户调查并提交反馈,可以通过官方指定的渠道完成…

    2026年5月10日
    000
  • JavaScript Electron桌面应用

    答案:使用JavaScript开发%ignore_a_1%桌面应用需结合Web技术与Node.js,通过主进程管理窗口、渲染进程展示界面,并利用IPC通信,调用系统功能如文件对话框,最后用electron-builder打包发布,注意安全与进程职责分离。 用JavaScript开发Electron桌…

    2026年5月10日
    000
  • 我有时使用 awk 而不是 Python 的四个原因

    Python 是一门强大的编程语言,但在某些特定场景下,Awk 的优势更为显著,尤其体现在可移植性、生命周期、代码简洁性和与其他工具的互操作性方面。 Python 脚本通常具有良好的可移植性,但并非总能在所有环境中完美运行,例如流行的 Docker 基础镜像 (如 Debian 和 Alpine)。…

    2026年5月10日
    000
  • Go语言连接外部MySQL数据库:DSN配置与常见错误解析

    本文详细阐述了go语言使用`go-sql-driver/mysql`驱动连接外部mysql数据库的正确方法。重点介绍了数据源名称(dsn)的规范格式,特别是主机地址部分的配置,以避免常见的“getaddrinfow: the specified class was not found.”等网络解析错…

    2026年5月10日
    000
  • Tensorflow 音乐预测

    在本文中,我展示了如何使用张量流来预测音乐风格。在我的示例中,我比较了电子音乐和古典音乐。 你可以在我的github上找到代码:https://github.com/victordalet/sound_to_partition i – 数据集 第一步,您需要创建一个数据集文件夹,并在里面…

    2026年5月10日
    000
  • Linux文件系统iostat命令使用技巧

    Linux文件系统iostat命令使用技巧Linux文件系统iostat命令使用技巧Linux文件系统iostat命令使用技巧Linux文件系统iostat命令使用技巧

    iostat是Linux系统中用于监控I/O设备负载的关键工具,能分析磁盘性能并识别瓶颈。默认输出包括CPU使用率和设备I/O统计,分为系统启动以来的平均值和当前采样周期数据。核心指标有:%util反映设备利用率,持续接近100%可能表示I/O瓶颈;await为平均I/O等待时间,过高说明响应变慢;…

    2026年5月10日 用户投稿
    000
  • 如何测试html5编码_测试HTML5页面编码兼容性方法【编码测试】

    HTML5页面编码兼容性测试需五步:一查meta charset是否正确且前置;二验HTTP响应头Content-Type charset是否为utf-8;三用file或chardet工具探测实际编码;四跨浏览器测试URL参数中中文、Emoji解析;五通过W3C验证服务检查编码声明与字节一致性。 如…

    2026年5月10日
    100
  • Go语言与Microsoft SharePoint集成指南

    Go语言可以有效集成Microsoft SharePoint,主要通过两种途径:一是利用SharePoint提供的RESTful API进行数据交互,Go的标准HTTP客户端库即可轻松实现;二是通过SharePoint应用模型开发自托管应用,这种模型支持使用包括Go在内的任何语言编写后端逻辑。 1.…

    2026年5月10日
    000
  • javascript生命周期钩子是什么_组件有哪些关键阶段?

    JavaScript原生无生命周期钩子,这是Vue、React等框架为组件设计的机制;Vue按创建、挂载、更新、卸载四阶段提供对应钩子,React类组件有明确生命周期方法,函数组件则通过useEffect模拟,其核心价值在于精准控制执行时机以避免DOM操作错误和内存泄漏。 JavaScript 本身…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信