phpyun人才系统作为一个专业的PHP招聘系统,可以很好地做本地化的线上招聘工作,天然适合配合社区系统做本地化社区,于是在搭建了phpyun人才系统之后,又搭建了discuz 社区。

我的个人博客:逐步前行STEP

问题来了,怎么让phpyun系统中的用户可以直接使用社区,或者说怎么让社区用户直接可以登录phpyun系统——Ucenter 就是为了解决这个问题而存在的。

Ucenter
(摘抄一段百度百科)

UCenter 的中文意思就是“用户中心”,其中的 U 代表 User 也代表 You ,取其中的含义就是“用户中心”,或者说“你(最终用户)的中心”。 UCenter 是 Comsenz 旗下各个产品之间信息直接传递的一个桥梁,通过 UCenter 站长可以无缝整合 Comsenz 系列产品,实现用户的一站式注册、登录、退出以及社区其他数据的交互的php框架。

(本文只讲解决问题的干货,不教怎么安装)

在开始下文之前,先说明一下,需要现在phpyun后台开启论坛整合:
路径:工具 -》登录-》整合论坛-》整合Ucenter
在这里插入图片描述
我的个人博客:逐步前行STEP

还要填写基本的Ucenter的连接信息:
在这里插入图片描述
暂停,先配置Ucenter吧。

如果是在安装discuz时选择了同时安装ucenter,那么在discuz后台有个导航可以直接进入ucenter后台:
在这里插入图片描述
否则的话需要另外下载源码,仅仅是安装的话很简单,这里就不赘述了。

我的个人博客:逐步前行STEP

殊途同归,进入ucenter后台:
在这里插入图片描述
左边菜单中,有一项:应用设置,点击进入,开始添加应用:
在这里插入图片描述
这里填写的时候,有坑,先填写这几项:

我的个人博客:逐步前行STEP

  • 应用类型 如:其它
  • 应用名称 如:STEP
  • 应用的主 URL 如:hezehua.net
  • 通信密钥 如:xxxxxx

注意,通信密钥,在phpyun或者discuz 和 Ucenter中都要填写一致,这些都是好理解的,关键在于这个配置项应用接口文件名称,这个简直绝了,看备注:
在这里插入图片描述
说好了 不含路径,而且有一项:
在这里插入图片描述
表明填写应用的物理路径,按理来说路径应该是:/www/data/phpyun 这样的,文件名称应该是:uc.php这样的

可惜,我用惨痛的教训证明了这个提示和备注简直是坑爹,根本就不是这样的,应用的物理路径还是在Ucenter的根路径下查找的,而接口文件名称最后还是可以包含路径的,下面细说。

如果ucenter是和Diszuc一起安装的,discuz 与 Ucenter的连接是很好解决的,只要填好应用名称、应用的主 URL、秘钥就通了。如果不是的话,类似的就按下面的教程来填写。

我的个人博客:逐步前行STEP

首先,正确解释下应用接口文件名称的开启方式,填写了这个文件名称后,比如按默认是uc.php,ucenter会去访问: 主URL/api/uc.php
而phpyun(我的是v5.0.0)中api目录下没有uc.php文件,所以是会连接失败的。
源码中,/api目录下带了俩目录:uc、uc_php7,一开始我使用了uc/uc.php,结果报错mysql_connection undifined,这个是因为我的是php7,已经不支持mysql_connection 了,换成uc_php7/uc.php,(看吧,输入框旁边的提示是不是错误的????)然后可以顺利连接了。
在这里插入图片描述
回到phpyun,完成剩下的配置:
在这里插入图片描述
这俩配置项,可以直接在Ucenter中找到,应用配置底下有这些信息:

在这里插入图片描述
按这个填上即可, 再次暂停。

我的个人博客:逐步前行STEP

这只是完成了第一步,下面要展示真正的坑了。

接下来还要打通登录、注册的用户信息,可以发现,无论是登录还是注册,只要开启了整合ucenter之后,登录注册都报错了,我经过调试已经解决了问题,我就直接把原因告诉大家:
1、登录、注册中使用的都是api/uc下的ucenter客户端
2、提示邮箱不合法

问题一好解决:找源码中所有引用api/uc/的代码都替换成api/uc_php7/

问题二就纳闷了,注册的时候也没有要填邮箱啊,为啥体式邮箱不合法????因为啊,在phpyun注册的时候也会去ucenter注册,而ucenter注册必须使用邮箱,所以报错了,回到整合ucenter的配置,下面这个配置项的用途出来了:
在这里插入图片描述
就是用来给phpyun用户注册ucenter的,填写好这个,再去注册,成功。
在Ucenter‘中查看用户也没有问题。

但是,再注册的时候,问题又来了:
在这里插入图片描述

我的个人博客:逐步前行STEP

