1、什么是橋接模式?
Decouple an abstraction from its implementation so that the two can vary independently.
橋接模式(Bridge Pattern):將抽象和實(shí)現(xiàn)解耦, 使得兩者可以獨(dú)立地變化。
另外一種解釋是:一個(gè)類(lèi)存在兩個(gè)(或多個(gè))獨(dú)立變化的維度,我們通過(guò)組合的方式,讓這兩個(gè)(或多個(gè))維度可以獨(dú)立進(jìn)行擴(kuò)展。
聽(tīng)起來(lái)可能還是很深?yuàn)W,沒(méi)關(guān)系,下面通過(guò)例子講解。
2、橋接模式定義
①、Abstraction
抽象化角色:它的主要職責(zé)是定義出該角色的行為, 同時(shí)保存一個(gè)對(duì)實(shí)現(xiàn)化角色的引用, 該角色一般是抽象類(lèi)。
②、Implementor
實(shí)現(xiàn)化角色:它是接口或者抽象類(lèi), 定義角色必需的行為和屬性。
③、RefinedAbstraction
修正抽象化角色:它引用實(shí)現(xiàn)化角色對(duì)抽象化角色進(jìn)行修正。
④、ConcreteImplementor
具體實(shí)現(xiàn)化角色:它實(shí)現(xiàn)接口或抽象類(lèi)定義的方法和屬性。
3、橋接模式通用代碼實(shí)現(xiàn)
實(shí)現(xiàn)化類(lèi):
public interface Implementor {
void doSomething();
}
具體實(shí)現(xiàn)化類(lèi):
public class ConcreteImplementor1 implements Implementor{
@Override
public void doSomething() {
// 具體業(yè)務(wù)邏輯處理
}
}
public class ConcreteImplementor2 implements Implementor{
@Override
public void doSomething() {
// 具體業(yè)務(wù)邏輯
}
}
這里定義了兩個(gè),可能有多個(gè)。
抽象化角色:
public abstract class Abstraction {
// 定義對(duì)實(shí)現(xiàn)化角色的引用
private Implementor implementor;
public Abstraction(Implementor implementor){
this.implementor = implementor;
}
// 自身的行為和屬性
public void request(){
this.implementor.doSomething();
}
// 獲取實(shí)現(xiàn)化角色
public Implementor getImplementor(){
return implementor;
}
}
修正抽象化角色:
public class RefinedAbstraction extends Abstraction{
// 覆寫(xiě)構(gòu)造函數(shù)
public RefinedAbstraction(Implementor implementor){
super(implementor);
}
// 修正父類(lèi)的行為
@Override
public void request() {
super.request();
}
}
測(cè)試:
public class BridgeClient {
public static void main(String[] args) {
// 定義一個(gè)實(shí)現(xiàn)化角色
Implementor implementor = new ConcreteImplementor1();
// 定義一個(gè)抽象化角色
Abstraction abstraction = new RefinedAbstraction(implementor);
// 執(zhí)行方法
abstraction.request();
}
}
如果我們的實(shí)現(xiàn)化角色有很多的子接口, 然后是一堆的子實(shí)現(xiàn)。在構(gòu)造函數(shù)中傳遞一個(gè)明確的實(shí)現(xiàn)者, 代碼也是很清晰的。
4、橋接模式經(jīng)典例子—JDBC
我們?cè)趧傞_(kāi)始用 JDBC 直連數(shù)據(jù)庫(kù)的時(shí)候,會(huì)有這樣一段代碼:
Class.forName("com.mysql.cj.jdbc.Driver");//加載及注冊(cè)JDBC驅(qū)動(dòng)程序
String url = "jdbc:mysql://localhost:3306/sample_db?user=root&password=your_password";
Connection con = DriverManager.getConnection(url);
Statement stmt = con.createStatement();
String query = "select * from test";
ResultSet rs=stmt.executeQuery(query);
while(rs.next()) {
rs.getString(1);
rs.getInt(2);
}
如果我們想要把 MySQL 數(shù)據(jù)庫(kù)換成 Oracle 數(shù)據(jù)庫(kù),只需要把第一行代碼中的 com.mysql.cj.jdbc.Driver 換成oracle.jdbc.driver.OracleDriver 就可以了。
這種優(yōu)雅的實(shí)現(xiàn)數(shù)據(jù)庫(kù)切換方式就是利用了橋接模式。
我們首先看 Driver 類(lèi):
package com.mysql.cj.jdbc;
import java.sql.DriverManager;
import java.sql.SQLException;
public class Driver extends NonRegisteringDriver implements java.sql.Driver {
public Driver() throws SQLException {
}
static {
try {
DriverManager.registerDriver(new Driver());
} catch (SQLException var1) {
throw new RuntimeException("Can't register driver!");
}
}
}
這段代碼 Class.forName("com.mysql.cj.jdbc.Driver") 作用有兩個(gè):
①、要求 JVM 查找并加載指定的 Driver 類(lèi)。
②、執(zhí)行該類(lèi)的靜態(tài)代碼,也就是將 MySQL Driver 注冊(cè)到 DriverManager 類(lèi)中。
接著我們看 DriverManager 類(lèi):
public class DriverManager {
private final static CopyOnWriteArrayList< DriverInfo > registeredDrivers = new CopyOnWriteArrayList< DriverInfo >();
//...
static {
loadInitialDrivers();
println("JDBC DriverManager initialized");
}
//...
public static synchronized void registerDriver(java.sql.Driver driver) throws SQLException {
if (driver != null) {
registeredDrivers.addIfAbsent(new DriverInfo(driver));
} else {
throw new NullPointerException();
}
}
public static Connection getConnection(String url, String user, String password) throws SQLException {
java.util.Properties info = new java.util.Properties();
if (user != null) {
info.put("user", user);
}
if (password != null) {
info.put("password", password);
}
return (getConnection(url, info, Reflection.getCallerClass()));
}
//...
}
當(dāng)我們把具體的 Driver 實(shí)現(xiàn)類(lèi)(比如,com.mysql.cj.jdbc.Driver)注冊(cè)到 DriverManager 之后,后續(xù)所有對(duì) JDBC 接口的調(diào)用,都會(huì)委派到對(duì)具體的 Driver 實(shí)現(xiàn)類(lèi)來(lái)執(zhí)行。而 Driver 實(shí)現(xiàn)類(lèi)都實(shí)現(xiàn)了相同的接口(java.sql.Driver ),這也是可以靈活切換 Driver 的原因。
5、橋接模式優(yōu)點(diǎn)
①、抽象和實(shí)現(xiàn)分離
這也是橋梁模式的主要特點(diǎn), 它完全是為了解決繼承的缺點(diǎn)而提出的設(shè)計(jì)模式。在該模式下, 實(shí)現(xiàn)可以不受抽象的約束, 不用再綁定在一個(gè)固定的抽象層次上。
②、優(yōu)秀的擴(kuò)充能力
看看我們的例子, 想增加實(shí)現(xiàn)?沒(méi)問(wèn)題!想增加抽象, 也沒(méi)有問(wèn)題!只要對(duì)外暴露的接口層允許這樣的變化, 我們已經(jīng)把變化的可能性減到最小。
③、實(shí)現(xiàn)細(xì)節(jié)對(duì)客戶(hù)透明
客戶(hù)不用關(guān)心細(xì)節(jié)的實(shí)現(xiàn), 它已經(jīng)由抽象層通過(guò)聚合關(guān)系完成了封裝。
6、橋接模式應(yīng)用場(chǎng)景
①、如果一個(gè)系統(tǒng)需要在構(gòu)件的抽象化角色和具體化角色之間增加更多的靈活性,避免在兩個(gè)層次之間建立靜態(tài)的繼承聯(lián)系,通過(guò)橋接模式可以使它們?cè)诔橄髮咏⒁粋€(gè)關(guān)聯(lián)關(guān)系。
②、對(duì)于那些不希望使用繼承或因?yàn)槎鄬哟卫^承導(dǎo)致系統(tǒng)類(lèi)的個(gè)數(shù)急劇增加的系統(tǒng),橋接模式尤為適用。
③、一個(gè)類(lèi)存在兩個(gè)獨(dú)立變化的維度,且這兩個(gè)維度都需要進(jìn)行擴(kuò)展。
-
接口
+關(guān)注
關(guān)注
33文章
8924瀏覽量
153152 -
數(shù)據(jù)庫(kù)
+關(guān)注
關(guān)注
7文章
3899瀏覽量
65726 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4367瀏覽量
64105 -
代碼
+關(guān)注
關(guān)注
30文章
4886瀏覽量
70216 -
JDBC
+關(guān)注
關(guān)注
0文章
25瀏覽量
13584
發(fā)布評(píng)論請(qǐng)先 登錄
AG32VF-MIPI應(yīng)用場(chǎng)景
STM32待機(jī)模式適合用于那些應(yīng)用場(chǎng)景?
關(guān)于橋接模式遇到的問(wèn)題
特斯拉電動(dòng)汽車(chē)不下電,除了哨兵模式和寵物模式外還有哪些應(yīng)用場(chǎng)景
COM和CORBA的橋接與應(yīng)用

網(wǎng)橋和橋接,網(wǎng)橋和橋接是什么意思
橋接模式的目標(biāo)與設(shè)計(jì)

設(shè)計(jì)模式結(jié)構(gòu)性:橋接模式

評(píng)論