
本文讲解如何将 laravel 中扁平的权限字符串集合(如 "post access")转换为嵌套结构数组,便于在 blade 模板中动态渲染权限复选框组。
在使用 spatie/laravel-permission 或自定义权限系统时,常需将类似 ["post access", "post create", "post read", "publish access"] 这样的字符串集合,按资源(resource)分组,转换为结构清晰、可直接绑定到 Blade 表单的数组格式:
[
[
'name' => 'post',
'access' => false,
'create' => false,
'read' => false,
'update' => false,
'delete' => false,
],
[
'name' => 'publish',
'access' => false,
],
]该结构天然适配 Blade 中的 @foreach 循环与复选框 name="permissions[post][read]" 等命名约定,大幅提升表单处理与权限批量赋值的可维护性。
✅ 推荐实现方式(简洁、健壮、可扩展):
use Illuminate\Support\Str;
use Illuminate\Support\Collection;
$permissions = collect([
"post access", "post create", "post read", "post update", "post delete",
"publish access", "user read", "user update"
]);
// 步骤 1:按资源名分组(提取空格前部分)
$grouped = $permissions->groupBy(fn ($item) => Str::before($item, ' '));
// 步骤 2:为每组生成结构化数组
$result = $grouped->map(function (Collection $items, string $resource) {
// 定义所有可能的权限动作(可按需扩展)
$actions = ['access', 'create', 'read', 'update', 'delete'];
// 初始化基础结构
$row = ['name' => $resource];
// 遍历每个权限项,提取动作并设为 false
foreach ($items as $permission) {
$action = Str::after($permission, ' ');
if (in_array($action, $actions)) {
$row[$action] = false;
}
}
return $row;
})->values()->toArray();
// 输出结果(可用于 Blade 的 @foreach)
dd($result);? 关键说明:
- Str::before($item, ' ') 精准提取资源名(如 "post"),避免正则或模糊匹配风险;
- Str::after($item, ' ') 安全获取动作名(如 "create"),即使含空格也稳定;
- 使用 groupBy + map 是 Laravel Collection 的惯用范式,语义清晰、链式流畅;
- ->values()->toArray() 确保返回纯数字索引数组,兼容 Blade @foreach ($permissions as $group)。
⚠️ 注意事项:
- 原始答案中的 Str::contains(['post','get'], $item) 逻辑错误(post 是资源而非关键词),且硬编码索引 $array[0] 易引发越界;
- 不建议手动管理数组下标(如 $array[0], $array[1]),应依赖 groupBy 动态识别资源;
- 若权限动作集固定,建议提取为常量或配置项,便于统一维护;
- 实际项目中,可将此逻辑封装为 PermissionTransformer 类或 Eloquent Accessor,提升复用性。
通过以上方式,你不仅能正确生成复选框所需的数据结构,还能轻松扩展支持新资源(如 category)或新动作(如 restore),真正实现权限 UI 的声明式开发。










