将开源PHP组件注册为服务提供者,以结巴分词为例

背景

packgist.orgcomposer组件库中有很多针对laravel框架的组件,它给我们提供了ServiceProvider文件,我们使用起来会非常方便。

但是组件库中大部分不是为Laravel开发的组件,这时使用起来其实也很方便。下载完毕后,直接利用命名空间使用即可。但为了符合Laravel的开发模式,我们可以将其注册为服务提供者,笔者经历一番测试之后,终于成功了,特此发一篇教程给大家。

举例

阿里大于针对Laravel的组件:alisms-for-laravel
网址:https://packagist.org/packages/iscms/alisms-for-laravel

Paste_Image.png

结巴分词组件:fukuball/jieba-php
网址:https://packagist.org/packages/fukuball/jieba-php

Paste_Image.png

那么就以结巴分词为例,开始注册服务提供者

打开组件网站,按提示使用composer安装,笔者不再赘述。安装完之后就是这样的效果,如下图:


Paste_Image.png

首先看一下功能和结构

功能分析:

  • 功能 1):分词
  • 功能 2):添加自定义词典
  • 功能 3):关键词提取
  • 功能 4):词性分词
  • 功能 5):切换繁体字典

具体演示代码,不做举例,网站上写的很清晰。

结构分析:
使用到的四个主要的类和两个辅助的工具类,其它文件为辅助分词的词典。

  • Fukuball\Jieba\Jieba 分词类
  • Fukuball\Jieba\Finalseg 必须依赖类
  • Fukuball\Jieba\JiebaAnalyse 可选词性类
  • Fukuball\Jieba\Posseg 可选词频类

分词的三种模式,默认精确模式、全模式、搜索引擎模式。

封装一个集合类

因为分词会用到多个类,如果不进行合并的话,这意味我们要注册多个类,这很麻烦。那我们可以做一个集成类,将所有需要的功能和依赖集合于其中。就像下面这样:

# Bootstrap.php
<?php
namespace Fukuball\Jieba;

ini_set('memory_limit', '1024M');

require_once __DIR__."/vendor/multi-array/MultiArray.php";
require_once __DIR__."/vendor/multi-array/Factory/MultiArrayFactory.php";
require_once __DIR__."/class/Jieba.php";
require_once __DIR__."/class/Finalseg.php";
require_once __DIR__."/class/JiebaAnalyse.php";
require_once __DIR__."/class/Posseg.php";
use Fukuball\Jieba\Jieba as Jieba;
use Fukuball\Jieba\Finalseg as Finalseg;
use Fukuball\Jieba\JiebaAnalyse as JiebaAnalyse;
use Fukuball\Jieba\Posseg as Posseg;

// 集成类
class Bootstrap
{
    public $jieba;
    public $finalseg;
    public $jiebaAnalyse;
    public $posseg;
    public $dic;            // Array 分词使用词典(大中小三种)
    public $user_dic;       // Path 用户自定义词典

    public function __construct($option){  // 可以加一些配置参数
         //实例化
        $this->jieba = new Jieba();
        $this->finalseg = new Finalseg();
        $this->jiebaAnalyse = new JiebaAnalyse();
        $this->posseg = new Posseg();

        // 初始化
        $this->jieba->init();
        $this->finalseg->init();
        // $this->jiebaAnalyse->init();     // 词性分析
        // $this->posseg->init();           // 词频分析

        $this->dictionary = ['dict' => 'small'];                    // 默认使用小词典
        $this->user_dic = dirname(__FILE__).'/dict/user_dict.txt';  // 用户自定义词典
        $this->jieba->loadUserDict($this->user_dic);                // 载入自定义词典
    }       

    public function jieba(){
        
        return $this->jieba;
    }

    public function finalseg(){
        
        return $this->finalseg;
    }

    public function jiebaAnalyse(){
        
        return $this->jiebaAnalyse;
    }

    public function posseg(){
        
        return $this->posseg;
    }
}

注意一点,php的内存限制要扩大到1G才可以。
上述代码基本上就是按照给定的例子改写的,只是简单的示例而已,至于细节之处,笔者是刚刚试验出来,还没完善。

创建一个服务提供者

<?php
namespace Fukuball\Jieba;

use Illuminate\Support\ServiceProvider;

class JiebaServiceProvider extends ServiceProvider
{
    /**
     * Bootstrap any application services.
     *
     * @return void
     */
    public function boot()
    {
        //
    }

    /**
     * Register any application services.
     *
     * @return void
     */
    public function register()
    {
        $this->app->singleton(\Fukuball\Jieba\Bootstrap::class, \Fukuball\Jieba\Bootstrap::class);
    }
}

复制一个,然后按照葫芦画瓢。
singleton方法是注册一个单例Bootstrap的类。

注册服务提供者

打开congif\app.php

// 注册服务提供者
'providers' => [
    Fukuball\Jieba\JiebaServiceProvider::class,
]
// 给使用类添加别名
'aliases' => [
    'JiebaTest' => Fukuball\Jieba\Bootstrap::class,
]

测试

添加路由,进行测试


Paste_Image.png

问题1 关于random的一些东西找不到

paragonie/random_compat/lib/random.php

结巴分词的依赖问题,composer install 即可。

问题2 服务提供者无法找到

这个类找不到。

Class "JiebaServiceProvider" is not Found.

最终测试结果,问题出子composer.json之中,composer.json之中autoload自动加载无法将命名空间解析到指定的文件路径之中。其默认autoload是这样的:


Paste_Image.png

发现这里面仅自动加载了默认的文件的路径,而新添加的文件未被解析。这时,我们要对其进行修改,让其解析命名空间是与路径对应起来。

你可能会修改成这样:


Paste_Image.png

但很遗憾,这样做没起作用(笔者也很纳闷,按理说这样是可以的,如果你测试成功了可以告诉我)。

但笔者退而求其次,又找了另一种解决方案。

在Laravel中的composer.jsonautoload中,有多种加载方式,其最终会是加载的是autoload_real.php文件,但很可惜,这是一个自动生成的文件,我们更改后会出现一些问题。

这时,笔者又找到了installed.json文件,该文件是composer安装组件的所有配置文件,经笔者测试,修改后是可行的。如下图所示:

Paste_Image.png

找到结巴分词的autoload位置,进行如下修改:

Paste_Image.png

注意:末尾的标点符号要干掉,否则会解析错误。
提醒:修改完毕后,要使用composer dump-autoload重新生成autoload文件

然后我们打开autoload_psr4.php看一下,发现Fukuball命名空间能够与文件路径对应了。


Paste_Image.png

最后我们在进行一下测试

Paste_Image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容