婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Laravel學習筆記之Artisan命令生成自定義模板的方法

Laravel學習筆記之Artisan命令生成自定義模板的方法

熱門標簽:亳州企業外呼系統 400 電話 辦理 打開百度地圖標注 山東電銷卡外呼系統原理是什么 蘇州外呼系統有效果嗎 兼職做地圖標注好賺錢嗎 海南外呼系統方案 智能電銷語音機器人資訊 地圖標注怎么做商戶驗證

說明:本文主要講述Laravel的Artisan命令來實現自定義模板,就如經常輸入的php artisan make:controller ShopController就會自動生成一個ShopController.php模板文件一樣,通過命令生成模板也會提高開發效率。同時,作者會將開發過程中的一些截圖和代碼黏上去,提高閱讀效率。

備注:個人平時在寫Repository代碼時會這樣寫,如先寫上ShopRepositoryInterface并定義好接口方法如all()create()update()delete()findBy()等等,然后再寫上接口對應的實現ShopRepository并注入對應的Model即Shop。別的PostRepository、TagRepository也會是這么寫(當然,對于很多重用的Repository方法可以集體拿到AbstractRepository抽象類里供子類繼承,實現代碼復用)。那能不能直接命令行生成模板文件呢,就不用自己一個個的寫了,就像輸入php artisan make:controller PostController給我一個Controller模板來。

關于使用Repository模式來封裝下Model邏輯,不讓Controller里塞滿了很多Model邏輯,這樣做是有很多好處的,最主要的就是好測試和代碼架構清晰,也符合SOLID原則。如果使用PHPUnit來做測試就知道了為啥說好測試了。SegmentFault上也有相關的文章描述。作者也打算最近新開一篇文章聊一聊這個,PHPUnit也打算過段時間聊一聊。

個人研究了下Artisan命令行,是可以的。經過開發后,結果是輸入自定義指令php artisan make:repository PostRepository --model=Post(這個option可要可不要),就會幫我生成一個PostRepositoryInterface和對應的接口實現PostRepository。

模板文件Stub

由于個人需要生成一個RepositoryInterface和對應接口實現Repository,那就需要兩個模板文件了。在resources/stubs新建兩個模板文件,以下是個人經常需要的兩個模板文件(你可以自定義):

 /**
   * @param array $columns
   * @return \Illuminate\Database\Eloquent\Collection|static[]
   */
  public function all($columns = array('*'))
  {
    return $this->$model_var_name->all($columns);
  }

  /**
   * @param int $perPage
   * @param array $columns
   * @return \Illuminate\Contracts\Pagination\LengthAwarePaginator
   */
  public function paginate($perPage = 15, $columns = array('*'))
  {
    return $this->$model_var_name->paginate($perPage, $columns);
  }

  /**
   * Create a new $model_var_name
   * @param array $data
   * @return \$model_namespace
   */
  public function create(array $data)
  {
    return $this->$model_var_name->create($data);
  }

   /**
    * Update a $model_var_name
    * @param array $data
    * @param $id
    * @return \$model_namespace
    */
  public function update($data = [], $id)
  {
    return $this->$model_var_name->whereId($id)->update($data);
  }

  /**
   * Store a $model_var_name
   * @param array $data
   * @return \$model_namespace
   */
  public function store($data = [])
  {
    $this->$model_var_name->id = $data['id'];
    //...
    $this->$model_var_name->save();
  }

  /**
   * Delete a $model_var_name
   * @param array $data
   * @param $id
   * @return \$model_namespace
   */
  public function delete($data = [], $id)
  {
    $this->$model_var_name->whereId($id)->delete();
  }

  /**
   * @param $id
   * @param array $columns
   * @return array|\Illuminate\Database\Eloquent\Collection|static[]
   */
  public function find($id, $columns = array('*'))
  {
    $$model_name = $this->$model_var_name->whereId($id)->get($columns);
    return $$model_name;
  }

  /**
   * @param $field
   * @param $value
   * @param array $columns
   * @return \Illuminate\Database\Eloquent\Collection|static[]
   */
  public function findBy($field, $value, $columns = array('*'))
  {
    $$model_name = $this->$model_var_name->where($field, '=', $value)->get($columns);
    return $$model_name;
  }

}

