Laravel 5使用faker Model Factory生成网站模拟演示数据

开发中,假如我需要一些演示的数据,那么,我可能得连接数据库,创建一大堆的演示数据,这是一件非常蛋疼的事情。还好,Laravel是优雅高效的,Laravel绝对不会干很蠢的事情。

按照大多数人的做法,按照令人繁琐的方式创建数据,比如我需要创建1000个用户,普通的做法就得按照下面的方式一个个追加,或者聪明点弄个循环,但假如我用户下面还有文章,岂不是更麻烦:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
// manually make a list of users
$users = [
     [
         'username' => 'firstuser' ,
         'name' => 'first user' ,
         'email' => '[email protected]'
     ],
     [
         'username' => 'seconduser' ,
         'name' => 'second user' ,
         'email' => '[email protected]'
     ]
     ...
];

这都是非常愚蠢的做法,浪费时间!多亏了 fzaninotto/faker 这个库,我们可以非常简单的创建一大堆模拟数据。

这个库可以在packagist下载,所以,我们可以通过composer下载下来。不过,从Laravel 5之后,就不需要自己下载安装了,因为Laravel直接内置了这个库。

?
1
composer require fzaninotto/faker --dev

安装好fzaninotto/faker 库,我们可以通过Laravel的DB seeder class进行操作,几行代码就创建我们需要的模拟数据。具体如何使用 Laravel 的 DB seeder class 操作,参考这篇文章https://laravel.com/docs/5.2/seeding:

?
1
2
3
4
5
6
7
8
9
10
11
public function run() {
     $faker = Faker\Factory::create();
 
     for ( $i = 0; $i < 1000; $i ++) {
         App\User::create([
             'username' => $faker ->userName,
             'name' => $faker ->name,
             'email' => $faker ->email
         ]);
     }
}

通过上面代码段,我们就可以创建1000个用户。

虽然上面的办法非常凑效,可以把你从最初的愚蠢做法解围出来,但是这并不是最佳的解决方案,我们总不可能每次都把上面的代码写一遍运行一次创建用户。

Laravel的开发原则是,永远保持你的代码优雅。那么我们可以按照下面的方式操作。

#Model Factories

模型工厂给我们提供了非常牛逼的方式去生成数据。自从Laravel 5版本之后,模型工厂已经成为Laravel的标配。

如何使用呢?打开Laravel项目下文件:database/factories/ModelFactory.php

我们可以创建一个所谓的工厂。Laravel提供了一个全局的对象 $factory 用来定义我们的工厂,比如:

?
1
2
3
4
5
6
7
$factory ->define(App\User:: class , function (Faker\Generator $faker ) {
     return [
          'username' => $faker ->userName,
         'email' => $faker ->email,
         'name' => $faker ->name
     ];
});

从上面代码可以看到,define方法有两个参数,一个是对象模型类,另一个是一个带了Faker\Generator类的闭包,返回用户对象数组。

定义好工厂,那么如何使用呢?

#Using the Factory

上一步我们定义好了我们的工厂,那么我们便可以随时使用,不过我们往往会在 测试脚本里 或者 seed 类里进行使用。我通过 factory 函数便可以使用工厂了,如下:

?
1
2
// create a user and save them to the database
$user = factory(App\User:: class )->create();

上面代码创建了一个用户,直接在数据库里生效的哦,假如我们需要创建大量的用户,直接给 factory 函数传递第二个参数即可。

?
1
2
// create 1000 users with just one line
$users = factory(App\User:: class , 1000)->create();

#Overriding Attributes

如果你想覆盖在工厂里定义的对象属性值,那么可以在create()方法里传递一个数组参数,定义你需要改变的属性值,这不会影响其他默认属性值,依然保持工厂里定义的属性值。

?
1
2
3
$user = factory(App\User:: class )->create([
     'username' => 'pizzamuncher'
]);

#Factory Make

有时候我们并不需要把数据存到数据库里,只是想通过工厂制造点数据查看下,那么,我们可以使用make 方法,而不是 create 方法。

?
1
$user = factory(App\User:: class )->make();

就像create方法一样,我们依然可以自定义一些属性值:

?
1
2
3
$user = factory(App\User:: class )->make([
     'username' => 'pitzanotpizza'
]);

#Multiple Factory Types

一个对象模型可能有不同的类型,我们可以定义对象工厂使用不同的类型。比如:我们希望一些用户成为管理员,我们可以使用 $factory->defineAs()方法,它具有3个参数,第一个是对象模型,第二个是类型,第三个是闭包(定义了属性值)。

?
1
2
3
4
5
6
7
8
$factory ->defineAs(App\User:: class , 'admin' , function (Faker\Generator $faker ) {
     return [
          'username' => $faker ->userName,
         'email' => $faker ->email,
         'name' => $faker ->name,
         'admin' => true
     ];
});

或者假如你希望拓展下模型,可以这么做:

?
1
2
3
4
5
$factory ->defineAs(App\User:: class , 'admin' , function ( $faker ) use ( $factory ) {
     $post = $factory ->raw( 'App\User' );
 
     return array_merge ( $post , [ 'admin' => true]);
});

定义好了上面的代码,我们便可以通过 factory 方法,直接生成 管理员 类的用户了,比如:

?
1
factory(App\User:: class , 'admin' )->create();
?
1
factory(App\User:: class , 'admin' , 10)->create();

#Create a Model with Relation

下面我们再来看另外一种需求,每个用户可能发布有不同(hasMany)的文章,那么这种带有对象关系的模型,如何在魔星工厂里使用呢?假如我们需要每个用户下面包含文章。那么,没问题的,我们可以这么做:创建用户,遍历用户,插入post保存即可。

?
1
2
3
factory(App\User:: class , 50)->create()->each( function ( $u ) {
     $u ->posts()->save(factory(App\Post:: class )->make());
});

#Using Model Factories While Testing

在测试的时候,我们便可以随心所欲的使用模型工厂生成数据了。假如你修改了你的属性字段,那么你需要做的仅仅是更新下工厂属性定义就可以了。

下面是两段使用模型工厂的测试脚本,可以参考学习下。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
 
use Illuminate\Foundation\Testing\WithoutMiddleware;
use Illuminate\Foundation\Testing\DatabaseMigrations;
use Illuminate\Foundation\Testing\DatabaseTransactions;
 
class ExampleTest extends TestCase
{
     public function setUp() {
         parent::setUp();
 
         Artisan::call( 'migrate' ); //run migrations
         Eloquent::unguard(); // disable eloquent guard
     }
 
     public function it_creates_at_least_hundred_fake_users() {
         $users = factory(App\Users:: class , mt_rand(100, 1000))->create();
         $user_count = count ( $users ) >= 100;
 
         $this ->assertTrue( $user_count );
     }
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public function it_creates_at_least_hundred_fake_users() {
     $users = [];
     $faker = Faker\Factory::create();
 
     for ( $i = 0; $i < mt_rand(100, 1000); $i ++) {
         $users [] = App\User::create([
             'username' => $faker ->userName,
             'email' => $faker ->email,
             'name' => $faker ->name
         ]);
     }
 
     $this ->assertTrue( count ( $users ) >= 100);
}

如果需要doc命令执行的情况下
php artisan tinker

factory(App\User::class,2)->create();//创建2条记录



感谢博主:http://www.jyguagua.com/?p=2573

你可能感兴趣的:(PHP)