-
定义
将一个接口转换成客户想要的另一个接口,适配器模式使接口不兼容的那些类可以一起工作。
UML类图如下:
其中类的定义及关系如下:
1. Target(目标抽象类):定义客户要用的特定领域的接口。
2. Adapter(适配器-公接口):调用另一个接口,作为一个转换器。
3. Adaptee(适配器-母接口):定义一个接口,Adapter需要接入。
4. Clinet(客户调用类):协同对象符合Adapter适配器。
典型应用的顺序图如下,客户的提交消息被传到Adaptee中。
-
实例1:化学数据银行
化学药品对象通过适配器接口访问化学数据银行。类图如下:
//目标抽象类
class ChemicalCompound
{
protected string name;
protected float boilingPoint;
protected float meltingPoint;
protected double molecularWeight;
protected string molecularFormula;
//构造函数
public ChemicalCompound(string name)
{
this.name = name;
}
//属性
public float BoilingPoint
{
get { return boilingPoint; }
}
public float MeltingPoint
{
get { return meltingPoint; }
}
public double MolecularWeight
{
get { return molecularWeight; }
}
public string MolecularFormula
{
get { return molecularFormula; }
}
}
//公接口
class Compound : ChemicalCompound
{
private ChemicalDataBank bank;
public Compound(string name)
: base(name)
{
//适配器(母接口)-化学数据银行
bank = new ChemicalDataBank();
boilingPoint = bank.GetCriticalPoint(name, "B");
meltingPoint = bank.GetCriticalPoint(name, "M");
molecularFormula = bank.GetMolecularStructure(name);
molecularWeight = bank.GetMolecularWeight(name);
}
public void Display()
{
Console.WriteLine("\n化合物:{0}--------", name);
Console.WriteLine(" 分子式:{0}", MolecularFormula);
Console.WriteLine(" 分子量:{0}", MolecularWeight);
Console.WriteLine(" 熔点:{0}", MeltingPoint);
Console.WriteLine(" 沸点:{0}", BoilingPoint);
}
}
//适配器(母接口)-化学数据银行
class ChemicalDataBank
{
//取得临界点
public float GetCriticalPoint(string compound,string point)
{
float temperature=0.0F;
if(point=="M")//临界点
{
switch(compound.ToLower())
{
case "water":
temperature=0.0F;
break;
case "benzene":
temperature=5.5F;
break;
case "alcohol":
temperature=-114.1F;
break;
}
}
else//沸点
{
switch(compound.ToLower())
{
case "water":
temperature=100.0F;
break;
case "benzene":
temperature=80.1F;
break;
case "alcohol":
temperature=78.3F;
break;
}
}
return temperature;
}
//取得分子式
public string GetMolecularStructure(string compound)
{
string structure="";
switch(compound.ToLower())
{
case "water":
structure="H2O";
break;
case "benzene":
structure="C6H6";
break;
case "alcohol":
structure="C2H6O2";
break;
}
return structure;
}
//取得分子量
public double GetMolecularWeight(string compound)
{
double weight=0.0;
switch(compound.ToLower())
{
case "water":
weight=18.015;
break;
case "benzene":
weight=78.1134;
break;
case "alcohol":
weight=46.0688;
break;
}
return weight;
}
}
//客户应用测试
class Client
{
[STAThread]
static void Main(string[] args)
{
//显示取得水的特性
Compound water = new Compound("Water");
water.Display();
//显示取得苯的特性
Compound benzene = new Compound("Benzene");
benzene.Display();
//显示取得乙醇的特性
Compound alcohol = new Compound("Alcohol");
alcohol.Display();
Console.Read();
}
}
-
实例2:清洁系统
例如有个清洁系统已经设计完成,但你想增加更多的工作,IExtra接口就是采用Adapter模式插入系统中的,类图如下:
//假定系统原来已定义Clean接口
public interface IClean
{
void MakeClean();
}
//Office类,实现Clean接口
public class Office:IClean
{
public void MakeClean()
{
Console.WriteLine("清洁办公室");
}
}
//WorkShop类,实现Clean接口
public class WorkShop:IClean
{
public void MakeClean()
{
Console.WriteLine("清洁工作室");
}
}
//IExtra接口扩展了Clean接口
public interface IExtra : IClean
{
void TakeCare();
}
//Facility设备类实现了IExtra接口,除了清洁还要养护
public class Facility:IExtra
{
public void MakeClean()
{
Console.WriteLine("清洁设备");
}
public void TakeCare()
{
Console.WriteLine("养护照料");
}
}
//客户应用测试
class Client
{
[STAThread]
static void Main(string[] args)
{
IExtra e = new Facility();
e.MakeClean();
e.TakeCare();
IClean c = new Office();
c.MakeClean();
c = new WorkShop();
c.MakeClean();
Console.Read();
}
}
-
优势和缺陷
适配器模式可以将一个类的接口和另一个类的接口匹配起来,使用的前提是你不能或不想修改原来的适配器母接口(adaptee)。例如,你向第三方购买了一些类、控件,但是没有源程序,这时,使用适配器模式,你可以统一对象访问接口。但客户调用可能需要变动。
-
应用情景
下面的情景很适合应用适配器模式:
1. 对象需要利用现存的并且接口不兼容的类。
2. 你需要创建可重用的类以协作其他接口不一定兼容的类。
3. 你需要使用若干个现存的子类但又不想派生这些子类的每一个接口。
本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:九.结构型设计模式——Adapter Pattern(适配器模式) - Python技术站