模板文件里包括參數,這些參數將會根據命令行中輸入的參數和選項被相應替換:

復制代碼 代碼如下:
['$repository_namespace', '$model_namespace', '$repository_interface_namespace', '$repository_interface', '$class_name', '$model_name', '$model_var_name']

Artisan命令生成Repository模板文件

生成Artisan命令并注冊

Laravel提供了Artisan命令自定義,輸入指令:

php artisan make:console MakeRepositoryCommand

然后改下簽名和描述:

// app/Console/Commands/MakeRepositoryCommand
  /**
   * The name and signature of the console command.
   *
   * @var string
   */
  protected $signature = 'make:repository {repository} {--model=}';

  /**
   * The console command description.
   *
   * @var string
   */
  protected $description = 'Make a repository and interface';

這里{repository}是必填參數并指明(選填參數加個?,就和路由參數一樣),將會被$this->argument('repository')方法捕捉到,{--model=}是選項,可填可不填,將會被$this->option('model')方法捕捉到。填上這個命令的描述,最后在Console的Kernel里注冊下命令:

// app/Console/Kernel
protected $commands = [
    // Commands\Inspire::class,
//    Commands\RedisSubscribe::class,
//    Commands\RedisPublish::class,
//    Commands\MakeTestRepositoryCommand::class,
    Commands\MakeRepositoryCommand::class,
  ];

然后輸入php artisan命令后就能看到這個make:repository命令了。

自動化生成RepositoryInterface和Repository文件

在MakeRepositoryCommand.php命令執行文件里寫上模板自動生成邏輯,代碼也不長,有些邏輯也有注釋,可看:

use Config;
use Illuminate\Console\Command;
use Illuminate\Filesystem\Filesystem;
use Illuminate\Support\Composer;

class MakeRepositoryCommand extends Command
{
  /**
   * The name and signature of the console command.
   *
   * @var string
   */
  protected $signature = 'make:repository {repository} {--model=}';

  /**
   * The console command description.
   *
   * @var string
   */
  protected $description = 'Make a repository and interface';

  /**
   * @var
   */
  protected $repository;

  /**
   * @var
   */
  protected $model;

  /**
   * Create a new command instance.
   *
   * @param Filesystem $filesystem
   * @param Composer $composer
   */
  public function __construct(Filesystem $filesystem, Composer $composer)
  {
    parent::__construct();

    $this->files  = $filesystem;
    $this->composer = $composer;
  }

  /**
   * Execute the console command.
   *
   * @return mixed
   */
  public function handle()
  {
    //獲取repository和model兩個參數值
    $argument = $this->argument('repository');
    $option  = $this->option('model');
    //自動生成RepositoryInterface和Repository文件
    $this->writeRepositoryAndInterface($argument, $option);
    //重新生成autoload.php文件
    $this->composer->dumpAutoloads();
  }

  private function writeRepositoryAndInterface($repository, $model)
  {
    if($this->createRepository($repository, $model)){
      //若生成成功,則輸出信息
      $this->info('Success to make a '.ucfirst($repository).' Repository and a '.ucfirst($repository).'Interface Interface');
    }
  }

  private function createRepository($repository, $model)
  {
    // getter/setter 賦予成員變量值
    $this->setRepository($repository);
    $this->setModel($model);
    // 創建文件存放路徑, RepositoryInterface放在app/Repositories,Repository個人一般放在app/Repositories/Eloquent里
    $this->createDirectory();
    // 生成兩個文件
    return $this->createClass();
  }