因为使用统一的EMAIL注册,所以报错了,回到ucenter允许一个email注册多个用户就行:
在这里插入图片描述

至此,phpyun人才系统 与 discuz 社区 会员 整合圆满完成。

为了解决上面的问题,其实花了三四个小时,都是小问题,花这些时间觉得很没有效率,而主要的原因就是:
1、没有文档没有教程 ,只言片语,一脸懵逼
2、phpyun的坑,框架丑的一批,难用死了,非常原始
3、备注和提示误导

我的个人博客:逐步前行STEP

我的个人博客:逐步前行STEP

在mysql应用中常常需要通过别的表的查询结果来更新本表,但很少会本表的查询结果再来更新本表的,下面就看看从本表查询结果更新本表应该怎么做吧。

表classify:

字段属性
idint(11)
namestring(255)

表production:

字段属性
idint(11)
classify_idint(11)

现在表classify中新增pro_cnt字段来统计一个分类下的商品数量:

字段属性
idint(11)
namestring(255)
pro_cntint(11)

新增字段之后,需要初始更新一下pro_cnt的值为当前分类下的商品数量,首先,通过leftjoin查询出每个分类下的商品数

    select count(production.id) pro_cnt from classify 
    left join production on classify.id = production.classify_id

然后将查询结果与classify再做一个连接,并且使用update...set语法做字段你更新:

    update classify inner join 
    (    
        select classify.id, count(production.id) pro_cnt from classify 
        left join production on classify.id = production.classify_id
    ) as tmp
    on using(id)
    set classify.pro_cnt = tmp.pro_cnt

1、如果是使用Eloquent ORM操作数据库的话,在sql查询时可以调用toSql()方法来获取sql:

    App\User::where('name','like','%hezehua%')->toSql();

2、如果是执行原生Sql查询,则不能使用toSql()方法了,而是开启查询日志:

    DB::enableQueryLog();
    DB::sselect("select * from users limit 10");
    $log = DB::getQueryLog();

如果当前请求可能会建立多个连接,可以指定在某一个连接中获取查询日志,未指定的话将从最近一个连接中获取查询日志:

    DB::connection($con)->enableQueryLog();

我的个人博客:逐步前行STEP

服务提供者是一个有效的将工具与业务解耦的方案,下面结合一个实用案例来解释服务提供者在实现提供基础服务的工具中的应用。

服务提供者

服务提供者是 Laravel 应用启动的中心,所有 Laravel 的核心服务都是通过服务提供者启动,通过使用服务提供者来管理类的依赖和执行依赖注入,可以很好地将一些底层服务与应用层代码解耦。

短信服务

短信服务对于一个系统来说,是基础的、通用的服务,不依赖于业务细节,所以应该将其与业务解耦。

系统设计

将不同服务商的sdk称为驱动,短信服务应该满足以下需求:

  1. 可替换驱动
  2. 驱动实例化对使用者透明

1、可替换驱动

要满足第1点,首先应该使用接口约束对外暴露的方法,只有驱动满足接口约束,才能不影响业务直接替换驱动,于是设计接口:

<?php

namespace App\Interfaces;

Interface SmsDriverInterface
{

    public function sendMessage($phones, $template_id, $parameters);
}

如果接入的厂商是七牛云,则创建七牛云短信驱动,并在驱动中调用SDK实现功能:

<?php
namespace App;

use Qiniu\Auth;
use Qiniu\Http\Client;
use Qiniu\Http\Error;
use Qiniu\Sms\Sms;
class QiniuDriver implements SmsDriverInterface
{
    const HOST = 'https://sms.qiniuapi.com';

    const VERSION = 'v1';
    public function __construct()
    {
        $this->auth  = new Auth(config('access_key'), config('secret_key'));

        $this->sms = new Sms($this->auth);
    }
 
    public function sendMessage($phones, $template_id, $parameters = [])
    {
        $phones = array_map(function($item)
        {
            if(!is_string($item))
            {
                $item = strval($item);
            }

            return $item;
        }, $phones);

        $ret = $this->sms->sendMessage($template_id, $phones, $parameters);

        $result = $this->getResultApiRet($ret);

        return $result;
    }
}

别的厂商的驱动也这样实现接口SmsDriverInterface,在更换厂商的时候,换一个驱动实例化就可以了。

2、驱动实例化对使用者透明

此处的使用者就是使用短信服务实现业务需求的工程师啦,因为短信服务的基础、通用的特性,会被在业务中很多地方使用,如果更换驱动的话,会涉及很多具体业务代码的修改,所以需要创建一个服务类,用来统筹驱动的使用,具体业务中再调用这个服务类,驱动的实例化就对业务透明了:

<?php
namespace App;
class SmsService
{
    public $driver;
    
