为什么程序员喜欢用大量的if... ...else if ;不喜欢用设计模式+反射+自定义注解

面向过程设计和面向对象设计的主要区别是:

   是否在业务逻辑层使用冗长的if else判断。如果你的代码还在大量使用if else,当然,界面表现层除外,即使你使用Java/C#这样完全面向对象的语言,也只能说明你的思维停留在传统的面向过程语言上。 

传统思维设计

    业务逻辑层为什么会使用大量的if else呢?其实目的也是为了重用,但是这是面向过程编程的重用,程序员只看到代码重用,因为他看到if else几种情况下大部分代码都是重复的,只有部分不同,因此使用if else可以避免重复代码。

好了,废话不多说!

    假定有这样的一种情况,需要根据传入的参数类型,选择不同的短信服务商来发送相应的短信操作。

传统的if...else来实现的话,就类似如下代码:

首先,SMSTypeEnum是一个枚举类,这个没啥可讲的,有问题自行百度一下

/**
 * 短信类型枚举类
 *
 * @author somnus
 * @date 2019年09月10日
 */
public enum SMSTypeEnum {
    TENGXUNYUN_SMS("txy","腾讯云短信"),
    SI_XUN_TONG_SMS("sxt","思迅通"),
    ALIYUN_SMS("aliyun","阿里云短信"),
    ZHONG_YU_WEI_XIN_SMS("zywx","中昱维信");
    String type;
    String name;
    SMSEnum(String type,String name){
        this.type=type;
        this.name=name;
    }
}
image.gif

执行的main方法,可以看出来这里臃肿。

public static void main(String[] args) {
        String type="aliyun";//阿里云短信
        if(type.equals(SMSTypeEnum.ALIYUN_SMS.getType())){
            //处理阿里云短信...
            System.out.println("执行:"+SMSTypeEnum.ALIYUN_SMS.getName());
        }else if(type.equals(SMSTypeEnum.TENGXUNYUN_SMS.getType())){
            //处理腾讯云短信短信...
        }else if(type.equals(SMSTypeEnum.SI_XUN_TONG_SMS.getType())){
            //处理思迅通短信...
        }else if(type.equals(SMSTypeEnum.ZHONG_YU_WEI_XIN_SMS.getType())){
            //处理中昱维信短信...
        }else{
            //...
        }
    }
image.gif

最后的返回结果:

image
image.gif

设计模式

  经常有人说用设计模式,设计模式是不错,但是很难用到,其实如果你使用if else来写代码时,就直接在写业务逻辑,只不过使用简单的判断语句来作为现实情况的替代者。设计模式我们就说一下策略模式,想到策略模式就想到了《三国演义》中的锦囊妙计。我们下面就开始通过设计模式

首先我们来建一个消息实体类:

/**
 * 短信消息封装
 *
 * @author somnus
 * @date 2019年09月10日
 */
public class SMSInfo {
    /**
     * 主键
     */
    private Integer id;
    /**
     * 平台信息
     */
    private String appId;
    /**
     * 短信类型1,发送2,校验
     */
    private Integer msgType;
    /**
     * 类型名称
     */
    private String msgName;
    /**
     * 短信验证码
     */
    private String code;
    /**
     * 模版id
     */
    private String templateId;
    /**
     * 短信内容
     */
    private String content;
    /**
     * 创建时间
     */
    private String createTime;

    //省略get,set方法
}
image.gif

我们先建一个策略接口

/**
 * 短信发送策略接口类
 *
 * @author somnus
 * @date 2019年09月10日
 */
public interface SMSStrategyInterface {
    /**
     * 短信发送
     * @param smsInfo
     * @return
     */
    public BaseResponse sendSms(SMSInfo smsInfo);
}
image.gif

然后是各短信服务商的实现:

/**
 * 阿里云短信发送策略处理
 *
 * @author somnus
 * @date 2019年09月10日
 */
public class AliyunSmsStrategy implements SMSStrategyInterface {
    /**
     * 短信发送
     * @param smsInfo
     * @return
     */
    public BaseResponse sendSms(SMSInfo smsInfo) {
        //以上处理发送短信逻辑这里就不处理了
        return BaseResponse.ok();
    }
}
image.gif

好了,剩下的就是我们的核心部分了

新建一个ManagerStrategy管理类

/**
 * 短信管理策略类
 *
 * @author somnus
 * @date 2019年09月10日
 */
public class ManagerStrategy {
    //短信策略接口
    SMSStrategyInterface smsStrategyInterface;

    ManagerStrategy(SMSStrategyInterface smsStrategyInterface){
        this.smsStrategyInterface=smsStrategyInterface;
    }

    /**
     * 执行短信发送
     * @param smsInfo
     * @return
     */
    public BaseResponse executeSendSMS(SMSInfo smsInfo){
        return smsStrategyInterface.sendSms(smsInfo);
    }
}
image.gif

