Laravel 創(chuàng)建自己的 Facade

博客原文:Laravel 創(chuàng)建自己的 Facade

前言

laravel 提供了一個(gè)靈活的模式屁倔,那就是 facade 〔嫒ぃ框架內(nèi)部的 DB泞边、Auth、File 等功能也有相關(guān)的 facade 實(shí)現(xiàn)疗杉。那么繁堡,該如何寫自己的 facade 呢?

Facade 是什么乡数?

首先椭蹄,facade 并不是 laravel 獨(dú)有的東西,它就是設(shè)計(jì)模式中的外觀模式(Facade)净赴。
當(dāng)然绳矩,這里就不長(zhǎng)篇大論去討論外觀模式的定義了。這篇文章寫的很不錯(cuò) : 設(shè)計(jì)模式(九)外觀模式Facade(結(jié)構(gòu)型)玖翅。
那么翼馆,laravel 的 facade 做了什么?
同樣的金度, laravel 實(shí)現(xiàn)了外觀模式的開關(guān)功能应媚,并且使用魔術(shù)方法 __callstatic 實(shí)現(xiàn)了靜態(tài)方式調(diào)用、動(dòng)態(tài)創(chuàng)建對(duì)象的功能猜极。參考 (官方文檔)

當(dāng)然你可能覺得這些概念很抽象中姜,都什么玩意。那么其實(shí)簡(jiǎn)單的講跟伏,laravel 的 facade 就是將某些功能封裝成工具類丢胚,而且能以靜態(tài)方式調(diào)用工具類的方法。

建立自己的 facade

首先受扳、以 laravel 5.1 框架携龟,我之前寫過的 Geoip facade 為例,說一下怎么去建立自己的 facade勘高。

下載 geoip 擴(kuò)展

geoip 是一個(gè)可以更具 IP 獲取國(guó)家峡蟋、地域坟桅、城市信息的 PHP 擴(kuò)展,基于 maxmind 數(shù)據(jù)庫(kù)蕊蝗。 github 在此仅乓。

首先,為 laravel 添加 geoip 擴(kuò)展匿又。
打開 composer.json,添加 "geoip2/geoip2": "~2.0" 到 require建蹄。
項(xiàng)目根目錄運(yùn)行 composer update ( 需要安裝 composer )更新一下碌更,geoip 的依賴和軟件包就被下載到 vendor 文件夾中了。

然后下載 geoip 依賴的數(shù)據(jù)庫(kù)洞慎,免費(fèi)庫(kù)的地址 : GeoLite2

我下載了 GeoLite2 Country 和 GeoLite2 City 庫(kù)痛单,放到了 storage/geoipdb 中。

建立 facade

在 app 目錄下新建 Facades 文件夾劲腿,里面新建 Facades/GeoIP/GeoIP.php 和 Facades/GeoIP/Facade/GeoIP.php (建議每個(gè)功能新建一個(gè)文件夾區(qū)分旭绒,比如我這里給 GeoIP 新建一個(gè)文件夾,關(guān)于GeoIP 的東西全放到這里)
注意焦人,F(xiàn)acades/GeoIP 下的 GeoIP.php 是你要對(duì) geoip 擴(kuò)展進(jìn)行封裝的類挥吵, Facades/GeoIP/Facade 下的 GeoIP.php 是你的 facade,用來給 laravel 解析使用花椭,這兩個(gè)文件可以不同名忽匈。


image.png

Facades/GeoIP/Facade/GeoIP.php 如下

<?php

namespace App\Facades\GeoIP\Facade;

use Illuminate\Support\Facades\Facade;

class GeoIP extends Facade
{
    protected static function getFacadeAccessor()
    {
        return 'geoip';
    }
}

注意你的 facade 現(xiàn)在只有一個(gè)方法,返回了一個(gè)字符串 'geoip' , 這個(gè)字符串是一個(gè)標(biāo)號(hào)矿辽,用來給 laravel 的服務(wù)提供者解析使用的丹允。

Facades/GeoIP/GeoIP.php 如下(吐槽:寫的有點(diǎn)隨意)

<?php

namespace App\Facades\GeoIP;

use GeoIp2\Database\Reader;

class GeoIP
{
    /**
     * GeoIP country db path (base on storage_path).
     *
     * @var GeoIP
     */
    private $_country_db = 'geoipdb/GeoLite2-Country.mmdb';