  private function createDirectory()
  {
    $directory = $this->getDirectory();
    //檢查路徑是否存在,不存在創建一個,并賦予775權限
    if(! $this->files->isDirectory($directory)){
      return $this->files->makeDirectory($directory, 0755, true);
    }
  }

  private function getDirectory()
  {
    return Config::get('repository.directory_eloquent_path');
  }

  private function createClass()
  {
    //渲染模板文件,替換模板文件中變量值
    $templates = $this->templateStub();
    $class   = null;
    foreach ($templates as $key => $template) {
      //根據不同路徑,渲染對應的模板文件
      $class = $this->files->put($this->getPath($key), $template);
    }
    return $class;
  }

  private function getPath($class)
  {
    // 兩個模板文件,對應的兩個路徑
    $path = null;
    switch($class){
      case 'Eloquent':
        $path = $this->getDirectory().DIRECTORY_SEPARATOR.$this->getRepositoryName().'.php';
        break;
      case 'Interface':
        $path = $this->getInterfaceDirectory().DIRECTORY_SEPARATOR.$this->getInterfaceName().'.php';
        break;
    }

    return $path;
  }

  private function getInterfaceDirectory()
  {
    return Config::get('repository.directory_path');
  }

  private function getRepositoryName()
  {
    // 根據輸入的repository變量參數,是否需要加上'Repository'
    $repositoryName = $this->getRepository();
    if((strlen($repositoryName)  strlen('Repository')) || strrpos($repositoryName, 'Repository', -11)){
      $repositoryName .= 'Repository';
    }
    return $repositoryName;
  }

  private function getInterfaceName()
  {
    return $this->getRepositoryName().'Interface';
  }

  /**
   * @return mixed
   */
  public function getRepository()
  {
    return $this->repository;
  }

  /**
   * @param mixed $repository
   */
  public function setRepository($repository)
  {
    $this->repository = $repository;
  }

  /**
   * @return mixed
   */
  public function getModel()
  {
    return $this->model;
  }

  /**
   * @param mixed $model
   */
  public function setModel($model)
  {
    $this->model = $model;
  }

  private function templateStub()
  {
    // 獲取兩個模板文件
    $stubs    = $this->getStub();
    // 獲取需要替換的模板文件中變量
    $templateData = $this->getTemplateData();
    $renderStubs = [];
    foreach ($stubs as $key => $stub) {
      // 進行模板渲染
      $renderStubs[$key] = $this->getRenderStub($templateData, $stub);
    }

    return $renderStubs;
  }

  private function getStub()
  {
    $stubs = [
      'Eloquent' => $this->files->get(resource_path('stubs/Repository').DIRECTORY_SEPARATOR.'Eloquent'.DIRECTORY_SEPARATOR.'repository.stub'),
      'Interface' => $this->files->get(resource_path('stubs/Repository').DIRECTORY_SEPARATOR.'repository_interface.stub'),
    ];

    return $stubs;
  }

  private function getTemplateData()
  {
    $repositoryNamespace     = Config::get('repository.repository_namespace');
    $modelNamespace        = 'App\\'.$this->getModelName();
    $repositoryInterfaceNamespace = Config::get('repository.repository_interface_namespace');
    $repositoryInterface     = $this->getInterfaceName();
    $className          = $this->getRepositoryName();
    $modelName          = $this->getModelName();

    $templateVar = [
      'repository_namespace'      => $repositoryNamespace,
      'model_namespace'        => $modelNamespace,
      'repository_interface_namespace' => $repositoryInterfaceNamespace,
      'repository_interface'      => $repositoryInterface,
      'class_name'           => $className,
      'model_name'           => $modelName,
      'model_var_name'         => strtolower($modelName),
    ];

    return $templateVar;
  }

  private function getRenderStub($templateData, $stub)
  {
    foreach ($templateData as $search => $replace) {
      $stub = str_replace('$'.$search, $replace, $stub);
    }

    return $stub;
  }

