是否可以在框架中包含不是专门为 L4 设计的包?如果是这样,它是如何完成的?我知道我需要将包添加到我composer.json
的文件夹中,但是我可以在数组vendor
中以某种方式注册它吗?providers
还有其他必要的步骤吗?
我想使用最初为 Yii 设计的Google checkout 包
是否可以在框架中包含不是专门为 L4 设计的包?如果是这样,它是如何完成的?我知道我需要将包添加到我composer.json
的文件夹中,但是我可以在数组vendor
中以某种方式注册它吗?providers
还有其他必要的步骤吗?
我想使用最初为 Yii 设计的Google checkout 包
在 Laravel 4 中使用第三方作曲家包
当开发人员创建 composer 包时,他们应该使用 PSR-0 或 PSR-4 标准映射自动加载。如果不是这种情况,在你的 Laravel 应用程序中加载包可能会出现问题。PSR-0 标准是:
{
"autoload": {
"psr-0": { "Acme": "src/" }
}
}
以及 PSR-4 标准:
{
"autoload": {
"psr-4": { "Acme\\": "src/" }
}
}
基本上,以上是告诉作曲家在哪里寻找名称空间文件的标准。如果您不使用自己的命名空间,则无需配置任何其他内容。
场景 1
Laravel 中的 PSR-0 标准以下包(带有自动加载类映射)
这是一个简单的,例如我将使用 facebook php sdk,可以找到:
https://packagist.org/packages/facebook/php-sdk
第1步:
将包包含在您的 composer.json 文件中。
"require": {
"laravel/framework": "4.0.*",
"facebook/php-sdk": "dev-master"
}
第2步:
run: composer update
第 3 步:
因为 facebook 包使用了一个开箱即用的类映射,所以您可以立即开始使用该包。(下面的代码示例直接来自普通视图。请在生产应用程序的视图中保留您的逻辑。)
$facebook = new Facebook(array(
'appId' => 'secret',
'secret' => 'secret'
));
var_dump($facebook); // It works!
情景 2
对于这个例子,我将使用来自 instagram php api 的包装器。这里需要进行一些调整才能加载包。试一试吧!包可以在这里找到:
https://packagist.org/packages/fishmarket/instaphp
第1步:
添加到作曲家 .json
"require": {
"laravel/framework": "4.0.*",
"fishmarket/instaphp": "dev-master"
}
然后就可以正常更新了(composer update)
接下来尝试像使用 facebook 包一样使用该包。同样,这只是视图中的代码。
$instagramconfig = array(
'client_id' => 'secret',
'client_secret'=> 'secret',
'access_token' => 'secret'
);
$api = Instaphp::Instance(null, $instagramconfig);
var_dump($api); // Epic fail!
如果您尝试上面的示例,您将收到此错误:
FatalErrorException: Error: Class 'Instaphp' not found in ...
所以我们需要解决这个问题。为此,我们可以检查 instagram composer.json,它的自动加载与 facebook php sdk 不同。
"autoload": {
"psr-0": { "Instaphp": "." }
}
与 facebook composer.json 相比:
"autoload": {
"classmap": ["src"]
}
(Composer 处理不同类型的自动加载,从文件和类映射到 PSR。看看你的vendor/composer/
文件夹,看看它是如何完成的。)
现在我们必须手动加载类。很简单,只需添加这个(控制器、模型或视图的顶部):
use Instaphp\Instaphp;
作曲家转储自动加载,它的工作原理!
第二步(可选)
另一种方法是(如果您不想使用“使用”语句,您可以简单地告诉作曲家直接从您的代码中查找文件。只需像这样更改实例:
// reference the name-spaced class straight in the code
$api = Instaphp\Instaphp::Instance(null, $instagramconfig);
var_dump($api); // It works
但是,我建议使用该use
语句让其他开发人员(以及您未来的自己)清楚地了解程序中使用了哪些(外部)类/包。
场景 3
这里我们使用 Laravel 内置的 IOC 容器来注册服务提供者。请注意,某些软件包可能不适合此方法。我将使用与场景 2 中相同的 Instagram 包。
又快又脏
如果您不关心设计模式和服务提供者,您可以像这样绑定一个类:
App::bind('Instaphp', function($app)
{
return new Instaphp\Instaphp;
});
你像这样解决它。
App::make('Instaphp');
快速而肮脏的结局
如果您正在处理一个更大的项目,并且您使用接口,您可能应该进一步抽象绑定。
第1步:
在您的应用程序文件夹中创建一个文件夹,例如“提供者”文件夹。
app/providers
确保 Laravel 自动加载该文件夹,您可以将一些附加信息传递给 composer.json,如下所示:
"autoload": {
"classmap": [
"app/commands",
"app/controllers",
"app/models",
"app/database/migrations",
"app/database/seeds",
"app/tests/TestCase.php",
"app/providers" // this was added
]
},
现在在名为 Instagram.php 的新文件夹中创建一个文件并将其放入:
<?php
use Illuminate\Support\ServiceProvider;
class InstagramServiceProvider extends ServiceProvider {
public function register()
{
$this->app->bind('Instaphp', function()
{
return new Instaphp\Instaphp;
});
}
}
现在再次运行 composer dump-autoload,就可以使用这个包了。请注意,instagram 包有一个final private function __construct()
,这意味着您不能在原始类之外使用该包而不将构造方法更改为公共。我并不是说这是一个好习惯,我建议使用场景 2,就 instagram 包而言。
无论如何,在此之后你可以使用这样的包:
$instagramInstance = App::make('Instaphp');
$instagramconfig = array(
'client_id' => 'secret',
'client_secret'=> 'secret',
'access_token' => 'secret'
);
$instagram = new $instagramInstance();
$userfeed = $instagram->Users->feed($instagramconfig);
var_dump($userfeed); // It works!
将此添加"tvr/googlecheckout": "dev-master"
到您的composer.json
.
运行composer install
,然后就可以使用 IoC 容器了。一些代码示例可以在 Laravel 4 的官方文档中找到:http: //four.laravel.com/docs/ioc#basic-usage