    /**
     * GeoIP city db path (base on storage_path).
     *
     * @var GeoIP
     */
    private $_city_db = 'geoipdb/GeoLite2-City.mmdb';

    /**
     * Instance for GeoIP .
     *
     * @var GeoIP
     */
    private $_instance;

    /**
     * Init instance.
     *
     */
    public function init($mode)
    {
        switch ($mode) {
          case 'getCountry':
            $path = $this->_country_db;
            break;
          case 'getCity':
            $path = $this->_city_db;
            break;
          default:
            break;
        }

        $this->_instance = new Reader(storage_path($path));
    }

    /**
     * Get Country infomations.
     *
     * @param  String  $ip
     * @return Array
     */
    public function getCountry($ip)
    {
      $this->init(__FUNCTION__);

      $record = $this->_instance->country($ip);

      // 國(guó)家信息
      $data['iso_code'] = $record->country->isoCode;
      $data['country_name'] = $record->country->name;
      $data['country_name_zh_cn'] = $record->country->names['zh-CN'];

      return $data;
    }
 
 /**
     * Get City infomations.
     *
     * @param  String  $ip
     * @return Array
     */
    public function getCity($ip)
    {
      $this->init(__FUNCTION__);

      $record = $this->_instance->city($ip);

      $data['iso_code'] = $record->country->isoCode;
      $data['country_name'] = $record->country->name;
      $data['country_name_zh_cn'] = $record->country->names['zh-CN'];

      // 省、州信息
      $data['sub_division_name'] = $record->mostSpecificSubdivision->name;
      $data['sub_division_name_zh_cn'] = $record->mostSpecificSubdivision->names['zh-CN'];
      $data['sub_division_code'] = $record->mostSpecificSubdivision->isoCode;

      // 城市信息
      $data['city_name'] = $record->city->name;
      $data['postal_code'] = $record->postal->code;

      // 經(jīng)緯度
      $data['latitude'] = $record->location->latitude;
      $data['longitude'] = $record->location->longitude;

      return $data;
    }
}

OK袋倔,現(xiàn)在 geoip 的常用功能已經(jīng)封裝到方法中了雕蔽。

注冊(cè)服務(wù)

完成了 facade 的創(chuàng)建和功能封裝,下面就要使用它了宾娜。自己創(chuàng)建的 facade 要在 laravel 使用是要進(jìn)行注冊(cè)的批狐,以便 laraval 在啟動(dòng)時(shí)能自動(dòng)注入依賴(請(qǐng)看 laravel 的依賴注入簡(jiǎn)介 : laravel 依賴注入 學(xué)院君)

編寫服務(wù)提供者

在 app/Providers 下新建 FacadesServiceProvider.php
可以手動(dòng)建,也可以用 artisan 命令來生成前塔,隨你喜歡贾陷。
app/Providers/FacadesServiceProvider.php 代碼如下:

<?php

namespace App\Providers;

use App\Service\ApiService;
use Illuminate\Support\ServiceProvider;

// include the class facade binded
use App\Facades\GeoIP\GeoIP;

class FacadesServiceProvider extends ServiceProvider
{
    /**
     * 在容器中注冊(cè)綁定。
     *
     * @return void
     */
    public function register()
    {
        $this->app->singleton('geoip', function ($app) {
            return new GeoIP($app);
        });
    }
}

上面代碼可知嘱根,服務(wù)提供者注冊(cè)時(shí)會(huì)注冊(cè)一個(gè)單例髓废,標(biāo)號(hào)為 'geoip',也就是我們自己的 facade 返回的那個(gè)该抒,然后回調(diào)函數(shù)會(huì)返回一個(gè)對(duì)象慌洪,也就是我們封裝 geoip 功能的那個(gè)類的實(shí)例顶燕,不明白的同學(xué)可以看看 laravel 的服務(wù)提供者和服務(wù)容器相關(guān)知識(shí)哦。(注意要 use 將 facade 和封裝類的命名空間引用一下哦)

注冊(cè)服務(wù)提供者

laravel 5.1 以上版本的話冈爹, config/app.php 中找到 providers 和 aliases 涌攻,將你的服務(wù)提供者和 facade 別名配置一下 :

providers 加入 :

App\Providers\FacadeServiceProvider::class,
aliases 加入(不用每次都寫很長(zhǎng)的命名空間前綴) :