  private function getModelName()
  {
    $modelName = $this->getModel();
    if(isset($modelName)  !empty($modelName)){
      $modelName = ucfirst($modelName);
    }else{
      // 若option選項沒寫,則根據repository來生成Model Name
      $modelName = $this->getModelFromRepository();
    }

    return $modelName;
  }

  private function getModelFromRepository()
  {
    $repository = strtolower($this->getRepository());
    $repository = str_replace('repository', '', $repository);
    return ucfirst($repository);
  }

}

這里把一些常量值放在config/repository.php配置文件里了:

?php
/**
 * Created by PhpStorm.
 * User: liuxiang
 * Date: 16/6/22
 * Time: 17:06
 */

return [

  'directory_path' => 'App'.DIRECTORY_SEPARATOR.'Repositories',
  'directory_eloquent_path' => 'App'.DIRECTORY_SEPARATOR.'Repositories'.DIRECTORY_SEPARATOR.'Eloquent',
  'repository_namespace' => 'App\Repositories\Eloquent',
  'repository_interface_namespace' => 'App\Repositories',

];

運行一下看可不可以吧,這里截個圖:


It is working!!!

是可以生成RepositoryInterface和對應的接口實現文件,這里一個是加了--model選項一個沒加的,沒加的話這里第一個指令就默認Model的名稱是Shop。

生成的文件內容不截圖了,看下新生成的ShopRepository.php文件,的確是我想要的模板文件:

?php
/**
 * Created by PhpStorm.
 * User: liuxiang
 */
namespace App\Repositories\Eloquent;

use App\Shop;
use App\Repositories\ShopRepositoryInterface;

class ShopRepository implements ShopRepositoryInterface
{
  /**
   * @var \App\Shop
   */
  public $shop;

  public function __construct(Shop $shop)
  {
    $this->shop = $shop;
  }

  /**
   * @param array $columns
   * @return \Illuminate\Database\Eloquent\Collection|static[]
   */
  public function all($columns = array('*'))
  {
    return $this->shop->all($columns);
  }

  /**
   * @param int $perPage
   * @param array $columns
   * @return \Illuminate\Contracts\Pagination\LengthAwarePaginator
   */
  public function paginate($perPage = 15, $columns = array('*'))
  {
    return $this->shop->paginate($perPage, $columns);
  }

  /**
   * Create a new shop
   * @param array $data
   * @return \App\Shop
   */
  public function create(array $data)
  {
    return $this->shop->create($data);
  }

   /**
    * Update a shop
    * @param array $data
    * @param $id
    * @return \App\Shop
    */
  public function update($data = [], $id)
  {
    return $this->shop->whereId($id)->update($data);
  }

  /**
   * Store a shop
   * @param array $data
   * @return \App\Shop
   */
  public function store($data = [])
  {
    $this->shop->id = $data['id'];
    //...
    $this->shop->save();
  }

  /**
   * Delete a shop
   * @param array $data
   * @param $id
   * @return \App\Shop
   */
  public function delete($data = [], $id)
  {
    $this->shop->whereId($id)->delete();
  }

  /**
   * @param $id
   * @param array $columns
   * @return array|\Illuminate\Database\Eloquent\Collection|static[]
   */
  public function find($id, $columns = array('*'))
  {
    $Shop = $this->shop->whereId($id)->get($columns);
    return $Shop;
  }

  /**
   * @param $field
   * @param $value
   * @param array $columns
   * @return \Illuminate\Database\Eloquent\Collection|static[]
   */
  public function findBy($field, $value, $columns = array('*'))
  {
    $Shop = $this->shop->where($field, '=', $value)->get($columns);
    return $Shop;
  }

}

