Skip to content
赞助商
虚位以待
赞助商
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待

Laravel Scout

介绍

Laravel Scout 为您的 Eloquent 模型 提供了一种简单的、基于驱动程序的解决方案,用于添加全文搜索。使用模型观察者,Scout 将自动保持您的搜索索引与 Eloquent 记录同步。

目前,Scout 附带一个 Algolia 驱动程序;然而,编写自定义驱动程序很简单,您可以自由扩展 Scout 以实现自己的搜索实现。

安装

首先,通过 Composer 包管理器安装 Scout:

php
composer require laravel/scout

接下来,您应该将 ScoutServiceProvider 添加到 config/app.php 配置文件的 providers 数组中:

php
Laravel\Scout\ScoutServiceProvider::class,

在注册 Scout 服务提供者之后,您应该使用 vendor:publish Artisan 命令发布 Scout 配置。此命令将 scout.php 配置文件发布到您的 config 目录:

php
php artisan vendor:publish --provider="Laravel\Scout\ScoutServiceProvider"

最后,将 Laravel\Scout\Searchable trait 添加到您希望进行搜索的模型中。此 trait 将注册一个模型观察者,以保持模型与您的搜索驱动程序同步:

php
<?php

namespace App;

use Laravel\Scout\Searchable;
use Illuminate\Database\Eloquent\Model;

class Post extends Model
{
    use Searchable;
}

队列

虽然使用 Scout 并不严格要求,但在使用该库之前,您应该强烈考虑配置一个 队列驱动程序。运行队列工作程序将允许 Scout 将所有操作排队,以将您的模型信息同步到您的搜索索引,从而为您的应用程序的 Web 界面提供更好的响应时间。

一旦您配置了队列驱动程序,请将 config/scout.php 配置文件中的 queue 选项的值设置为 true

php
'queue' => true,

驱动程序先决条件

Algolia

使用 Algolia 驱动程序时,您应该在 config/scout.php 配置文件中配置您的 Algolia idsecret 凭据。配置好凭据后,您还需要通过 Composer 包管理器安装 Algolia PHP SDK:

php
composer require algolia/algoliasearch-client-php

配置

配置模型索引

每个 Eloquent 模型都与给定的搜索“索引”同步,该索引包含该模型的所有可搜索记录。换句话说,您可以将每个索引视为一个 MySQL 表。默认情况下,每个模型将被持久化到与模型的典型“表”名称匹配的索引中。通常,这是模型名称的复数形式;但是,您可以通过覆盖模型上的 searchableAs 方法来自定义模型的索引:

php
<?php

namespace App;

use Laravel\Scout\Searchable;
use Illuminate\Database\Eloquent\Model;

class Post extends Model
{
    use Searchable;

    /**
     * 获取模型的索引名称。
     *
     * @return string
     */
    public function searchableAs()
    {
        return 'posts_index';
    }
}

配置可搜索数据

默认情况下,给定模型的整个 toArray 形式将被持久化到其搜索索引中。如果您想自定义同步到搜索索引的数据,可以覆盖模型上的 toSearchableArray 方法:

php
<?php

namespace App;

use Laravel\Scout\Searchable;
use Illuminate\Database\Eloquent\Model;

class Post extends Model
{
    use Searchable;

    /**
     * 获取模型的可索引数据数组。
     *
     * @return array
     */
    public function toSearchableArray()
    {
        $array = $this->toArray();

        // 自定义数组...

        return $array;
    }
}

索引

批量导入

如果您正在将 Scout 安装到现有项目中,您可能已经有需要导入到搜索驱动程序的数据库记录。Scout 提供了一个 import Artisan 命令,您可以使用它将所有现有记录导入到您的搜索索引中:

php
php artisan scout:import "App\Post"

添加记录

一旦您将 Laravel\Scout\Searchable trait 添加到模型中,您只需 save 一个模型实例,它就会自动添加到您的搜索索引中。如果您已将 Scout 配置为 使用队列,此操作将由您的队列工作程序在后台执行:

php
$order = new App\Order;

// ...

$order->save();

通过查询添加

如果您希望通过 Eloquent 查询将一组模型添加到您的搜索索引中,您可以将 searchable 方法链接到 Eloquent 查询上。searchable 方法将 分块查询结果 并将记录添加到您的搜索索引中。同样,如果您已将 Scout 配置为使用队列,所有分块将由您的队列工作程序在后台添加:

php
// 通过 Eloquent 查询添加...
App\Order::where('price', '>', 100)->searchable();

// 您还可以通过关系添加记录...
$user->orders()->searchable();

// 您还可以通过集合添加记录...
$orders->searchable();

searchable 方法可以被视为“插入或更新”操作。换句话说,如果模型记录已经在您的索引中,它将被更新。如果它不存在于搜索索引中,它将被添加到索引中。

