laravel 基础教程 —— Eloquent

Eloquent: 起步

简介

Laravel 的 Eloquent ORM 提供了一种漂亮简洁的关系映射的模型来与数据库进行交互。所有的数据库表都有相应的模型,这些模型被用来与表进行交互。模型允许你直接查询数据库表中的数据,及插入新的记录到数据表中。

在开始之前,你需要确保完成了 config/database.php 配置文件中的数据库配置。对于更多的配置数据库相关的信息,请参考 文档

定义模型

在开始之前,让我们先来创建一个 Eloquent 模型。模型通常存放在 app 目录下,但是你也可以自由的放置在任何地方,只要它能够根据你的 composer.json 文件的指导进行自动的加载。所有的 Eloquent 模型都继承自 Illuminate\Database\Eloquent\Model 类。

最简单的创建一个模型类的方式就是使用 make:model Artisan 命令:

php artisan make:model User

如果你希望在你生成模型的同时生成一份数据库迁移,你可以使用 --migration 或者 -m 选项:

php artisan make:model User --migration

php artisan make:model User -m

Eloquent 模型约定

现在,让我们来看一个 Flight 模型的示例,我们将通过它获取和存储数据库中 flights 表的数据:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Flight extends Model
{
  //
}

表名称

你需要注意我们上面的代码中并没有指出 Flight 模型使用哪个数据库的表。如果你没有明确的指出模型所对应的表,那么 Eloquent 将使用类的蛇形命名的复数形式来使用相应的数据表。所以,在这个例子中,Eloquent 将会假定 Flight 模型存储的记录被放在 flights 表中。你可以使用 table 属性来指定表名:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Flight extends Model
{
  /**
   * The table associated with the model.
   *
   * @var string
   */
   protected $table = 'my_flights';
}

主键

Eloquent 也会假定所有表的主键列名为 id。你也可使用 $primaryKey 属性来手动的指定表的主键。

另外,Eloquent 也会假定主键是一个自增的整型值,这意味着,在默认情况下主键会被自动的转为 int。如果你希望使用非自增或者非数字的主键,那么你需要在模型中将公开的 $incrementing 属性设置为 false

时间戳

默认的,Eloquent 期望模型表中存在 created_atupdated_at 列,如果你不希望 Eloquent 自主的管理这两列,你可以在模型中设置 $timestamps 属性为 false:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Flight extends Model
{
  /**
   * Indicates if the model should be timestamped.
   *
   * @var bool
   */
   public $timestamps = false;
}

如果你需要自定义时间戳的格式,你可以设置 $dateFormat 属性。这个属性用来决定日期属性存储在数据库中的格式,以及模型在进行序列化为数组或者 JSON 时的显示样式:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Flight extends Model
{
  /**
   * The storage format of the model's date columns.
   *
   * @var string
   */
   protected $dateFormat = 'U';
}

数据库连接

默认的,所有的 Eloquent 模型都会使用应用配置的默认的数据库连接。如果你希望模型使用不同的数据库连接,你可以使用 $connection 属性:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Flight extends Model
{
  /**
   * The connection name for the model.
   *
   * @var string
   */
   protected $connection = 'connection-name';
}

检索多个模型

一旦你创建了模型和其相应的数据表,你就为从数据库中检索数据做好了准备。你可以把 Eloquent 模型作为一个强大的查询生成器来使用,它允许通过模型流畅的查询相应的表中的数据。比如:

<?php

namespace App\Http\Controllers;

use App\Flight;
use App\Http\Controllers\Controller;

class FlightController extends Controller
{
  /**
   * Show a list of all available flights.
   *
   * @return Response
   */
  public function index()
  {
    $flights = Flight::all();

    return view('flight.index', ['flights' => $flights]);
  }
}

访问列的值

如果你获得了 Eloquent 模型的实例,那么你可以通过模型中相应的列名的属性来访问表中列的值。比如,让我们循环查询的结果,并通过获取的 Flight 实例来获得 name 列的值:

foreach ($flights as $flight) {
  echo $flight->name;
}

添加额外的条件

