0

0

优化 Laravel 数据库通知:实现聚合与避免重复创建

霞舞

霞舞

发布时间:2025-11-23 12:27:01

|

240人浏览过

|

来源于php中文网

原创

优化 laravel 数据库通知:实现聚合与避免重复创建

本文详细阐述了在 Laravel 应用中如何实现数据库通知的聚合,以避免在短时间内向用户发送大量相似通知。核心策略是在特定时间窗口内,通过更新现有通知的计数和内容,而非创建新的通知,来优化用户体验。文章将深入分析 `toDatabase` 方法的机制,并提供关键代码示例,展示如何在更新操作完成后,阻止 Laravel 自动创建新的通知记录。

引言:优化用户通知体验

在现代Web应用中,通知是与用户互动的重要方式。然而,当系统在短时间内触发大量相似事件时(例如,多个新帖子在30分钟内发布),如果每个事件都生成一个独立的通知,用户可能会感到信息过载,从而降低应用的使用体验。为了解决这一问题,一种常见的策略是通知聚合:将相似的通知合并为一个,并通过更新其内容(如增加计数)来反映事件的累积。

Laravel 数据库通知机制概述

Laravel 提供了强大的通知系统,其中数据库通知允许我们将通知存储在数据库中。当使用 Illuminate\Notifications\Notification 类时,我们需要实现 toDatabase 方法来定义通知的数据结构。这个方法返回一个数组,Laravel 会将这个数组序列化并存储在 notifications 表的 data 字段中。

例如,一个基本的 toDatabase 方法可能如下所示:

public function toDatabase($notifiable): array
{
    return [
        'content' => [
            'en' => "1 new post matched with your saved search {$this->search->title} has been posted, Press here to view more.",
        ],
        'count' => 1,
        'search' => $this->search->id,
        'parameters' => $this->search->parameters
    ];
}

当 Notification::send() 或 Notifiable::notify() 被调用时,Laravel 会执行这个 toDatabase 方法,并将其返回的数据保存为新的通知记录。

现有尝试与核心挑战

为了实现通知聚合,一个直观的想法是在 toDatabase 方法中首先检查是否存在符合条件的近期通知。如果存在,就更新它的计数和内容;否则,创建一个新的通知。

以下是这种思路的一个初步实现:

Open Voice OS
Open Voice OS

OpenVoiceOS是一个社区驱动的开源语音AI平台

下载
// 初始尝试的代码片段
public function toDatabase($notifiable): array
{
    $count = 1;
    // 尝试查找在过去30分钟内,与当前搜索相关的现有通知
    if ($notification = $notifiable->notifications()
        ->where('data->search', $this->search->id)
        ->where('updated_at', '>=', now()->subMinutes(30))
        ->first()) {

        // 如果找到,更新其计数
        $count = isset($notification->data['count']) ? $notification->data['count'] + 1 : 1;
        $notification->update([
            'data' => [
                'content' => [
                    'en' => "{$count} new posts matched with your saved search {$this->search->title} has been posted, Press here to view more.",
                ],
                'count' => $count,
                // 确保保留其他原有数据
                'search' => $this->search->id,
                'parameters' => $this->search->parameters
            ]
        ]);
    }

    // 问题所在:无论是否更新了现有通知,这里都会返回一个数组
    return [
        'content' => [
            'en' => "{$count} new post matched with your saved search {$this->search->title} has been posted, Press here to view more.",
        ],
        'count' => $count,
        'search' => $this->search->id,
        'parameters' => $this->search->parameters
    ];
}

上述代码的核心挑战在于,即使成功更新了现有的通知,toDatabase 方法的最后依然会 return []。根据 Laravel 的通知发送机制,只要 toDatabase 方法返回一个非空数组,Laravel 就会将其视为一个新的通知数据,并将其保存到数据库中。这就导致了即使我们更新了旧通知,系统仍然会创建一个新的通知,从而无法达到聚合的目的。

解决方案:条件性阻止新通知创建

解决这个问题的关键在于,当成功更新现有通知时,阻止 toDatabase 方法返回一个有效的通知数据数组。Laravel 的 DatabaseChannel 在处理 toDatabase 的返回值时,会检查其是否为 null 或空数组。如果返回 null 或空数组,则不会创建新的通知记录。

因此,我们可以在更新现有通知后,直接返回 null 或一个空数组 [],以告知 Laravel 不需要创建新的通知。

以下是优化后的 toDatabase 方法实现:

