深入理解Java设计模式——外观模式

目录

  • 一、什么是外观模式
  • 二、外观模式应用场景
  • 三、相关代码实现
    • 1. 需要重构的代码
    • 2. 创建业务逻辑封装
    • 3. 创建门面接口
    • 4. 优点
      • 4.1 松散耦合
      • 4.2 简单易用
      • 4.3 更好的划分访问层次
    • 5. 缺点

一、什么是外观模式

外观模式(Facade),他隐藏了系统的复杂性,并向客户端提供了一个可以访问系统的接口。这种类型的设计模式属于结构性模式。为子系统中的一组接口提供了一个统一的访问接口,这个接口使得子系统更容易被访问或者使用。

二、外观模式应用场景

深入理解Java设计模式——外观模式_第1张图片
简单来说,该模式就是把一些复杂的流程封装成一个接口供给外部用户更简单的使用。这个模式中,设计到3个角色。
1).门面角色:外观模式的核心。它被客户角色调用,它熟悉子系统的功能。内部根据客户角色的需求预定了几种功能的组合。
2).子系统角色:实现了子系统的功能。它对客户角色和Facade时未知的。它内部可以有系统内的相互交互,也可以由供外界调用的接口。
3).客户角色:通过调用Facede来完成要实现的功能。

三、相关代码实现

Maven:

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.0.1.RELEASE</version>
    </parent>
    <dependencies>
        <!-- sprinboot web -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.16.10</version>
        </dependency>
        <dependency>
            <groupId>commons-lang</groupId>
            <artifactId>commons-lang</artifactId>
            <version>2.6</version>
        </dependency>

    </dependencies>

mapper:

package com.demo.mapper;

public interface PaymentTransactionMapper {
    void updatePaymentStatus();
}

utils:

package com.demo.utils;

import lombok.extern.slf4j.Slf4j;

@Slf4j
public class HttpClientUtils {
    public static String doPost(String url, String text) {
        log.info(">>>Url:{}, text:{}", url, text);
        return "success";
    }
}

启动类:

package com.demo;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class AppFacade {
    public static void main(String[] args) {
        SpringApplication.run(AppFacade.class);
    }
}

1. 需要重构的代码

package com.demo.service;

import com.demo.mapper.PaymentTransactionMapper;
import com.demo.utils.HttpClientUtils;
import lombok.extern.slf4j.Slf4j;

import java.util.Map;

@Slf4j
public class PayCallbackService {
    // 用户下单成功后,有哪些操作?
    // 1.增加支付回调接口日志
    // 2.修改订单数据库状态为已经成功
    // 3.调用积分服务接口
    // 4.调用消息服务平台服务接口

    public boolean callback(Map<String, String> verifySignature) {
        // 1.第一步打印日志信息
        String orderId = verifySignature.get("orderId"); // 获取后台通知的数据,其他字段也可用类似方式获取
        String respCode = verifySignature.get("respCode");
        log.info("orderId:{},respCode:{}", orderId, respCode);
        // 2.修改订单状态为已经支付
        new PaymentTransactionMapper() {
            @Override
            public void updatePaymentStatus() {
                log.info(">>>修改订单状态为已经支付>>>>>");
            }
        }.updatePaymentStatus();
        // 3.调用积分接口增加积分
        HttpClientUtils.doPost("jifen.com", "积分接口");
        // 4.调用消息服务平台提示
        HttpClientUtils.doPost("msg.com", "调用消息接口");
        return true;
    }
}

2. 创建业务逻辑封装

package com.demo.service;

import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;

@Slf4j
@Service
public class LogService {
    public void logService() {
        // 1.第一步打印日志信息
        log.info(">>第一个模块:日志的收集与打印...");
    }
}

package com.demo.service;

import com.demo.mapper.PaymentTransactionMapper;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;

@Slf4j
@Service
public class PaymentService {
    public void updatePaymentStatus() {
        new PaymentTransactionMapper() {
            @Override
            public void updatePaymentStatus() {
                // 2.修改订单状态为已经支付
                log.info(">>>第二个模块:修改订单状态为已经支付>>>>>");
            }
        }.updatePaymentStatus();
    }
}