Eloquent 的 all 方法会返回模型表中所有的记录。由于所有的 Eloquent 模型都可以作为查询生成器来进行服务,所以你可以在这些查询中增加额外的条件,然后使用 get 方法来检索结果:

$flights = App\Flight::where('active', 1)
             ->orderBy('name', 'desc')
             ->take(10)
             ->get();

注意:由于 Eloquent 模型也是查询生成器,所以你可以回顾一下查询生成器中所有可用的方法,因为它们同样可以在 Eloquent 中进行使用。

集合

对于像 allget 这样的 Eloquent 方法会返回多个结果,这将返回一个 Illuminate\Database\Eloquent\Collection 实例。Collection 类提供了各种有用的方法与 Eloquent 结果进行交互。当然,你可以简单的对集合进行循环,因为他实现了 ArrayAccess 接口:

foreach ($flights as $flight) {
  echo $flight->name;
}

对结果进行分块

如果你需要处理数千条 Eloquent 记录,那么你可以使用 chunk 方法。chunk 方法会从 Eloquent 模型中检索出一小块记录,然后将其传递给 Closure 进行处理。使用 chunk 方法在处理大量结果集时可有有效的降低内存的消耗:

Flight::chunk(200, function ($flights) {
  foreach ($flights as $flight) {
    //
  } 
});

传递到方法的第一个参数是你要设置进行分块的大小。而传递到第二个参数的闭包将会在从数据库检索出每块内容时进行调用。

检索单个模型 / 统计

当然,除了可以从表中检索出所有的记录之外,你也可以使用 findfirst 方法来从数据库中检索出单条的数据。这将会返回一个单独的模型实例:

// Retrieve a model by its primary key...
$flight = App\Flight::find(1);

// Retrieve the first model matching the query constraints...
$flight = App\Flight::where('active', 1)->first();

你也可以使用 find 方法时传递一个包含主键所组成的数组,这将返回所有匹配到记录的集合:

$flights = App\Flight::find([1, 2, 3]);

未发现时的异常

有时候,你可能希望在没有找到匹配的模型时抛出一个异常。这对路由或者控制器来说尤其有用。findOrFailfirstOrFail 方法可以从查询中检索出首个匹配的结果,但是,如果结果中没有匹配的项,那么会抛出一个 Illuminate\Database\Eloquent\ModelNotFoundException 异常:

$model = App\Flight::findOrFail(1);

$model = App\Flight::where('legs', '>', 100)->firstOrFail();

如果异常没有被捕获,那么会直接传递给用户一个 404 HTTP 响应。所以当你在使用这些方法时是没有必要编写明确的检查并返回 404 响应的:

Route::get('/api/flights/{id}', function ($id) {
  return App\Flight::findOrFail($id); 
});

检索统计

当然,你可以使用 countsummax 和其他 查询生成器 所提供的的 聚合函数。这些方法会返回适当的数值,而不是完整的模型实例:

$count = App\Flight::where('active', 1)->count();

$max = App\Flight::where('active', 1)->max('price');

插入 & 更新模型

基础插入

想要在数据库中插入新的记录,只需要简单的创建模型实例,然后设置模型的属性,再调用 save 方法就可以了:

<?php

namespace App\Http\Controllers;

use App\Flight;
use Illuminate\Http\Request;
use App\Http\Controllers\Controller;

class FlightController extends Controller
{
  /**
   * Create a new flight instance.
   *
   * @param Request $request
   * @return Response
   */
   public function store(Request $request)
   {
     // Validate the request...

     $flight = new Flight;

     $flight->name = $request->name;

     $flight->save();
   }
}

在这个例子中,我们简单的从流入的 HTTP 请求中的 name 参数分配到 App\Flight 模型实例的 name 属性上。当我们调用 save 方法时,这条记录将会插入到数据库中,同时 created_atupdated_at 时间戳也会自动的被进行设置,所以这并不需要进行手动的设置它们。

基础的更新

save 方法也可用来更新数据库中已经存在的数据模型。为了更新模型,你应该首先检索到它们,然后设置任何你想要更新的属性,接着调用 save 方法。这一次,updated_at 时间戳会自动的进行更新,所以你并不需要手动的更新这个值:

