当我们处理用户或帖子时,我们经常需要的功能之一是生成用户名或 slug。 然而,自动处理重复并不总是那么简单。 让我们看看如何使用一个非常简单的解决方案:文章URL 生成器。
持久化模型时生成用户名
假设我们想在将模型存储到数据库时自动从用户名生成用户名(或 slug)。 通常,这没什么大不了的,我们可以使用一个简单的 Eloquent 事件回调和 Str::slug() 助手
class User extends Authenticatable implements MustVerifyEmail
{
protected static function booted(): void
{
static::creating(static function (self $user): void {
$user->username = Str::slug($user->name, '.');
});
}
}
对于slugs,通常我们使用–作为分隔符,但对于用户名,它更可能是.
这段代码运行良好,干净且富有表现力。但是,如果有两个同名用户怎么办?用户名应该是唯一的。
生成增量用户名或URL
使用非常简单的正则表达式模式,我们可以轻松地跟踪数据库中相似的用户名,并为用户名添加或增加数字后缀。让我们看一下代码并更新模型中的创建回调
protected static function booted(): void
{
static::creating(static function (self $user): void {
$user->username = static::generateUniqueUsername(
Str::slug($user->name, '.')
);
});
}
protected static function generateUniqueUsername(string $value): string
{
$username = static::query()
->where('username', 'regexp', preg_quote($value).'[\\d]?$')
->orderByDesc('username')
->take(1)
->value('username');
$number = (int) filter_var($username, FILTER_SANITIZE_NUMBER_INT);
return $number === 0 ? $value : $value.++$number;
}
注意,在 MySQL 中我们需要使用双反斜杠:\\
在代码中,我们正在搜索与当前生成的用户名匹配的“最新”用户名,并在需要时添加或增加数字前缀。因此它会自动生成以下模式:用户名、用户名1、用户名2、用户名3,等等
总结
这是一个非常简单的解决方案,对于更简单的应用程序应该可以很好地工作。但是,当您需要更复杂的 slug 或用户名解决方案时,您可以扩展此解决方案或引入一个包