假设我有一个CrawlerInterface
实现PageCrawler
和FeedCrawler
的接口;如果我们碰巧在控制器中需要这两个类,如何通过构造函数注入来实现呢?
以前我们使用中央ServiceProvider
来注册(即 App::bind
)这样的类,但在大多数情况下,我们只有一个接口的1个实现,所以还没有想到上述问题。
PS:我也想知道这个问题是否表明我们应该拆分控制器。
更新:
感谢您的评论和回复,解释一下,所述接口只有一个公共方法:crawl($uri)
,并且页面/提要爬虫都将其实现为given a resource identifier, return resource.
我的后续问题:
假设我们处于一个calculator
场景中,加法、减法和乘法共享同一个接口Operation
,它只有 1 个公共方法run
,在某些时候我们仍然会遇到这个问题,对吗?我们如何处理这种情况,一般情况下ServiceProvider
?
如果每个爬网程序的存在原因不同,您可以为实例使用任意名称,例如:
App::bind('crawler.allArticles', 'PageCrawler');
App::bind('crawler.latestArticles', 'FeedCrawler');
对于控制器:
App::bind('CrawlerController', function($app) {
return new CrawlerController(
App::make('crawler.allArticles'),
App::make('crawler.latestArticles')
);
});
然后,您的控制器代码将以不同的方式使用每个爬网程序:
public function showLatestArticlesAction()
$latestArticles = $this->latestArticlesCrawler->crawl();
// ...
}
public function showAllArticlesAction()
$allArticles = $this->allArticlesCrawler->crawl();
// ...
}
如果您只有一个爬虫列表,其中每个爬虫都用于同一事物,您可能希望执行以下操作:
App::bind('crawlers', function($app) {
return [
App::make('PageCrawler'),
App::make('FeedCrawler'),
];
});
在您的控制器中,您将通过如下所示进行配置来获得"爬虫"列表:
App::bind('CrawlerController', function($app) {
return new CrawlerController(App::make('crawlers'));
});
您的控制器代码可能如下所示:
public function showArticlesAction()
$allArticles = array();
foreach ($this->crawlers as $crawler) {
$allArticles = array_merge($allArticles, $this->crawler->crawl());
}
// ...
}
让我们假设你有一个CrawlerController
class CrawlerController extends BaseController
{
protected $crawler1;
protected $crawler2;
public function __construct(CrawlerInterface $c1, CrawlerInterface $c2)
{
$this->crawler1 = $c1;
$this->crawler2 = $c2;
}
}
一个接口
interface CrawlerInterface{}
以及称为PageCrawler
和FeedCrawler
的智能的具体实施
class PageCrawler implements CrawlerInterface{}
class FeedCrawler implements CrawlerInterface{}
您可以通过编写服务定位器来注入依赖项,例如
App::bind('CrawlerController', function($app) {
$controller = new CrawlerController(
new PageCrawler,
new FeedCrawler
);
return $controller;
});
但正如其他人所建议的那样,你应该重新考虑你的逻辑,只有在这种逻辑时才使用它的建筑是不可避免的
我认为在这种情况下界面对您没有帮助。
通过做:
App::bind('CrawlerInterface', '<implementation>');
您需要选择一个:
App::bind('CrawlerInterface', 'PageCrawler');
或
App::bind('CrawlerInterface', 'FeedCrawler');
然后拉拉维尔会注入它:
class CrawlerController {
public function __construct(CrawlerInterface $crawler)
{
}
}
要同时拥有两者,您有 2 个选择
- 有2个不同的接口
-直接注入实现:
class CrawlerController {
public function __construct(PageCrawler $pageCrawler, FeedCrawler $feedCrawler)
{
}
}
但我也认为,如果你需要这样的东西,你最好重新考虑你的逻辑。