$flight = App\Flight::find(1);

$flight->name = 'New Flight Name';

$flight->save();

你也可以针对查询匹配的多个模型进行更新操作。比如,所有 active 并且 destinationSan Diego 的航班将会被标记为延迟:

App\Flight::where('active', 1)
          ->where('destination', 'San Diego')
          ->update(['delayed' => 1]);

update 方法接收一个想要更新的列的键值对数组。

批量赋值

你也可以使用 create 方法来在一行中存储一个新的模型。一个新增的模型实例将会从方法中返回。事实上,在开始做这些之前,你需要先指定模型的 fillable 或者 guarded 属性,它们是针对批量赋值时的保护措施。

当用户通过 HTTP 请求传递一些意外的参数时,这可能会造成批量赋值时一些问题的出现,或许参数中存在一些你并不想改变的列值的参数。比如,一个恶意的用户可能会在请求中嵌入一个 is_admin 参数,然后这些参数映射到你的 create 方法中,这就使用户将自己升级为了超级管理员。

所以,在开始之前,你需要先定义哪些模型属性是可以进行批量分配的。你可以使用模型 $fillable 属性,让我们在 Flight 模型中添加允许 name 属性的批量赋值:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Dlight extends Model
{
  /**
   * The attributes that are mass assignable.
   *
   * @var array
   */
   protected $fillable = ['name'];
}

一旦我们使一个属性可以进行批量赋值。那么我们可以使用 create 方法直接的添加记录到数据中。create 方法会返回存储后的模型实例:

$flight = App\Flight::create(['name' => 'Flight 10']);

$fillable 就像是为批量赋值提供了一种白名单的机制。你也可以选择使用 $guarded$guarded 属性应该包含一个你不想要进行批量赋值的属性所组成的数组。所有不在这个数组中的属性都将可以进行批量赋值。所以,$guarded 就像一个黑名单的功能。当然,你应该只使用 $fillable 或者 $guarded 中的一个,而不是全部:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Flight extends Model
{
  /**
   * The attributes that aren't mass assignable.
   *
   * @var array
   */
   protected $guarded = ['price'];
}

在上面的例子中,除了 price,所有的属性都可以进行批量赋值。

其他创建方法

这里也有其他两种方法可以使用批量赋值进行模型的创建:firstOrCreatefirstOrNewfirstOrCreate 方法会尝试根据给定的列的键值对从数据库定位相关的模型。如果相关的模型没有找到,那么会根据给定的属性在数据库中新增一条记录。

firstOrNew 方法和 firstOrCreate 方法一样,但是它在模型没找到时只会返回一个新的模型实例,并不会在数据库中新增一条记录。你需要手动的使用 save 方法来将其存储到数据库中:

// Retrieve the flight by the attributes, or create it if it doesn't exists...
$flight = App\Flight::firstOrCreate(['name' => 'Flight 10']);

// Retrieve the flight by the attributes, or instantiate a new instance...
$flight = App\Flight::firstOrNew(['name' => 'Flight 10']);

删除模型

你可以在模型实例中使用 delete 方法来删除模型:

$flight = App\Flight::find(1);

$flight->delete();

根据键删除存在的模型

在上面的例子中,我们先从数据库中检索出相关的模型,然后才调用 delete 方法来进行删除。事实上,如果你知道了模型的主键,那么你完全可以不用去检索到它。你可以直接使用 destroy 方法来进行删除:

App\Flight::destroy(1);

App\Flight::destroy([1, 2, 3]);

App\Flight::destroy(1, 2, 3);

通过查询删除模型

当然,你也可以通过查询删除一个模型集。在这个例子中,我们将删除所有标记为未启用的航班:

$deletedRows = App\Flight::where('active', 0)->delete();

软删除