更新记录

要更新可搜索模型,您只需更新模型实例的属性并将模型 save 到数据库。Scout 将自动将更改持久化到您的搜索索引:

php
$order = App\Order::find(1);

// 更新订单...

$order->save();

您还可以在 Eloquent 查询上使用 searchable 方法来更新一组模型。如果模型不存在于您的搜索索引中,它们将被创建:

php
// 通过 Eloquent 查询更新...
App\Order::where('price', '>', 100)->searchable();

// 您还可以通过关系更新...
$user->orders()->searchable();

// 您还可以通过集合更新...
$orders->searchable();

删除记录

要从索引中删除记录,只需从数据库中 delete 模型即可。这种删除形式甚至与 软删除 模型兼容:

php
$order = App\Order::find(1);

$order->delete();

如果您不想在删除记录之前检索模型,可以在 Eloquent 查询实例或集合上使用 unsearchable 方法:

php
// 通过 Eloquent 查询删除...
App\Order::where('price', '>', 100)->unsearchable();

// 您还可以通过关系删除...
$user->orders()->unsearchable();

// 您还可以通过集合删除...
$orders->unsearchable();

暂停索引

有时您可能需要对模型执行一批 Eloquent 操作,而无需将模型数据同步到您的搜索索引。您可以使用 withoutSyncingToSearch 方法来执行此操作。此方法接受一个将立即执行的回调。回调中发生的任何模型操作都不会同步到模型的索引:

php
App\Order::withoutSyncingToSearch(function () {
    // 执行模型操作...
});

搜索

您可以使用 search 方法开始搜索模型。搜索方法接受一个将用于搜索您的模型的字符串。然后,您应该将 get 方法链接到搜索查询上,以检索与给定搜索查询匹配的 Eloquent 模型:

php
$orders = App\Order::search('Star Trek')->get();

由于 Scout 搜索返回 Eloquent 模型的集合,您甚至可以直接从路由或控制器返回结果,它们将自动转换为 JSON:

php
use Illuminate\Http\Request;

Route::get('/search', function (Request $request) {
    return App\Order::search($request->search)->get();
});

Where 子句

Scout 允许您向搜索查询添加简单的“where”子句。目前,这些子句仅支持基本的数字相等性检查,主要用于通过租户 ID 限制搜索查询。由于搜索索引不是关系数据库,目前不支持更高级的“where”子句:

php
$orders = App\Order::search('Star Trek')->where('user_id', 1)->get();

分页

除了检索模型集合,您还可以使用 paginate 方法对搜索结果进行分页。此方法将返回一个 Paginator 实例,就像您 对传统 Eloquent 查询进行分页 一样:

php
$orders = App\Order::search('Star Trek')->paginate();

您可以通过将数量作为第一个参数传递给 paginate 方法来指定每页要检索多少个模型:

php
$orders = App\Order::search('Star Trek')->paginate(15);

一旦您检索到结果,您可以显示结果并使用 Blade 渲染页面链接,就像您对传统 Eloquent 查询进行分页一样:

php
<div class="container">
    @foreach ($orders as $order)
        {{ $order->price }}
    @endforeach
</div>

{{ $orders->links() }}

自定义引擎

编写引擎

如果内置的 Scout 搜索引擎之一不符合您的需求,您可以编写自己的自定义引擎并将其注册到 Scout。您的引擎应扩展 Laravel\Scout\Engines\Engine 抽象类。此抽象类包含五个方法,您的自定义引擎必须实现:

php
use Laravel\Scout\Builder;

abstract public function update($models);
abstract public function delete($models);
abstract public function search(Builder $builder);
abstract public function paginate(Builder $builder, $perPage, $page);
abstract public function map($results, $model);

您可能会发现查看 Laravel\Scout\Engines\AlgoliaEngine 类上这些方法的实现很有帮助。此类将为您提供一个良好的起点,以了解如何在自己的引擎中实现这些方法。

注册引擎

一旦您编写了自定义引擎,您可以使用 Scout 引擎管理器的 extend 方法将其注册到 Scout。您应该从 AppServiceProviderboot 方法或应用程序使用的任何其他服务提供者中调用 extend 方法。例如,如果您编写了一个 MySqlSearchEngine,您可以这样注册它:

php
use Laravel\Scout\EngineManager;

/**
 * 启动任何应用程序服务。
 *
 * @return void
 */
public function boot()
{
    resolve(EngineManager::class)->extend('mysql', function () {
        return new MySqlSearchEngine;
    });
}

一旦您的引擎注册完毕,您可以在 config/scout.php 配置文件中将其指定为默认的 Scout driver

php
'driver' => 'mysql',