策略模式小记

策略模式的demo

策略一:ALI_PAY(“1”,“com.fjw.model2.AliPay”),加载类的实例调用

场景:不同code,获取支付类支付方式不同

创建公共的策略枚举类
PayEnumStrategy:
public enum PayEnumStrategy {
    ALI_PAY("1","com.fjw.model2.AliPay"),  
    WX_PAY("2","com.fjw.model2.WxPay");

    private String code;
    private  String className;

    PayEnumStrategy() {
    }
    PayEnumStrategy(String code, String className) {
        this.code = code;
        this.className = className;
    }

    public static  String getClassByCode(String code){
        String className = "";
        if (StringUtils.isEmpty(code)) {
            System.out.println("code错误");
            return className;
        }
        //遍历所有的可选策略值,找到对应的类
        for (PayEnumStrategy payEnumStrategy : PayEnumStrategy.values()) {
            if (payEnumStrategy.code.equalsIgnoreCase(code)){
               className =  payEnumStrategy.className;
               break;
            }
        }
        return className;
    }
    
    //不需要setter
    public String getCode() {
        return code;
    }
    public String getClassName() {
        return className;
    }
}
创建共有的行为
public interface PayStrategy {
    String pay();
}
创建策略可以执行的方式
public class AliPay implements PayStrategy {
    @Override
    public String pay() {
        System.out.println("ali支付");
       return "支付包支付";
    }
}

public class WxPay implements PayStrategy {
    @Override
    public String pay() {
        System.out.println("微信支付");
       return "微信支付";
    }
}
根据编码获取执行方式策略实例
public class StrategyFactory {
    public static  PayStrategy getPayStrategyByCode(String code){
        try {
           return (PayStrategy) 				          Class.forName(PayEnumStrategy.getClassByCode(code)).newInstance();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
}
业务的调用执行
public class PayContextStrategy {
    public  static String toPay(String code){
        if (StringUtils.isEmpty(code)){
            return "code不能为空";
        }
        PayStrategy strategy = StrategyFactory.getPayStrategyByCode(code);
        return  strategy.pay();
    }

    public static void main(String[] args) {
        PayStrategy strategy = StrategyFactory.getPayStrategyByCode("1");
        strategy.pay();
    }

}
策略二:实现接口的,以map<方式,要执行的方法>交给spring管理

场景:根据不同类型采取不同的解析方式

策略模式的使用:
  • 一个接口或者抽象类,里面两个方法(一个方法匹配类型,一个可替换的逻辑实现方法)
  • 不同策略的差异化实现(就是说,不同策略的实现类)
  • 使用策略模式
一个接口,两个方法
public interface IFileStrategy {
    //属于哪种文件解析类型
    FileTypeResolveEnum gainFileType();
    
    //封装的公用算法(具体的解析方法)
    void resolve(Object objectparam);
}

不同策略的差异化实现
//A类型策略具体实现
@Component
public class AFileResolve implements IFileStrategy {
    
    @Override
    public FileTypeResolveEnum gainFileType() {
        return FileTypeResolveEnum.File_A_RESOLVE;
    }

    @Override
    public void resolve(Object objectparam) {
      logger.info("A 类型解析文件,参数:{}",objectparam);
      //A类型解析具体逻辑
    }
}

@Component  //B
public class BFileResolve implements IFileStrategy {
   
    @Override
    public FileTypeResolveEnum gainFileType() {
        return FileTypeResolveEnum.File_B_RESOLVE;
    }


    @Override
    public void resolve(Object objectparam) {
      logger.info("B 类型解析文件,参数:{}",objectparam);
      //B类型解析具体逻辑
    }
}

@Component //默认
public class DefaultFileResolve implements IFileStrategy {

    @Override
    public FileTypeResolveEnum gainFileType() {
        return FileTypeResolveEnum.File_DEFAULT_RESOLVE;
    }

    @Override
    public void resolve(Object objectparam) {
      logger.info("默认类型解析文件,参数:{}",objectparam);
      //默认类型解析具体逻辑
    }
}
使用策略模式

借助spring的生命周期,使用ApplicationContextAware接口,把用的策略,初始化到map里面。然后对外提供resolveFile方法即可。map<枚举类,执行方法>

@Component
public class StrategyUseService implements ApplicationContextAware{

  
    private Map<FileTypeResolveEnum, IFileStrategy> iFileStrategyMap = new ConcurrentHashMap<>();

    public void resolveFile(FileTypeResolveEnum fileTypeResolveEnum, Object objectParam) {
        IFileStrategy iFileStrategy = iFileStrategyMap.get(fileTypeResolveEnum);
        if (iFileStrategy != null) {
            iFileStrategy.resolve(objectParam);
        }
    }

    //把不同策略放到map
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        Map<String, IFileStrategy> tmepMap = applicationContext.getBeansOfType(IFileStrategy.class);
        
        tmepMap.values().forEach(strategyService -> iFileStrategyMap.put(strategyService.gainFileType(), strategyService));
    }
}
Protobuf是一种高效的序列化协议,可以用于数据交换和数据存储。它的主要优势是大小小,速度快,可扩展性强。下面是使用Protobuf的一些小记: 1. 定义消息格式 首先,需要定义消息格式,以便Protobuf可以将数据序列化和反序列化。消息格式定义在.proto文件中,使用protobuf语言编写。例如,下面是一个简单的消息格式定义: ``` syntax = "proto3"; message Person { string name = 1; int32 age = 2; } ``` 这个消息格式定义了一个名为Person的消息,包含两个字段:name和age。 2. 生成代码 一旦消息格式定义好,就可以使用Protobuf编译器生成代码。编译器将根据消息格式定义生成相应的代码,包括消息类、序列化和反序列化方法等。可以使用以下命令生成代码: ``` protoc --java_out=. message.proto ``` 这将生成一个名为message.pb.javaJava类,该类包含Person消息的定义以及相关方法。 3. 序列化和反序列化 一旦生成了代码,就可以使用Protobuf序列化和反序列化数据。例如,下面是一个示例代码,将一个Person对象序列化为字节数组,并将其反序列化为另一个Person对象: ``` Person person = Person.newBuilder() .setName("Alice") .setAge(25) .build(); byte[] bytes = person.toByteArray(); Person deserializedPerson = Person.parseFrom(bytes); ``` 这个示例代码创建了一个Person对象,将其序列化为字节数组,然后将其反序列化为另一个Person对象。在这个过程中,Protobuf使用生成的代码执行序列化和反序列化操作。 以上是使用Protobuf的一些基本步骤和注意事项,希望对你有所帮助!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值