yufan.me/source/_posts/2015-03-19.md
2024-06-14 02:15:18 +08:00

6.7 KiB
Raw Blame History

title tags categories permalink featureImage date
如何更好地创建对象
Java
学习 the-better-way-to-make-objects https://cat.yufan.me/cats/ame/ruby-code-snippet.jpg 2015-03-19 01:03:00

写Java一类的面相对象语言的程序员常常会遇到这么一个冷笑话我今年都30了还没找到对象怎么办简单啊new 一个对象就好。

当然这只是一个冷笑话,所谓的 new 一个对象无非不就是调用这个类的构造方法去创建对象。乍一看也没什么问题写个类ABC用的时候new ABC()就好了。那么,会想一下,我们读书的时候,老师一定会说一个类的构造方法可以允许传入参数,甚至根据传入参数的不同创建多个构造方法。

学过面向对象的你一定会说没错啊就是这样方法重载签名校验。都是一个方法Perfect那么我们设想下面这么一种情况

假如我们有一个DTO类 CredentialsAuthParam 作为对外接口的传入参数作为一个POJO类我们一般就是定义一堆属性然后一堆Getter、Setter比如我们可以这么定义

package me.yufan.dto;

import org.apache.commons.lang3.builder.ToStringBuilder;
import org.apache.commons.lang3.builder.ToStringStyle;
import java.io.Serializable;

public class CredentialsAuthParam implements Serializable {

    private static final long serialVersionUID = -1L;

    private String source;

    private String validationCode;

    private String operator;

    private String remark;

    public CredentialsAuthParam() {
    }

    public String getValidationCode() {
        return validationCode;
    }

    public void setValidationCode(String validationCode) {
        this.validationCode = validationCode;
    }

    public String getOperator() {
        return operator;
    }

    public void setOperator(String operator) {
        this.operator = operator;
    }

    public String getRemark() {
        return remark;
    }

    public void setRemark(String remark) {
        this.remark = remark;
    }

    public String getSource() {
        return source;
    }

    public void setSource(String source) {
        this.source = source;
    }

    @Override
    public String toString() {
        return ToStringBuilder.reflectionToString(this,
                ToStringStyle.SHORT_PREFIX_STYLE);
    }
}

看着很棒,多么简单,用 CredentialsAuthParam 的时候 new 一下,作为传入参数调用接口的方法就好。

那么如果我的对象要设置属性怎么办一个个Setter为何不能在创建对象时创建呢比如构造方法里面指定参数没错然后我们就有了下面的一堆构造方法。

public CredentialsAuthParam(String source, String validationCode) {
    this.source = source;
    this.validationCode = validationCode;
}

public CredentialsAuthParam(String source, String validationCod, String operator) {
    this.source = source;
    this.validationCode = validationCode;
    this.operator = operator;
}

public CredentialsAuthParam(String source, String validationCod,
                              String operator, String remark) {
    this.source = source;
    this.validationCode = validationCode;
    this.operator = operator;
    this.remark = remark;
}

那么问题来了参数这么多又都是String创建一个对象多麻烦。又比如我只想指定其中3个属性又都是String但现在已经有了一个为3个String 参数的构造方法。怎么办?

Java对方法重载的判断是按照签名的类型进行校验。所以方法参数顺序需要开发在使用时自己指定也许我方法参数是String source, String validationCode。但结果我因为复制粘贴不仔细变成了this.source = validationCode; this.validationCode = source;(别笑,你忙着写垃圾代码的时候就会出错) 或者构造方法使用者弄混了两个参数的顺序。那么就会出事啦。

其实如果你看过Java圣经Effective Java的话一定会注意到里面说过构造器Builder多个参数的构造方法一定要考虑使用构造器。比如我们可以这么写

package me.yufan.dto;

public class CredentialsAuthParamBuilder {

    private String source = "";
    private String validationCode;
    private String operator = "";
    private String remark = "";

    public CredentialsAuthParamBuilder setSource(String source) {
        this.source = source;
        return this;
    }

    public CredentialsAuthParamBuilder setValidationCode(String validationCode) {
        this.validationCode = validationCode;
        return this;
    }

    public CredentialsAuthParamBuilder setOperator(String operator) {
        this.operator = operator;
        return this;
    }

    public CredentialsAuthParamBuilder setRemark(String remark) {
        this.remark = remark;
        return this;
    }

    public CredentialsAuthParam createCredentialsAuthParam() {
        return new CredentialsAuthParam(source, validationCode, operator, remark);
    }
}

然后将原来类的构造方法定义为Protected然后创建对象的时候只需要

new CredentialsRequestParamBuilder()
.setRemark("remark")
.createCredentialsRequestParam();

按照需求,设置几个属性就加几个 set 方法。

接下来,我们说说单例模式:

单例Singleton顾名思义就是只被实例化一次的类。比如我在MVC的拦截器中需要调用一个公共类它里面存放的东西是大家共享的我可以这么写

public class AuthorityInterceptorHelper {

    private static AuthorityInterceptorHelper instance;

    private AuthorityInterceptorHelper() {

    }

    public static AuthorityInterceptorHelper getInstance() {
        if (instance == null) {
            synchronized (AuthorityInterceptorHelper.class) {
                if (instance == null) {
                    instance = new AuthorityInterceptorHelper();
                }
            }
        }
        return instance;
    }
//    other code ...
}

我首先要在内部定义一个自身的静态对象然后将构造方法私有getInstance()会先去看静态对象存在否,不存在,先加锁,也许加锁期间其他方法先调用此方法创建对象,再看看对象是否存在,不存在,创建对象解锁。

看起来没什么问题,代码严密,十分规范,大家都是这么写的。但是,单例了么?定义为私有的方法一定没法访问了么?反射呢?

其实枚举类便可以轻松实现需求,我们只需如下写法:

public enum AuthorityInterceptorHelper {

    INSTANCE;

//    other function ...
}