除了从数据库中真实的删除数据之外,Eloquent 也可以进行软删除操作。当模型是一个软删除模型时,它们并不会真正的从数据库中清除记录。实际上,它们会将模型的 deleted_at 属性进行设置,并且将其更新到数据库中。如果模型中的 deleted_at 的值不是 NULL,那么它就被标记为软删除了。如果你需要启用软删除模型,你需要在模型中引入 Illuminate\Database\Eloquent\SoftDeletes trait,并且在模型的 $dates 属性中添加 deleted_at 列:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\SoftDeletes;

class Flight extends Model
{
  use SoftDeletes;

  /**
   * The attributes that should be mutated to dates.
   *
   * @var array
   */
   protected $dates = ['deleted_at'];
}

当然,你应该在数据表中添加 deleted_at 列。Laravel 的 结构生成器 中提供了生成该列的方法:

Schema::table('flights', function ($table) {
  $table->softDeletes(); 
});

现在,当我们调用 delete 方法时,deleted_at 列会被设置为当前时间。并且,当查询启用软删除的模型时,已经被软删除的模型将自动从结果中剔除。

你可以使用 trashed 方法来判断所给定的模型是否已经被软删除:

if ($flight->trashed()) {
  //
}

查询软删除的模型

包含软删除的模型

就如上面我们所提到的,被软删除的模型将自动的从结果中进行剔除。事实上,你可以使用 withTrashed 方法来强制结果中显示已经被软删除的模型:

$flights = App\Flight::withTrashed()
             ->where('account_id', 1)
             ->get();

whitTrashed 方法也可以在关联查询中进行使用:

$flight->history()->withTrashed()->get();

只检索被软删除的模型

onlyTrashed 方法会从数据库中检索被软删除的模型记录:

$flights = App\Flight::onlyTrashed()
             ->where('airline_id', 1)
             ->get();

还原被软删除的模型

有时候你可能会希望还原已经被软删除的模型,你可以使用 restore 方法来将模型从软删除中解除:

$flight->restore();

你也可以在查询中使用 restore 方法来快速的重启多个模型:

App\Flight::withTrashed()
  ->where('airline_id', 1)
  ->restore();

就像 withTrashed 方法一样,restore 方法也可以在关联查询中使用:

$flight->history()->restore();

永久的删除模型

有时候你可能希望从数据库中直接删除这个模型。你可以使用 forceDelete 方法来将模型从数据库中永久的删除:

// Force deleting a single model instance...
$flight->forceDelete();

// Force deleting all related modls...
$flight->history()->forceDelete();

查询区间

全局区间

全局区间允许你在给定的模型上的所有查询进行约束的添加。Laravel 自己的软删除功能就是利用了全局区间来从数据库中拉取未删除的数据。编写自己的全局区间可以方便的对给定模型的所有查询进行约束。

编写全局区间

编写一个全局区间非常简单,你只需要定义一个实现了 Illuminate\Database\Eloquent\Scope 接口的类。这个接口只要求你实现一个方法:applyapply 方法可以在需要的时添加 where 约束:

<?Php

namespace App\Scopes;

use Illuminate\Database\Eloquent\Scope;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Builder;

class AgeScope implements Scope
{
  /**
   * Apply the scope to a given Eloquent query builder.
   *
   * @param \Illuminate\Database\Eloquent\Builder $builder
   * @param \Illuminate\Database\Eloquent\Model $model
   * @return void
   */
   public function apply(Builder $builder, Model $model)
   {
     return $builder->where('age', '>', 200);
   }
}

Laravel 没有为区间预置存放的目录,所以你可以自由的创建自己的 Scopes 目录来进行管理。

应用全局区间

你需要在给定的模型中复写 boot 方法并且使用 addGlobalScope 方法来分配全局区间:

<?php

namespace App;

use App\Scopes\AgeScope;
use Illuminate\Database\Eloquent\Model;

class User extends Model
{
  /**
   * The "booting" method of the model
   *
   * @return void
   */
   protected static function boot()
   {
     paraent::boot();

     static::addGlobalScope(new AgeScope);
   }
}

在添加完区间之后,调用 User::all() 的查询会产生下述的 SQL:

select * from `users` where `age` > 200

匿名全局区间