package com.demo.service;

import com.demo.utils.HttpClientUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;

@Slf4j
@Service
public class IntegralService {

    public void addIntegral() {
        // 3.调用积分接口增加积分
        HttpClientUtils.doPost("jifen.com", "积分接口");
        log.info(">>第三个模块:调用增加积分模块..");
    }
}

package com.demo.service;

import com.demo.utils.HttpClientUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;

@Slf4j
@Service
public class MsgService {

    public void sendMsg() {
        // 4.调用消息服务平台提示
        HttpClientUtils.doPost("msg.com", "调用消息接口");
        log.info(">>第四个模块:调用消息服务平台模块..");
    }
}

3. 创建门面接口

package com.demo.service;

import com.demo.facade.PayFacade;
import com.demo.mapper.PaymentTransactionMapper;
import com.demo.utils.HttpClientUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.util.Map;

@Slf4j
@Component
public class PayCallbackService {
    // 用户下单成功后,有哪些操作?
    // 1.增加支付回调接口日志
    // 2.修改订单数据库状态为已经成功
    // 3.调用积分服务接口
    // 4.调用消息服务平台服务接口
    @Autowired
    private PayFacade payFacade;

//    public boolean callback(Map verifySignature) {
//        // 1.第一步打印日志信息
//        String orderId = verifySignature.get("orderId"); // 获取后台通知的数据,其他字段也可用类似方式获取
//        String respCode = verifySignature.get("respCode");
//        log.info("orderId:{},respCode:{}", orderId, respCode);
//        // 2.修改订单状态为已经支付
//        new PaymentTransactionMapper() {
//            @Override
//            public void updatePaymentStatus() {
//                log.info(">>>修改订单状态为已经支付>>>>>");
//            }
//        }.updatePaymentStatus();
//        // 3.调用积分接口增加积分
//        HttpClientUtils.doPost("jifen.com", "积分接口");
//        // 4.调用消息服务平台提示
//        HttpClientUtils.doPost("msg.com", "调用消息接口");
//        return true;
//    }

    public boolean callback() {
        payFacade.facadecallback();
        return true;
    }

    public void baseCallback() {}
}
package com.demo.facade;

import com.demo.service.IntegralService;
import com.demo.service.LogService;
import com.demo.service.MsgService;
import com.demo.service.PaymentService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

@Component
public class PayFacade {

    @Autowired
    private LogService logService;
    @Autowired
    private PaymentService paymentService;
    @Autowired
    private IntegralService integralService;
    @Autowired
    private MsgService msgService;

    public void facadecallback() {
        // 1. 第一步 日志收集
        logService.logService();
        // 2. 第二步 修改订单状态
        paymentService.updatePaymentStatus();
        // 3. 第三步 调用积分服务接口
        integralService.addIntegral();
        // 4. 调用消息服务平台
        msgService.sendMsg();
    }
}

package com.demo.controller;

import com.demo.service.PayCallbackService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class PayController {
    @Autowired
    private PayCallbackService payCallbackService;

    @RequestMapping("/payCallback")
    public String payCallback() {
        payCallbackService.callback();
        return "success";
    }
}

运行启动类:
深入理解Java设计模式——外观模式_第2张图片
访问项目地址:http://127.0.0.1:8080/payCallback
深入理解Java设计模式——外观模式_第3张图片
深入理解Java设计模式——外观模式_第4张图片

4. 优点

4.1 松散耦合

使得客户端和子系统之间解耦,让子系统内部的模块功能更容易扩展和维护;

4.2 简单易用

客户端根本不需要知道子系统内部的实现,或者根本不需要知道子系统内部的构成,它只需要跟Facade类交互即可。

4.3 更好的划分访问层次

有些方法是对系统外的,有些方法是系统内部相互交互的使用的。子系统把那些暴露给外部的功能集中到门面中,这样就可以实现客户端的使用,很好的隐藏了子系统内部的细节。

5. 缺点

业务封装类越来越多

你可能感兴趣的:(设计模式,设计模式,外观模式)