    public function driver($driver_name)
    {
        switch($driver_name)
        {
            case 'qiniu':
                $this->driver = new QiniuDriver();
            break;
            case 'aliyun':
                $this->driver = new AliyunDriver();
            break;
        }
    }
    
    public function senndMessage($phone, $template_id)
    {
        return $this->driver->sendMessage($phone, $template_id);
    }
} 

再做改进,将传参选择驱动类型改为从配置中获取驱动类型:

<?php
namespace App;
class SmsService
{
    public $driver;
    
    public function driver()
    {
        switch(config('driver'))
        {
            case 'qiniu':
                $this->driver = new QiniuDriver();
            break;
            case 'aliyun':
                $this->driver = new AliyunDriver();
            break;
        }
    }
    ......
}

至此,基本满足了刚才所说的2点基本需求,但是,在增加驱动的时候要去改动driver()中的代码,增加其它驱动项,在服务类中枚举出所有的驱动似乎不够简洁,这里可以使用服务提供者再做优化:

<?php

namespace App\Providers;

use App\Interfaces\SmsDriverInterface;
use App\NullSms;
use App\QiniuSms;
use App\SmsService;
use Illuminate\Support\ServiceProvider;

class SmsServiceProvider extends ServiceProvider
{

    protected $defer = true;

    /**
     * Bootstrap the application services.
     *
     * @return void
     */
    public function boot()
    {

    }

    /**
     * Register the application services.
     *
     * @return void
     */
    public function register()
    {
        $this->app->singleton(SmsDriverInterface::class, function ($app) {

            switch(config('driver'))
            {
                case 'qiniu':
                    return new QiniuDriver();
            }

            return new NullSms();
        });
    }

    public function provides()
    {
        return [SmsDriverInterface::class];
    }
}

这里将驱动接口与配置中指定的具体驱动绑定,在服务类中的实例化方式可以改成依赖注入,因为短信驱动应该使用单例并且为了调用短信服务方便,将服务类中方法改为静态方法:

namespace App;
class SmsService
{
    public $driver;
    
    public function __construct(SmsDriverInterface $driver)
    {
        $this->driver = $driver;
    }
    
    public static function driver()
    {
        return app(self::class)->driver;
    }
    public static function sendMessage($phone, $template_id)
    {
        return SmsSend::driver()->sendMessage($phone, $template_id);
    }
}

最后将短信服务提供者添加到config/app.php文件的providers列表中,短信服务者的开发就完成了。

在这个案例中,短信服务提供者通过服务容器的依赖注入,实现了驱动在服务类中的自动实例化,在逻辑上将底层驱动与上层服务解耦。

这个案例比较简单,还有很多可以完善的地方,比如,不在服务提供者中能够使用switch去匹配驱动类型,而是增加一个驱动管理器,根据命名规则去实例化对应的驱动,这样的话就达到了增加并更换驱动的时候,只增加驱动类,以及更换config配置,就能“平滑”替换驱动的目的。

我的个人博客:逐步前行STEP

搜索引擎结构

文本预处理

分词

    文本分词一般是先将文本分成独立的单词,然后取出标点符号,去除停止词,停止词就是一些使用频率高但是没有实际意义的词,比如:a、an、the、are等等

语言处理

对于英文来说,先是统一大小写,再对单词进行缩减,缩减过程主要有两种,一种是词干提取,即抽取词的词干或者词根,如:tional 缩减为 tion (去除形容词后缀)二是词形还原,即把词汇还原成一般形式,如:books 缩减为 book。

建立索引

经过文本预处理之后,得到的结果为词(Term),将得到的词创建成字典,然后对字典按字母顺序排序,最后合并相同的词,形成文档倒排表(Posting List),因为这个表中,是根据词来查找文档的,所以称为文档倒排表或者倒排索引。

搜索

根据用户的输入,进行词法分析、语法分析、语言处理然后从文档倒排表中获取结果,为了将最相关的结果显示在最前面,需要对搜索结果做相关性排序。

相关性排序

把查询语句也视为一个文档,查询语句与文档的相关性就转变成了文档之间的相关性问你题,而一个文档中的每个词对于表达一个主题的重要性不一样,不同权重的词共同表达了这个主题。
有两个元素会影响一个词在文档中的重要性:一是词频率(Term Frequency,tf),表示一个词在此文档中出现的次数,它的值越大说明这个词越重要;二是文档频率(Document Frequency,df),表示多少文档中包含这个词,它的值越大,说明这个词越不重要。

以下公式用于计算一个词的权重:
在这里插入图片描述

而衡量两个文档相似度可以将两个文档当成向量,向量夹角越小越靠近,也就是说文档向量夹角越小,相似度越高,即相关性可以用以下公式来描述:
![在这里插入图片描述](https://img-blog.csdnimg.cn/20191217004018414.jpg?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2hoaHp1YQ==,size_16,color_FFFFFF,t_70)