代码到接口,而不是实现;程序到接口,使用抽象,而不是具体。
这些都是同一件事,在开发中,我们的应用程序应该依赖抽象,而不是具体的类。
why?
刚开始听,感觉不是多次一举吗?为什么不直接写控制器,写方法,写逻辑就行了。
实际上,从架构师的角度看,没有什么不是变化的,就是随着时间的推移,业务的变化,需求都是变化的,为了适应变化,就不能依赖具体。
代码到接口,使我们代码松散耦合且灵活。
请举例:
class Logger {
public function log($content) {
//输出 Log 日志到文件。
echo "Log to file";
}
}
一个简单Logger类,我们在控制器控制他
class LogController extends Controller {
public function log() {
$logger = new Logger();
$logger->log('Log this');
}
}
(简书对代码排版,真是支持的很不友好啊!!!)
他要记录其他位置呢?如数据库,文件,云或其他?
需要再Log类中加几个方法
然后再控制器中写对应方法改进
好了,我们现在可以通过配置文件把日志输出到各种终端。
但我们如果还要再输出日志到 redis 呢?我们还需要再增加一个方法,并且在控制器中再加一次判断。
控制器代码很快就变得臃肿,如果还要输出日志到更多地方呢?Logger 类中每个方法如果还需要扩展呢?这对于后期维护来说并不好。
这样做同时也不符合 SOLID 原则,我们先来拆分一下 Logger 类,将职责拆分成不同的类。
再来修改LogController
看上去拆分了log,那继续添加日志到redis,那就继续加case吧。
但依然有一个问题就是我们的控制器「知道太多了」,它应该只去调用一个 log() 方法来记录,而不应该知道使用哪个 Logger 类,也不应该去实例化任何类,这样在将来有改动的时候,不论是要输出到哪里,我们都不需要再来修改 LogController 的代码,那应该怎么做呢?
这时候就是 interface出厂,听下回分解。
接口定义是可以执行操作的描述
interface LogInterface{
public function log($content);
}
我们一般把接口文件放在 项目目录 App\Contracts文件里。
接口只声明,不实现,这是他 抽象的原因。
我们实现接口时,必须提供实现接口的类的声明方法
class LogController extends controller{
public function Log(LogInterface $logger){
$logger->log('log to');
}
}
namespace App\Logs;
use App\Contracts\LogInterface;
class DBLogger implents LogInterface{
public function log($content){
//输出日志到DB
}
}
namespace App\Logs;
use App\Contracts\LogInterface;
class FileLogger implents LogInterface{
public function log($content){
//输出日志到file
}
}
然后是依赖注入
在使用Laravel框架时,我们可以利用它服务容器来实现自动注入接口实现
新建 config/log.php
<?php
return [
'default' => env('Log_TARGET','file');
'file' => [
'class' => App\Logs\FileLogger,
],
'db' => [
'class'=>App\Logs\DBLogger
]
];
并在 app\Providers\AppServiceProvider.php 添加下代码
public function register(){
$default = config('log.default');
$logger = config("log.{$default}.class");
$this->app->bind(
\App\Contracts\LogInterface::class,
$logger
);
}
我们从配置文件中选用日志类型,然后再config配置中绑定,注册服务,当我们请求接口时,容器就会解析返回实例
总结:
接口允许我们创建松散耦合的代码,同时提供一定程度的抽象,这样在快速切换时候,就不用现写代码。
对于大型应用会随着变化业务变化,建议用这种,一劳永逸,不要嫌弃当时写的代码多,至于小型应用,那看你心情。
真着急看,看 interface 在Laravel开发中应用