Eloquent 也允许你通过一个闭包来定义全局区间,这通常对于不需要分离到单独一个类文件中的简单区间尤其有用:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Builder;

class User extends Model
{
  /**
   * The "booting" method of the model.
   *
   * @return void
   */
   protected static function boot()
   {
     parent::boot();

     static::addGlobalScope('age', function (Builder $builder) {
       $builder->where('age', '>', 200);
     });
   }
}

传递到 addGlobalScope() 方法的首个参数将作为区间的唯一标识,你可以通过标识将其排除:

User::withoutGlobalScope('age')->get();

删除全局区间

如果你希望从给定的查询中移除全局区间,你可以使用 withoutGlobalScope 方法:

User::withoutGlobalScope(AgeScope::class)->get();

如果你希望删除多个或者全部的全局区间,你可以使用这么使用 withoutGlobalScopes 方法:

User::withoutGlobalScopes()->get();

User::withoutGlobalScopes([FirstScope::class, SecondScope::class])->get();

当前区间

当前区间允许你在模型中定义一些可以复用的常用的约束。比如,你可能经常需要检索一些受欢迎的用户。你可以简单的在模型方法中前置 scope 命名来定义一个区间.

区间应该总是返回一个查询生成器的实例:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class User extends Model
{
  /**
   * Scope a query to only include popular users.
   *
   * @return \Illuminate\Database\Eloquent\Builder
   */
   public function scopePopular($query)
   {
     return $query->where('votes', '>', 100);
   }

   /**
    * Scope a query to only include active users.
    *
    * @return \Illuminate\Database\Eloquent\Builder
    */
    public function scopePopular($query)
    {
      return $query->where('votes', '>', 100);
    }

    /**
     * Scope a query to only include active users.
     *
     * @return \Illuminate\Database\Eloquent\Builder
     */
     public function scopeActive($query)
     {
       return $query->where('active', 1);
     }
}

利用区间查询

一旦区间进行了定义,你可以在模型进行查询时调用区间方法,事实上,你在调用区间方法时并不需要包含 scope 前缀。你甚至可以链式的调用其它的区间:

$users = App\User::popular()->active()->orderBy('created_at')->get();

动态区间

有时候你可能希望定义一个可以接收参数的区间。在开始之前,你仅仅只需要在你的区间中添加一些额外的参数。区间参数应该在 $query 参数之后进行定义:

<?php

namespace App;

use Illuminate\Database\Eloquent\Model;

class User extends Model
{
  /**
   * Scope a query to only include users of a given type.
   *
   * @return \Illuminate\Database\Eloquent\Builder
   */
   public function scopeOfType($query, $type)
   {
     return $query->where('type', $type);
   }
}

现在,你可以在调用区间时传递一些参数了:

$users = App\User::ofType('admin')->get();

事件

Eloquent 模型可以触发多种事件,这允许你在模型的生命周期的各个关键点进行 hook 操作。你可以使用下面的方法进行 Hook:creatingcreatedupdatingupdatedsavingsaveddeletingdeletedrestoringrestored。事件允许你轻松的在模型进行存储或更新操作时进行执行额外的操作。

基础用法

当一个新的模型首次进行存储操作时,会触发 creatingcreated 事件。如果模型已经存在于数据库中,并且调用 save 方法,那么 updating / updated 事件将会被触发。事实上,在这两种情况下,savingsaved 事件都会被触发。

举个示例,让我们在服务提供者中定义一个 Eloquent 事件监听器。在我们的事件监听器中,我们将在给定的模型中调用 isValid 方法,当模型并没有通过验证时将返回 false。如果从 Eloquent 事件监听器中返回 false,那么将取消 save / update 操作:

<?php

namespace App\Providers;

use App\User;
use Illuminate\Support\ServiceProvider;

class AppServiceProvider extends ServiceProvider
{
  /**
   * Bootstrap any application services.
   *
   * @return void
   */
   public function boot()
   {
     User::creating(function ($user) {
       if (! $user->isValid()) {
         return false;
       }
     });
   }

   /**
    * Register the service provider.
    *
    * @return void
    */
    public function register()
    {
      //
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容