最近在维护老项目时,又一次用到了 Phinx。
这个工具我已经用了很多年,几乎每个项目都会用上它。它属于那种平时不常用,但每个项目都离不开 的工具。
问题在于,它用得不频繁,每次写迁移脚本时总会忘记某个参数怎么写、某个字段该用什么类型。
这些当然可以去查官方文档,但 Phinx 的文档虽然内容齐全,却总让我觉得信息分散、查起来不够顺手。
于是,我干脆花点时间,把自己常用的命令、配置方式、字段类型和参数说明都系统地整理了一遍。
一方面方便自己查阅,另一方面也希望能帮到同样在项目中使用 Phinx 的开发者。
如果你也在 PHP 项目里用 Phinx 管理数据库迁移,这篇文章或许能成为你的「快捷参考手册」。
前期配置
安装 Phinx
composer require robmorgan/phinx
初始化配置
执行初始化命令:
vendor/bin/phinx init
该命令会在项目根目录下生成一个 phinx.php 配置文件。
配置文件中:
- paths用于指定迁移脚本与填充脚本的路径;
- environments用于定义不同环境(例如开发、测试、生产)的数据库配置;
- 若执行迁移命令时不指定环境,则默认使用 default_environment指定的环境。
环境配置与 .env 支持
为了统一管理数据库配置,我习惯用 .env 文件来维护连接信息。
但由于 phinx 脚本并不运行在 PHP 框架生命周期内,常见的 .env 加载方式(如 Laravel 的)无法直接使用,因此我通常会额外安装 vlucas/phpdotenv:
composer require vlucas/phpdotenv
然后在 phinx.php 中添加如下配置:
<?php
$dotenv = Dotenv\Dotenv::createImmutable(__DIR__);
$dotenv->load();
return ["paths" => ["migrations" => "database/migrations","seeds" => "database/seeds"],"environments" => ["default_migration_table" => "phinxlog","default_environment" => "production","production" => ["adapter" => "mysql","host" => $_SERVER['DB_HOST'],"name" => $_SERVER['DB_NAME'],"user" => $_SERVER['DB_USER'],"pass" => $_SERVER['DB_PASS'],"port" => $_SERVER['DB_PORT'],"charset" => $_SERVER['DB_CHARSET']]]
];
常用命令
创建迁移文件
vendor/bin/phinx create 迁移名称
Phinx 会自动根据时间戳与名称生成迁移文件。
命名建议保持统一规范:
- 创建表:Create + 表名→ 例如CreateUsersTable
- 修改表:Modify + 表名 + 操作→ 例如ModifyUsersAddStatus
- 删除表:Delete + 表名→ 例如DeleteOldLogsTable
Phinx 会自动将大驼峰命名转为蛇形文件名,例如
CreateUsersTable→20251028123045_create_users_table.php
执行迁移
没什么好说的,执行这个命令即可执行迁移
vendor/bin/phinx migrate
参数说明:
- -e:指定环境(默认production)
- -t:指定执行到的版本号(不传则执行到最新)
- --dry-run:仅打印 SQL 而不实际执行
注意:
-t不是只执行单个版本,而是顺序执行直到目标版本号为止。
设置断点(Breakpoint)
生产环境中执行迁移后,建议立刻设置断点。
断点能防止误操作导致大规模回滚。
vendor/bin/phinx breakpoint
参数:
- -e环境名
- -t版本号
- -r删除断点
回滚迁移
慎用!
迁移的回滚会撤销数据库结构更改,例如删除表或字段,表中数据会全部丢失。
生产环境不建议直接回滚,而是通过新的迁移来完成结构恢复。
vendor/bin/phinx rollback
参数说明:
- -e环境名
- -t指定版本号(或0回滚全部)
- -d按日期回滚(YYYYmmddHHiiss)
- -f强制回滚(忽略断点)
- --dry-run仅打印 SQL 不执行
查看状态
用于查看迁移执行状态,确认哪些迁移尚未执行。
vendor/bin/phinx status
参数说明:
- -e:环境名
创建填充(Seeder)
vendor/bin/phinx seed:create 填充名称
命名建议使用表名 + 填充描述,例如:
UsersAddDemoDataSeeder
执行填充:
vendor/bin/phinx seed:run
参数:
- -e指定环境
- -s指定执行的 Seeder,可多次传入
迁移脚本详解
支持的字段类型
Phinx 在 MySQL 下支持的字段类型如下:
| 类型 | MySQL 类型 | 必需参数 | 说明 | 
|---|---|---|---|
|  smallinteger | SMALLINT | — | 小整数 | 
|  integer | INT | — | 常规整数 | 
|  biginteger | BIGINT | — | 大整数 | 
|  float | FLOAT | — | 单精度浮点数 | 
|  double | DOUBLE | — | 双精度浮点数 | 
|  decimal | DECIMAL(M,D) |  precision,scale | 定点小数(金额等) | 
|  bit | BIT | — | 位字段 | 
|  boolean | TINYINT(1) | — | 布尔值(0/1) | 
|  char | CHAR(n) |  limit | 定长字符串 | 
|  string | VARCHAR(n) |  limit | 可变长度字符串 | 
|  text | TEXT | — | 文本 | 
|  enum | ENUM(...) |  values | 枚举类型 | 
|  set | SET(...) |  values | 集合类型 | 
|  uuid | CHAR(36) | — | UUID字符串 | 
|  date | DATE | — | 日期 | 
|  time | TIME | — | 时间 | 
|  datetime | DATETIME | — | 日期与时间 | 
|  timestamp | TIMESTAMP | — | 时间戳 | 
|  binary | BINARY(n) |  limit | 定长二进制数据 | 
|  blob | BLOB | — | 二进制大对象 | 
|  tinyblob | TINYBLOB | — | 小型二进制 | 
|  mediumblob | MEDIUMBLOB | — | 中型二进制 | 
|  longblob | LONGBLOB | — | 超大二进制 | 
|  json | JSON | — | JSON 数据类型 | 
常用字段参数
| 参数名 | 类型 | 说明 | 
|---|---|---|
|  limit / length | int | 长度或字节限制 | 
|  default | mixed | 默认值 | 
|  null | bool | 是否允许为 NULL | 
|  after | string | 放置在某个字段之后(或 \Phinx\Db\Adapter\MysqlAdapter::FIRST ) | 
|  comment | string | 字段注释 | 
|  precision | int | 小数总位数 | 
|  scale | int | 小数位数 | 
|  signed | bool | 是否允许负数 | 
|  values | array/string | 枚举或集合可选值,英文逗号隔开的字符串或数组 | 
|  identity | bool | 是否自增(需搭配 null:false) | 
各种操作调用代码
如果对表格形式不感兴趣,可以直接通过以下完整迁移脚本理解字段定义方式。
<?phpdeclare(strict_types=1);use Phinx\Db\Adapter\MysqlAdapter;
use Phinx\Migration\AbstractMigration;final class CreateTable extends AbstractMigration
{/*** Change Method.** Write your reversible migrations using this method.** More information on writing migrations is available here:* https://book.cakephp.org/phinx/0/en/migrations.html#the-change-method** Remember to call "create()" or "update()" and NOT "save()" when working* with the Table class.*/public function change(): void{// 前置说明:// create() 操作: 表不存在的情况下,创建表,并保存字段操作// update() 操作: 表存在的情况下,变更表,并保存字段操作// 所有的操作记得到最后一定需要对构建的表对象执行保存操作,否则不会执行// 判断表是否存在: $this->hasTable('table_name');// 删除表: $this->table('table_name')->drop()->save();// 构建表$table = $this->table('table_name', ['id' => 'table_id', 'comment' => '这是 table 表']);// 第二个参数字段,可不传,主要用于在 create() 时构建表级的基本信息// 支持的参数:// id:            主键 ID 列,传入字符串或者布尔类型//                    如果是字符串:按照字符串内容创建自增主键//                    如果是false:不自动创建主建//                    如果不传:自动创建名称为 id 的自增主键// comment:       设置表的注释// row_format:    设置表的行格式(MySQL 特有,例如 DYNAMIC, COMPACT, REDUNDANT, COMPRESSED)// engine:        设置表的存储引擎(MySQL 特有,例如 InnoDB, MyISAM, MEMORY)// collation:     设置字符集的排序规则,默认 utf8mb4_unicode_ci// signed:        整数类型是否允许负数,默认 false// limit:         设置主键最大长度// 变更表注释$table->changeComment('这是 table 表新的注释');// 更改主键,可以设置多个列组联合主键$table->changePrimaryKey(['new_id']);// 重命名表$table->rename('new_table_name');// 检查字段是否存在:$table->hasColumn('username');// 重命名字段$table->renameColumn('老字段名', '新字段名');// 删除字段$table->removeColumn('字段名');// 添加字段$table->addColumn('字段名', '类型', ['参数'])->addColumn('avatar', 'string', ['comment' => '头像', 'null' => true, 'after' => 'name'])->addColumn('created_at', 'integer', ['comment' => '创建时间', 'null' => false, 'limit' => MysqlAdapter::INT_BIG]);// 类型:// 很多不必要或可以简化定义的类型,例如数字方面,可以全部都设置为 `integer` 通过 `limit` 参数设置长度即可自动创建对应类型// 以下是支持的类型:// smallinteger(SMALLINT):  小整数// integer(INT):            常规整数// biginteger(BIGINT):      大整数// float(FLOAT):            单精度浮点数// double(DOUBLE):          双精度浮点数// decimal(DECIMAL(M,D)):   定点小数,必须携带 `precision` 与 `scale` 参数// bit(BIT):                位字段// boolean(TINYINT(1)):     布尔值(0/1)// char(CHAR(n)):           定长字符串,必须携带 `limit` 参数// string(VARCHAR(n)):      可变长度字符串,必须携带 `limit` 参数// text(TEXT):              文本// enum(ENUM(...)):         枚举类型,必须携带 `values` 参数// set(SET(...)):           集合类型,必须携带 `values` 参数// uuid(CHAR(36)):          UUID字符串// date(DATE):              日期(YYYY-MM-DD)// time(TIME):              时间(HH:MM:SS)// datetime(DATETIME):      日期与时间// timestamp(TIMESTAMP):    时间戳// binary(BINARY(n)):       定长二进制数据,必须携带 `limit` 参数// blob(BLOB):              二进制大对象// tinyblob(TINYBLOB):      小型二进制// mediumblob(MEDIUMBLOB):  中型二进制// longblob(LONGBLOB):      超大二进制// json(JSON):JSON          数据类型// 参数:// 以下是支持的参数:// limit / length:          长度,int 类型,内容跟随字段类型变化// default:                 默认值,类型根据字段类型变化// null:                    是否允许为 null,bool 类型,true 或者 false// after:                   指定字段应该放在哪个列之后,string 类型传递列名,或者 \Phinx\Db\Adapter\MysqlAdapter::FIRST// comment:                 字段注释,string 类型// precision:               结合 scale 设置小数精度,int 类型// scale:                   结合 precision 设置小数精度,int 类型// signed:                  是否允许负数,bool 类型// values:                  枚举类型,可以是英文逗号分隔的string,或array// identity:                数字类型自增长,bool类型,需要设置 null:false// 修改字段$table->changeColumn('字段名', '类型', ['参数']);// 设置索引$table->addIndex(['字段名']);// 删除索引$table->removeIndex(['字段名']);// 或者$table->removeIndexByName('索引名');}
}
填充脚本(Seeder)示例
<?phpdeclare(strict_types=1);use Phinx\Seed\AbstractSeed;class TableAddDemoDataSeeder extends AbstractSeed
{/*** Run Method.** Write your database seeder using this method.** More information on writing seeders is available here:* https://book.cakephp.org/phinx/0/en/seeding.html*/public function run(): void{// 构建数据$data = [['body'    => 'foo','created' => date('Y-m-d H:i:s'),],['body'    => 'bar','created' => date('Y-m-d H:i:s'),]];// 构建表$table = $this->table('table_name');// 添加数据,并保存$table->insert($data)->saveData();// 清空表中的所有数据$table->truncate();}
}
老项目迁移接入
如果你是从已有数据库切换到 Phinx,可使用 phinx-migrations-generator 自动生成迁移文件。
composer require odan/phinx-migrations-generator --dev
它会直接读取现有的 phinx.php 配置,无需额外设置。
执行以下命令即可生成当前数据库结构的迁移文件:
vendor/bin/phinx-migrations generate
最后
Phinx 是一个非常轻量却强大的迁移工具,适用于几乎所有独立 PHP 项目。
写这篇笔记的初衷很简单——自己每次都要重新翻文档太麻烦了,不如干脆整理一份系统化的版本。
如果你也有相同的困扰,希望这篇文章能让你变得方便一些
觉得有用的话,别忘了收藏这篇文章~
更多内容可以在我的网站 https://hejunjie.life 查看