'GeoIP' => App\Facades\GeoIP\Facade\GeoIP::class,
對(duì)于 lumen 5.2 以上,需要在 bootstrap/app.php 中添加
$app->register(App\Providers\FacadesServiceProvider::class);
注:放到AppServiceProvider類的register方法里更合理一些
$this->app->register(App\Providers\FacadesServiceProvider::class);
注冊(cè)完畢后频伤,每次使用 facade::function 的時(shí)候恳谎,laravel 會(huì)自動(dòng)解析 facade, 然后創(chuàng)建一個(gè)對(duì)象給用戶使用憋肖,因痛,而無(wú)需用戶自己去 new 一個(gè)對(duì)象出來。

使用
現(xiàn)在岸更,在任何一個(gè)控制器鸵膏,或者路由的回調(diào)函數(shù)中,使用

$res = GeoIP::getCountry('75.101.195.215');
var_dump($res);

你會(huì)發(fā)現(xiàn)怎炊,facade 已經(jīng)可以好好工作了谭企,enjoy!

參考文章

【1】設(shè)計(jì)模式(九)外觀模式Facade(結(jié)構(gòu)型)
【2】Laravel 服務(wù)容器實(shí)例教程 —— 深入理解控制反轉(zhuǎn)(IoC)和依賴注入(DI)
【3】Laravel 服務(wù)提供者實(shí)例教程 —— 創(chuàng)建 Service Provider 測(cè)試實(shí)例

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末评肆,一起剝皮案震驚了整個(gè)濱河市债查,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌瓜挽,老刑警劉巖攀操,帶你破解...
    沈念sama閱讀 219,539評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異秸抚,居然都是意外死亡速和,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,594評(píng)論 3 396
  • 文/潘曉璐 我一進(jìn)店門剥汤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來颠放,“玉大人,你說我怎么就攤上這事吭敢∨鲂祝” “怎么了?”我有些...
    開封第一講書人閱讀 165,871評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵鹿驼,是天一觀的道長(zhǎng)欲低。 經(jīng)常有香客問我,道長(zhǎng)畜晰,這世上最難降的妖魔是什么砾莱? 我笑而不...
    開封第一講書人閱讀 58,963評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮凄鼻,結(jié)果婚禮上腊瑟,老公的妹妹穿的比我還像新娘聚假。我一直安慰自己,他們只是感情好闰非,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,984評(píng)論 6 393
  • 文/花漫 我一把揭開白布膘格。 她就那樣靜靜地躺著,像睡著了一般财松。 火紅的嫁衣襯著肌膚如雪瘪贱。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,763評(píng)論 1 307
  • 那天辆毡,我揣著相機(jī)與錄音菜秦,去河邊找鬼。 笑死胚迫,一個(gè)胖子當(dāng)著我的面吹牛喷户,可吹牛的內(nèi)容都是我干的唾那。 我是一名探鬼主播访锻,決...
    沈念sama閱讀 40,468評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼闹获!你這毒婦竟也來了期犬?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,357評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤避诽,失蹤者是張志新(化名)和其女友劉穎龟虎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體沙庐,經(jīng)...
    沈念sama閱讀 45,850評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡鲤妥,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,002評(píng)論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了拱雏。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片棉安。...
    茶點(diǎn)故事閱讀 40,144評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖铸抑,靈堂內(nèi)的尸體忽然破棺而出贡耽,到底是詐尸還是另有隱情,我是刑警寧澤鹊汛,帶...
    沈念sama閱讀 35,823評(píng)論 5 346
  • 正文 年R本政府宣布蒲赂,位于F島的核電站,受9級(jí)特大地震影響刁憋,放射性物質(zhì)發(fā)生泄漏滥嘴。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,483評(píng)論 3 331
  • 文/蒙蒙 一至耻、第九天 我趴在偏房一處隱蔽的房頂上張望氏涩。 院中可真熱鬧届囚,春花似錦、人聲如沸是尖。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,026評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)饺汹。三九已至蛔添,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間兜辞,已是汗流浹背迎瞧。 一陣腳步聲響...
    開封第一講書人閱讀 33,150評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留逸吵,地道東北人凶硅。 一個(gè)月前我還...
    沈念sama閱讀 48,415評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像扫皱,于是被迫代替她去往敵國(guó)和親足绅。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,092評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容