将一个类的接口转换成客户希望的另外一个接口。Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作
主要角色
目标(Target)角色:定义客户端使用的与特定领域相关的接口,这也就是我们所期待得到的
源(Adaptee)角色:需要进行适配的接口
适配器(Adapter)角色:对Adaptee的接口与Target接口进行适配;适配器是本模式的核心,适配器把源接口转换成目标接口,此角色为具体类
适用性
1、你想使用一个已经存在的类,而它的接口不符合你的需求
2、你想创建一个可以复用的类,该类可以与其他不相关的类或不可预见的类协同工作
3、你想使用一个已经存在的子类,但是不可能对每一个都进行子类化以匹配它们的接口。对象适配器可以适配它的父类接口(仅限于对象适配器)
在实际应用中,适配器模式分为类适配器和对象适配器
1 类适配器
//目标角色
interface ITarget
{
function operation1();
function operation2();
}
//源角色
interface IAdaptee
{
function operation1();
}
class Adaptee implements IAdaptee
{
public function operation1()
{
echo "原方法";
}
}
//适配器角色
class Adapter extends Adaptee implements IAdaptee, ITarget
{
public function operation2()
{
echo "适配方法";
}
}
class Client
{
public function test()
{
$adapter = new Adapter();
$adapter->operation1();//原方法
$adapter->operation2();//适配方法
}
}
1 对象适配器
//目标角色
interface ITarget
{
function operation1();
function operation2();
}
//源角色
interface IAdaptee
{
function operation1();
}
class Adaptee implements IAdaptee
{
public function operation1()
{
echo "原方法";
}
}
//适配器角色
class Adapter implements ITarget
{
private $adaptee;
public function __construct($adaptee)
{
$this->adaptee = $adaptee;
}
public function operation1()
{
return $this->adaptee->operation1();
}
public function operation2()
{
echo "适配方法";
}
}
class Client
{
public function test()
{
$adapter = new Adapter(new Adaptee(null));
$adapter->operation1();//原方法
$adapter->operation2();//适配方法
}
}
类适配器中适配器继承原有的Adaptee类,自己实现原类没有的操作,使用的是继承模式,而对象适配器使用的是组合模式,将adaptee作为adapter的一个引用。由于组合在耦合性上小于继承,对象适配器显得更加灵活但缺点是增加代码量。 需要重写adapee中的方法的数量太大的话,可以考虑在adapter类中添加__call方法委托adapee取得客户端调用的方法
public function __call($func, $args)
{
if (is_callable(array($this->adaptee, $func))) {
return $this->adaptee->$func($args);
}
trigger_error('*********', E_USER_ERROR);
}