结构型模式主要是解决如何将对象和类组装成较大的结构, 并同时保持结构的灵活和?效。
结构型模式包括:适配器、桥接、组合、装饰器、外观、享元、代理,这7类
桥接模式的主要作?就是通过将抽象部分与实现部分分离,把多种可匹配的使?进?组合。说?了核?实现也就是在A类中含有B类接?,通过构造函数传递B类的实现,这个B类就是设计的 桥 。
那么这样的桥接模式,在我们平常的开发中有哪些场景呢?
JDBC多种驱动程序的实现、同品牌类型的台式机和笔记本平板、业务实现中的多类接?同组过滤服务等。这些场景都?较适合使?桥接模式进?实现,因为在?些组合中如果有如果每?个类都实现不同的服务可能会出现笛卡尔积,?使?桥接模式就可以?常简单。
模拟?个第三?平台来承接各个?付能?,同时使??家的?脸让?户?付起来更加容易。那么这?就出现了多?付与多模式的融合使?,如果给每?个?付都实现?次不同的模式,即使是继承类也需要开发好多。
?且随着后?接?了更多的?付服务或者?付?式,就会呈爆炸似的扩展。
这样的场景该如何实现?
没有?个类写不完的需求
public class PayController {
private Logger logger = LoggerFactory.getLogger(PayController.class);
public boolean doPay(String uId, String tradeId, BigDecimal amount, int channelType, int modeType) {
// 微信支付
if (1 == channelType) {
logger.info("模拟微信渠道支付划账开始。uId:{} tradeId:{} amount:{}", uId, tradeId, amount);
if (1 == modeType) {
logger.info("密码支付,风控校验环境安全");
} else if (2 == modeType) {
logger.info("人脸支付,风控校验脸部识别");
} else if (3 == modeType) {
logger.info("指纹支付,风控校验指纹信息");
}
}
// 支付宝支付
else if (2 == channelType) {
logger.info("模拟支付宝渠道支付划账开始。uId:{} tradeId:{} amount:{}", uId, tradeId, amount);
if (1 == modeType) {
logger.info("密码支付,风控校验环境安全");
} else if (2 == modeType) {
logger.info("人脸支付,风控校验脸部识别");
} else if (3 == modeType) {
logger.info("指纹支付,风控校验指纹信息");
}
}
return true;
}
}
?户ID 、 交易ID 、 ?额 、 渠道 、 模式
,来控制?付?式ifelse
应该是最差的?种写法,即使写 ifelse
也是可以优化的?式去写的。【测试验证】
@Test
public void test_pay() {
PayController pay = new PayController();
System.out.println("\r\n模拟测试场景;微信支付、人脸方式。");
pay.doPay("weixin_1092033111", "100000109893", new BigDecimal(100), 1, 2);
System.out.println("\r\n模拟测试场景;支付宝支付、指纹方式。");
pay.doPay("jlu19dlxo111","100000109894",new BigDecimal(100), 2, 3);
}
虽然已经满?了我们的不同?付类型和?付模式的组合,但是这样的代码在后?的维护以及扩展都会变得?常复杂。
接下来使?桥接模式来进?代码优化,也算是?次很?的重构。
从上?的 ifelse
?式实现来看,这是两种不同类型的相互组合。那么就可以把?付?式和?付模式进?分离通过抽象类依赖实现类的?式进?桥接,通过这样的拆分后?付与模式其实是可以单独使?的,当需要组合时候只需要把模式传递给?付即可。
桥接模式的关键是选择的桥接点拆分,是否可以找到这样类似的相互组合,如果没有就不必要?得使?桥接模式。
【工程结构】
【桥接模式模型结构】
Pay
是?个抽象类
,往下是它的两个?付类型实现;微信?付、?付宝?付。IPayMode
是?个接?
,往下是它的两个?付模型;刷脸?付、指纹?付。注意,每种?付?式的不同,刷脸和指纹校验逻辑也有差异,可以使?适配器模式进?处理,这?不做介绍,可以看适配器模式。
【?付类型桥接抽象类】
public abstract class Pay {
protected Logger logger = LoggerFactory.getLogger(Pay.class);
protected IPayMode payMode;
public Pay(IPayMode payMode) {
this.payMode = payMode;
}
public abstract String transfer(String uId, String tradeId, BigDecimal amount);
}
transfer
,以及桥接接? IPayMode
,并在构造函数中?户???选择?付?式。【两个?付类型的实现】
public class WxPay extends Pay {
public WxPay(IPayMode payMode) {
super(payMode);
}
public String transfer(String uId, String tradeId, BigDecimal amount) {
logger.info("模拟微信渠道支付划账开始。uId:{} tradeId:{} amount:{}", uId, tradeId, amount);
boolean security = payMode.security(uId);
logger.info("模拟微信渠道支付风控校验。uId:{} tradeId:{} security:{}", uId, tradeId, security);
if (!security) {
logger.info("模拟微信渠道支付划账拦截。uId:{} tradeId:{} amount:{}", uId, tradeId, amount);
return "0001";
}
logger.info("模拟微信渠道支付划账成功。uId:{} tradeId:{} amount:{}", uId, tradeId, amount);
return "0000";
}
}
public class ZfbPay extends Pay {
public ZfbPay(IPayMode payMode) {
super(payMode);
}
public String transfer(String uId, String tradeId, BigDecimal amount) {
logger.info("模拟支付宝渠道支付划账开始。uId:{} tradeId:{} amount:{}", uId, tradeId, amount);
boolean security = payMode.security(uId);
logger.info("模拟支付宝渠道支付风控校验。uId:{} tradeId:{} security:{}", uId, tradeId, security);
if (!security) {
logger.info("模拟支付宝渠道支付划账拦截。uId:{} tradeId:{} amount:{}", uId, tradeId, amount);
return "0001";
}
logger.info("模拟支付宝渠道支付划账成功。uId:{} tradeId:{} amount:{}", uId, tradeId, amount);
return "0000";
}
}
【 定义?付模式接?】
public interface IPayMode {
boolean security(String uId);
}
【三种?付模式?控(刷脸、指纹、密码)】
public class PayCypher implements IPayMode{
protected Logger logger = LoggerFactory.getLogger(PayCypher.class);
public boolean security(String uId) {
logger.info("密码支付,风控校验环境安全");
return true;
}
}
public class PayFaceMode implements IPayMode{
protected Logger logger = LoggerFactory.getLogger(PayCypher.class);
public boolean security(String uId) {
logger.info("人脸支付,风控校验脸部识别");
return true;
}
}
public class PayFingerprintMode implements IPayMode{
protected Logger logger = LoggerFactory.getLogger(PayCypher.class);
public boolean security(String uId) {
logger.info("指纹支付,风控校验指纹信息");
return true;
}
}
实现了三种?付模式(刷脸、指纹、密码)的?控校验,在?户选择不同?付类型的时候,则会进?相应的?控拦截以此保障?付安全。
【单元测试】
@Test
public void test_pay() {
System.out.println("\r\n模拟测试场景;微信支付、人脸方式。");
Pay wxPay = new WxPay(new PayFaceMode());
wxPay.transfer("weixin_1092033111", "100000109893", new BigDecimal(100));
System.out.println("\r\n模拟测试场景;支付宝支付、指纹方式。");
Pay zfbPay = new ZfbPay(new PayFingerprintMode());
zfbPay.transfer("jlu19dlxo111","100000109894",new BigDecimal(100));
}
通过模拟微信与?付宝两个?付渠道在不同的?付模式下, 刷脸 、 指纹 、 密码 ,的组合从?体现了桥接模式的在这类场景中的合理运?。简化了代码的开发,给后续的需求迭代增加了很好的扩展性。
从桥接模式的实现形式来看满?了单?职责和开闭原则,让每?部分内容都很清晰易于维护和拓展,但如果我们是实现的?内聚的代码,那么就会很复杂。所以在选择重构代码的时候,需要考虑好整体的设计,否则选不到合理的设计模式,将会让代码变得难以开发。
任何?种设计模式的选择和使?都应该遵顼符合场景为主,不要刻意使?。?且统?场景因为业务的复杂从?可能需要使?到多种设计模式的组合,才能将代码设计的更加合理。但这种经验需要从实际的项?中学习经验,并提不断的运?。