總結:本文主要用Laravel的Artisan命令來自動生成個人需要的模板,減少平時開發中重復勞動。就像Laravel自帶了很多模板生成命令,用起來會節省很多時間。這是作者在平時開發中遇到的問題,通過利用Laravel Artisan命令解決了,所以Laravel還是挺好玩的。有興趣的可以把代碼扒下來玩一玩,并根據你自己想要的模板做修改。這兩天想就Repository模式封裝Model邏輯的方法和好處聊一聊,到時見。希望對大家的學習有所幫助,也希望大家多多支持腳本之家

您可能感興趣的文章:
  • Laravel如何自定義command命令淺析
  • 詳細Laravel5.5執行表遷移命令出現表為空的解決方案
  • laravel通過創建自定義artisan make命令來新建類文件詳解
  • laravel 創建命令行命令的圖文教程

標簽:呼倫貝爾 綏化 清遠 安康 紹興 萊蕪 金華 溫州

巨人網絡通訊聲明:本文標題《Laravel學習筆記之Artisan命令生成自定義模板的方法》,本文關鍵詞  Laravel,學習,筆記,之,Artisan,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Laravel學習筆記之Artisan命令生成自定義模板的方法》相關的同類信息!
  • 本頁收集關于Laravel學習筆記之Artisan命令生成自定義模板的方法的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    91在线免费看| 国产精品久久777777| 男女视频一区二区| 久久久99久久| 日韩一区二区在线免费观看| 国产拍欧美日韩视频二区| 日韩激情视频在线观看| 国产婷婷色一区二区三区四区| 美女网站一区二区| 亚洲午夜一区二区三区| 国产午夜精品理论片a级大结局| 黄网站免费久久| 日韩免费观看2025年上映的电影 | 欧美日韩免费电影| 成人免费毛片app| 亚洲老司机在线| 国产精品国产三级国产三级人妇| 91小宝寻花一区二区三区| 国产精品亚洲成人| 国产一区二区三区美女| 国产一区亚洲一区| 国产激情一区二区三区桃花岛亚洲| 久久综合九色综合久久久精品综合 | 一区二区理论电影在线观看| 国产日本亚洲高清| 国产精品污污网站在线观看| 亚洲国产电影在线观看| 91美女福利视频| 99久久伊人精品| 图片区小说区国产精品视频| 亚洲一区二区不卡免费| 日韩天堂在线观看| 精品欧美一区二区在线观看| 精品成人一区二区三区| 国产人伦精品一区二区| 一区精品在线播放| 亚洲午夜影视影院在线观看| 免费的成人av| 成人一二三区视频| 欧美在线观看一区二区| 久久蜜桃av一区二区天堂 | 日av在线不卡| 免费精品视频最新在线| 中文字幕在线一区二区三区| 99久久er热在这里只有精品66| 亚洲美女免费在线| 亚洲电影第三页| 欧美国产乱子伦| 亚洲精品成人少妇| 日韩**一区毛片| 成人午夜视频在线观看| 欧洲av一区二区嗯嗯嗯啊| 欧美mv日韩mv国产网站app| 国产精品免费人成网站| 日韩综合在线视频| 国产精品一区二区黑丝| 日韩黄色小视频| 中文字幕一区二区三区在线观看 | 色婷婷久久综合| 欧美日韩精品是欧美日韩精品| 99精品视频一区二区三区| 欧美视频中文字幕| 中文字幕的久久| 欧美成人aa大片| 亚洲欧洲美洲综合色网| 亚洲超碰精品一区二区| 亚洲色大成网站www久久九九| 26uuu国产电影一区二区| 一区免费观看视频| 国产资源精品在线观看| 欧美日韩一区二区欧美激情| 国产精品久久久久久久久免费相片 | 欧美人妇做爰xxxⅹ性高电影| 成人av影院在线| 欧美酷刑日本凌虐凌虐| 国产精品久久久久aaaa| 日韩精品成人一区二区在线| 91在线观看下载| 欧美不卡激情三级在线观看| 亚洲高清免费视频| 色一情一乱一乱一91av| 国产精品乱人伦| 久久99精品久久久久久| 欧美一区二区三区啪啪| 亚洲动漫第一页| 亚洲国产精品麻豆| 91在线视频观看| 欧日韩精品视频| 91在线国产福利| 精品国产一区二区三区四区四| 欧美猛男男办公室激情| 91福利视频网站| 久久色.com| 欧美精品一区二区久久婷婷| 欧美极品另类videosde| 毛片av中文字幕一区二区| av在线不卡观看免费观看| 久久综合久久99| 国产欧美日韩视频在线观看| 青青草91视频| 欧美综合色免费| 91精品国产综合久久久久久久久久| 在线一区二区三区四区五区| 中文字幕巨乱亚洲| 春色校园综合激情亚洲| 国产精品免费丝袜| 91免费看视频| 一级特黄大欧美久久久| 欧美三级日韩三级国产三级| 亚洲成年人网站在线观看| 欧美性一级生活| 亚洲一区二区av电影| 免费xxxx性欧美18vr| 欧美精品视频www在线观看| 欧美在线短视频| 午夜精品久久一牛影视| 麻豆久久久久久久| 日韩一区二区三区精品视频| 久久机这里只有精品| 欧美r级电影在线观看| 国产成人啪免费观看软件| 1000精品久久久久久久久| 日本欧美久久久久免费播放网| 国内精品伊人久久久久av一坑 | 欧美一区在线视频| 天堂久久久久va久久久久| 日韩欧美一区在线观看| 成人精品一区二区三区中文字幕| 884aa四虎影成人精品一区| 日本大胆欧美人术艺术动态| 久久综合av免费| 亚洲成人综合网站| 制服丝袜一区二区三区| 国产麻豆精品久久一二三| 精品视频123区在线观看| 亚洲成人av一区二区三区| 正在播放亚洲一区| 国产欧美视频一区二区| eeuss鲁片一区二区三区| 日韩欧美国产小视频| 国产成人午夜视频| 欧美调教femdomvk| 精品1区2区3区| 国产主播一区二区三区| 亚洲女厕所小便bbb| 亚洲午夜国产一区99re久久| 久久众筹精品私拍模特| 在线观看欧美精品| 国产一区不卡精品| 午夜一区二区三区视频| 国产欧美精品区一区二区三区 | 色综合视频在线观看| 中文字幕不卡的av| 国产精品一二三四五| 欧美精品乱人伦久久久久久| 欧美大黄免费观看| 欧美性猛片xxxx免费看久爱| 一区二区三区不卡视频| 欧美不卡在线视频| 欧美一区二区三区白人| 一本色道久久综合亚洲精品按摩 | av一区二区三区四区| 久久66热偷产精品| 一区二区欧美在线观看| 国产精品久久久久天堂| 日本一区二区三区四区在线视频| 国产成人啪午夜精品网站男同| 国产亚洲午夜高清国产拍精品| 国产传媒欧美日韩成人| 老司机午夜精品| 亚洲国产精品成人久久综合一区| 粉嫩av一区二区三区在线播放 | 欧美变态凌虐bdsm| 国产成人av一区二区三区在线 | 久久国产精品免费| 亚洲一区免费视频| 亚洲欧美另类小说视频| 国产精品视频看| 中文欧美字幕免费| 国产精品毛片大码女人| 亚洲国产高清在线| 国产精品天天摸av网| 国产精品人成在线观看免费| 中文字幕久久午夜不卡| 中文字幕一区二区三区四区不卡| 欧美系列一区二区| 欧洲精品在线观看| 884aa四虎影成人精品一区| 成人永久免费视频| 亚洲一区二区三区自拍| 成人欧美一区二区三区黑人麻豆 | 7777精品伊人久久久大香线蕉完整版 | 在线看日韩精品电影| 粉嫩蜜臀av国产精品网站| 日韩精品免费视频人成| 日本视频在线一区| 国产一区二区三区日韩| 成人av免费网站| 欧美日韩综合不卡| 精品国产乱码久久|