下面我们下个main方法测试一下,这个是可以优化的,我们可以结合下面的反射以及自定义注解的方式进行优化

    public static void main(String[] args) {
        ManagerStrategy managerStrategy=new ManagerStrategy(new AliyunSmsStrategy());
        SMSInfo smsInfo=new SMSInfo();
        //省略短信信息处理逻辑
        System.out.println(managerStrategy.executeSendSMS(smsInfo));
    }
image.gif

我们可以看到返回结果:

image
image.gif

反射

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

我们先来新建一个短信反射枚举类:

/**
 * 反射短信枚举类
 *
 * @author somnus
 * @date 2019年09月10日
 */
public enum  RefelSMSEnum {

    ALIYUN_TYPE_SMS("aliyun","com.yunyi.common.service.reflect.AliyunSMS");
    public String type;
    public String clazz;
    RefelSMSEnum(String type,String clazz){
        this.type=type;
        this.clazz=clazz;
    }
    //省略get,set方法
}
image.gif

通过修改ManagerStrategy管理类来处理

/**
 * 短信管理反射类
 *
 * @author somnus
 * @date 2019年09月10日
 */
public class ManagerStrategy {
    //通过map方式管理类型以及包名
    private static Map<String,String> strategyMap = new HashMap<String, String>();
    static {
        for(RefelSMSEnum refelSMSEnum:RefelSMSEnum.values()){
            strategyMap.put(refelSMSEnum.type,refelSMSEnum.clazz);
        }
    }

    /**
     * 执行短信发送
     * @param type
     * @return
     */
    public static void executeSendSMS(String type){
        String classPath=strategyMap.get(type);
        if(null==classPath){
            throw new NullPointerException("获取短信类型不存在");
        }
        try{
            /*
             * 通过反射将RefelSMSEnum中映射的类实例化
             * */
            Class clazz=Class.forName(classPath);
            Method excute =clazz.getDeclaredMethod("sendSms");
            excute.invoke(clazz.newInstance());
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}
image.gif

我们建一个main方法来测试一下

 public static void main(String[] args) {
        String type="aliyun";
        ManagerStrategy.executeSendSMS(type);
    }
image.gif

执行结果:

image
image.gif

自定义注解

    从JDK5开始,Java增加对元数据的支持,也就是注解,注解与注释是有一定区别的,可以把注解理解为代码里的特殊标记,这些标记可以在编译,类加载,运行时被读取,并执行相应的处理。通过注解开发人员可以在不改变原有代码和逻辑的情况下在源代码中嵌入补充信息。

我们新建建一个SMSAnnotation注解类:

/**
 * 短信发送自定义注解
 *
 * @author somnus
 * @date 2019年09月10日
 */
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface SMSAnnotation {
    //短信类型
    String type() default "aliyun";
    //短信类型名称
    String name();
}
image.gif

我们可以通过以下的方式来为这个 value 传值:

@Target(value = {ElementType.FIELD})
image.gif

被这个 @Target 注解修饰的注解将只能作用在成员字段上,不能用于修饰方法或者类。其中,ElementType 是一个枚举类型,有以下一些值:

  • ElementType.TYPE:允许被修饰的注解作用在类、接口和枚举上
  • ElementType.FIELD:允许作用在属性字段上
  • ElementType.METHOD:允许作用在方法上
  • ElementType.PARAMETER:允许作用在方法参数上
  • ElementType.CONSTRUCTOR:允许作用在构造器上
  • ElementType.LOCAL_VARIABLE:允许作用在本地局部变量上
  • ElementType.ANNOTATION_TYPE:允许作用在注解上
  • ElementType.PACKAGE:允许作用在包上

@Retention 用于指明当前注解的生命周期,它的基本定义如下:

同样的,它也有一个 value 属性:

@Retention(value = RetentionPolicy.RUNTIME
image.gif

这里的 RetentionPolicy 依然是一个枚举类型,它有以下几个枚举值可取:

  • RetentionPolicy.SOURCE:当前注解编译期可见,不会写入 class 文件
  • RetentionPolicy.CLASS:类加载阶段丢弃,会写入 class 文件
  • RetentionPolicy.RUNTIME:永久保存,可以反射获取

在实现类上加上我们的自定义注解:

/**
 * 阿里云短信实现类
 *
 * @author somnus
 * @date 2019年09月10日
 */
@SMSAnnotation(type = "aliyun",name="阿里云短信")
public class AliyunSMS implements SMSStrategyInterface {
    /**
     * 短信发送
     * @param smsInfo
     * @return
     */
    public BaseResponse sendSms(SMSInfo smsInfo) {
        //省略短信逻辑处理
        return BaseResponse.ok();
    }

}
image.gif
   以上操作执行完毕后,继续通过修改ManagerStrategy管理类来处理,唯一变的需要指定扫描的包,通过扫描包的方式去查找我们的自定义注解,自定义注解执行方式和反射执行差不多,所以这里就不做过多的操作。

本片文章是受网上一个恶搞的图片整理的,知道有更好的处理问题的方式,为什么还要选择用最笨的和最麻烦的方式去处理呢?火车,动车,飞机,到达的目的地是一样。只是速度不一样,火车不是照样有人做吗?看完这篇文章,你有什么想说的,欢迎留言交流。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容