search = $search;
    }

    /**
     * 获取通知的交付渠道。
     *
     * @param  mixed  $notifiable
     * @return array
     */
    public function via($notifiable)
    {
        return ['database'];
    }

    /**
     * 获取通知的数据库表示。
     *
     * @param  mixed  $notifiable
     * @return array|null // 关键:返回类型可以是 array 或 null
     */
    public function toDatabase($notifiable): ?array
    {
        // 1. 定义聚合的时间窗口(例如30分钟)
        $timeWindow = now()->subMinutes(30);

        // 2. 尝试查找在指定时间窗口内,针对当前用户和当前搜索的同类型通知
        // 务必添加 'type' 过滤,以确保只聚合相同类型的通知
        $existingNotification = $notifiable->notifications()
            ->where('type', self::class) // 过滤通知类型
            ->where('data->search', $this->search->id) // 过滤搜索ID
            ->where('updated_at', '>=', $timeWindow) // 过滤时间窗口
            ->first();

        if ($existingNotification) {
            // 3. 如果找到现有通知,则更新其计数和内容
            $newCount = isset($existingNotification->data['count']) ? $existingNotification->data['count'] + 1 : 1;
            $existingNotification->update([
                'data' => [
                    'content' => [
                        'en' => "{$newCount} new posts matched with your saved search {$this->search->title} has been posted, Press here to view more.",
                    ],
                    'count' => $newCount,
                    'search' => $this->search->id,
                    'parameters' => $this->search->parameters,
                ]
            ]);

            // 4. 关键步骤:返回 null 或空数组以阻止 Laravel 创建新的通知记录
            return null; // 或者 return [];
        }

        // 5. 如果没有找到现有通知,则创建新的通知
        return [
            'content' => [
                'en' => "1 new post matched with your saved search {$this->search->title} has been posted, Press here to view more.",
            ],
            'count' => 1,
            'search' => $this->search->id,
            'parameters' => $this->search->parameters
        ];
    }
}

注意事项与最佳实践

  1. 通知类型过滤 (where('type', self::class)): 在查询现有通知时,务必通过 type 字段进行过滤。type 字段存储了通知类的完整命名空间,这确保了我们只聚合相同类型的通知,避免将不同类型的通知错误地合并。
  2. updated_at 字段的利用: Laravel 数据库通知表自带 updated_at 字段,它会在每次记录更新时自动更新。这使得它非常适合作为判断通知是否在特定时间窗口内的依据。
  3. 数据结构的一致性: 在更新 data 字段时,要确保其结构与首次创建时一致。如果只更新 count 而丢失了 search 或 parameters 等其他重要信息,可能会导致通知显示不完整或功能异常。因此,在更新时应确保所有相关数据都被重新包含在 data 数组中。
  4. 可读性与维护性: 尽管在 toDatabase 方法中包含复杂的逻辑是可行的,但对于更复杂的聚合需求,可以考虑将通知查找和更新逻辑封装到单独的服务类或仓库中,以提高代码的可读性和可维护性。
  5. 前端展示: 当通知被聚合后,前端需要能够正确解析 data 字段中的 count 和 content,以展示聚合后的信息。

总结

通过在 toDatabase 方法中巧妙地运用条件判断和返回值控制,我们能够有效地实现 Laravel 数据库通知的聚合功能。当检测到在特定时间窗口内存在相似通知时,系统会更新现有通知而非创建新通知,并通过返回 null 来阻止 Laravel 的默认新通知创建行为。这一策略显著改善了用户在短时间内接收大量通知时的体验,使得通知系统更加智能和用户友好。

相关专题

更多
laravel组件介绍
laravel组件介绍

laravel 提供了丰富的组件,包括身份验证、模板引擎、缓存、命令行工具、数据库交互、对象关系映射器、事件处理、文件操作、电子邮件发送、队列管理和数据验证。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

316

2024.04.09

laravel中间件介绍
laravel中间件介绍

laravel 中间件分为五种类型:全局、路由、组、终止和自定。想了解更多laravel中间件的相关内容,可以阅读本专题下面的文章。

271

2024.04.09

laravel使用的设计模式有哪些
laravel使用的设计模式有哪些

laravel使用的设计模式有:1、单例模式;2、工厂方法模式;3、建造者模式;4、适配器模式;5、装饰器模式;6、策略模式;7、观察者模式。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

369

2024.04.09

thinkphp和laravel哪个简单
thinkphp和laravel哪个简单

对于初学者来说,laravel 的入门门槛较低,更易上手,原因包括:1. 更简单的安装和配置;2. 丰富的文档和社区支持;3. 简洁易懂的语法和 api;4. 平缓的学习曲线。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

370

2024.04.10

laravel入门教程
laravel入门教程

本专题整合了laravel入门教程,想了解更多详细内容,请阅读专题下面的文章。

81

2025.08.05

laravel实战教程
laravel实战教程

本专题整合了laravel实战教程,阅读专题下面的文章了解更多详细内容。

64

2025.08.05

laravel面试题
laravel面试题

本专题整合了laravel面试题相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.08.05

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

231

2023.09.22

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

6

2026.01.15

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 8.7万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 7万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号