PHP单体转微服务要改哪些地方_迁移思路【教程】

发布时间 - 2025-12-27 00:00:00    点击率:
微服务拆分后应弃用$_SESSION,改用JWT无状态认证;$_COOKIE仅存非敏感字段并设Domain/SameSite;数据库事务改用消息队列实现最终一致性;公共代码抽为独立Composer包;各服务独立部署、配置FPM参数并提供标准健康检查接口。

单体 PHP 里的 $_SESSION$_COOKIE 怎么办

微服务拆分后,用户会跨多个服务(如 auth-serviceorder-service)请求,而 PHP 默认的文件或 Redis session 存储只绑定在单一服务进程里,其他服务无法读取 $_SESSION。硬共享 session 存储(比如全用同一个 Redis DB + 相同 session_id)看似可行,但实际会引发并发写冲突、过期策略不一致、敏感数据泄露等问题。

更稳妥的做法是彻底弃用 $_SESSION,改用无状态认证:

  • 登录成功后,auth-service 签发 JWT(含 user_idroleexp),通过 HTTP Header(如 Authorization: Bearer xxx)透传给下游服务
  • 所有服务统一校验 JWT 签名和有效期,不再依赖 session 存储
  • $_COOKIE 仅保留非敏感字段(如语言偏好),且必须设置 DomainSameSite 属性适配多子域(如 Domain=.example.com

数据库连接和事务怎么拆

原单体常共用一个 MySQL 实例,用事务包裹跨模块操作(如“扣库存 + 写订单 + 发通知”)。微服务要求每个服务独占数据库 Schema,跨服务事务无法靠本地 START TRANSACTION 保证一致性。

必须改成最终一致性方案:

  • 订单服务创建订单时,只写本地 orders 表,状态设为 pending
  • 通过消息队列(如 RabbitMQ 或 Kafka)异步发 inventory.deduct 消息给库存服务
  • 库存服务处理成功后,再发 order.confirmed 回调;失败则触发补偿任务(如自动取消订单)
  • 避免在 PHP-FPM 中直接调用其他服务的 HTTP 接口做同步事务——超时、级联失败风险极高

原来用 require_once 引入的公共函数库怎么复用

单体里把工具函数、模型类放在 app/Helpers/app/Models/ 下,用 require_once 或 Composer 自动加载。微服务中这些代码不能直接跨服务引用,否则形成强耦合和部署依赖。

正确做法是分层抽象:

  • 把通用逻辑(如密码哈希、ID 生成、HTTP 客户端封装)抽成独立 Composer 包(如 myorg/php-common-utils),发布到私有 Packagist 或 Git repo,各服务按需 composer require
  • 领域模型(如 UserOrder)**不要共享类**,各服务定义自己的 DTO 或 request struct,通过 API Schema(OpenAPI)或 Protobuf 明确约定字段和类型
  • 禁止在服务 A 的代码里 new ServiceBClient() 调用服务 B——应通过 API Gateway 统一路由,或使用 SDK 封装(SDK 只负责 HTTP 请求构造,不包含业务逻辑)

PHP-FPM 配置和部署方式必须变

单体通常一个 Nginx + 一组 PHP-FPM 进程跑全部逻辑;微服务需要每个服务独立部署、扩缩容、监控。这意味着:

  • 每个服务必须有自己的 Dockerfile,基于 php:8.2-cliphp:8.2-apache 构建,不共用同一套 FPM 配置
  • Nginx 不再直接代理 PHP-FPM,而是作为 API Gateway,用 upstream 分发请求到不同服务的容器(如 auth-service:8080order-service:8080
  • FPM 的 pm.max_childrenpm.start_servers 等参数要按服务负载单独调优——例如日志服务可设低内存限制,订单服务需更高并发连接数
  • 健康检查接口(如 /healthz)必须返回 JSON 格式、明确状态码,供 Kubernetes 或 Consul 做服务发现
GET /healthz
HTTP/1.1 200 OK
Content-Type: application/json

{"status":"ok","timestamp":1717023456,"service":"order-service"}

真正难的不是改代码,是改协作习惯:接口变更要先更新 OpenAPI 文档,数据库 schema 变更必须配套迁移脚本并测试回滚路径,任何服务都不能假设其他服务永远可用——超时、重试、熔断得写进 PHP 代码里,而不是靠运维兜底。


# mysql  # php  # redis  # js  # git  # json  # docker  # composer  # apache  # nginx  # rabbitmq  # gateway  # kafka  # 封装  # require  # Session  # 接口  # Struct  # 并发  # 异步  # consul  # 数据库  # kubernetes  # http  # 自己的  # 放在  # 多个  # 设为  # 更高  # 要先  # 极高  # 绑定  # 回调  # 再发 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 1688铺货到淘宝怎么操作 1688一键铺货到自己店铺详细步骤  Laravel如何实现事件和监听器?(Event & Listener实战)  Laravel定时任务怎么设置_Laravel Crontab调度器配置  Python正则表达式进阶教程_复杂匹配与分组替换解析  百度输入法全感官ai怎么关 百度输入法全感官皮肤关闭  如何用腾讯建站主机快速创建免费网站?  如何在腾讯云免费申请建站?  如何破解联通资金短缺导致的基站建设难题?  JavaScript 输出显示内容(document.write、alert、innerHTML、console.log)  如何在自有机房高效搭建专业网站?  北京企业网站设计制作公司,北京铁路集团官方网站?  百度输入法ai组件怎么删除 百度输入法ai组件移除工具  php485函数参数是什么意思_php485各参数详细说明【介绍】  标题:Vue + Vuex 项目中正确使用 JWT 进行身份认证的实践指南  ChatGPT常用指令模板大全 新手快速上手的万能Prompt合集  如何在景安云服务器上绑定域名并配置虚拟主机?  VIVO手机上del键无效OnKeyListener不响应的原因及解决方法  Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议  使用PHP下载CSS文件中的所有图片【几行代码即可实现】  轻松掌握MySQL函数中的last_insert_id()  Laravel怎么配置S3云存储驱动_Laravel集成阿里云OSS或AWS S3存储桶【教程】  大学网站设计制作软件有哪些,如何将网站制作成自己app?  高性能网站服务器配置指南:安全稳定与高效建站核心方案  进行网站优化必须要坚持的四大原则  小视频制作网站有哪些,有什么看国内小视频的网站,求推荐?  Laravel API路由如何设计_Laravel构建RESTful API的路由最佳实践  Laravel如何生成和使用数据填充?(Seeder和Factory示例)  Bootstrap整体框架之JavaScript插件架构  php做exe能调用系统命令吗_执行cmd指令实现方式【详解】  如何用好域名打造高点击率的自主建站?  如何在阿里云完成域名注册与建站?  香港服务器建站指南:外贸独立站搭建与跨境电商配置流程  极客网站有哪些,DoNews、36氪、爱范儿、虎嗅、雷锋网、极客公园这些互联网媒体网站有什么差异?  创业网站制作流程,创业网站可靠吗?  企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?  齐河建站公司:营销型网站建设与SEO优化双核驱动策略  C#如何调用原生C++ COM对象详解  香港服务器租用费用高吗?如何避免常见误区?  高端网站建设与定制开发一站式解决方案 中企动力  怎么用AI帮你设计一套个性化的手机App图标?  如何在万网开始建站?分步指南解析  高端建站三要素:定制模板、企业官网与响应式设计优化  Laravel如何操作JSON类型的数据库字段?(Eloquent示例)  Gemini怎么用新功能实时问答_Gemini实时问答使用【步骤】  Laravel的路由模型绑定怎么用_Laravel Route Model Binding简化控制器逻辑  作用域操作符会触发自动加载吗_php类自动加载机制与::调用【教程】  小米17系列还有一款新机?主打6.9英寸大直屏和旗舰级影像  如何在云服务器上快速搭建个人网站?  Laravel如何实现本地化和多语言支持?(i18n教程)  Laravel怎么实现观察者模式Observer_Laravel模型事件监听与解耦开发【指南】