美文网首页java进阶干货
类的创建与销毁(一)

类的创建与销毁(一)

作者: TinyAndNeo | 来源:发表于2017-08-10 16:49 被阅读0次

优雅的创建实例

通过构造函数创建实例

通过多参数构造函数重载,提供多个实例化入口

class Demo {
    Demo(){
        this(0, 0);
    }
    Demo(int var){
        this(var, 0);
    }
    Demo(int var1, int var2){
        //implement logic
    }
}

优点:

  • 简单的实现了多个实例化入口,可以通过级联调用

缺点:

  • 实例返回的对象无法子类化、无法单例化等,灵活性差
  • 多实例参数时并且这些参数不确定需要时,需要提供大量构造重载入口,繁琐

通过静态工厂方法创建实例

通过静态方法提供多内容的实例化

class Demo {
    private Demo() {}
    public static Demo getInstance() {
        return new Demo();
    }
}

优点:

  • 可以根据不同的实例用途进行方法命名,提高调用者的对不同实例入口的认知
  • 可以根据业务需求和当前状态的选择是返回一个新的实例还是一个已缓存的类实例
  • 返回类型可以是返回的具体实例的超类,该超类可以是一个协议接口,从而实现了返回值的子类化

缺点:

  • 当工厂方法不在需要实例化的类中时,需要被实例化的类无法做到子类化
  • 多实例参数并且这些参数不确定需要时,需要提供大量静态方法,繁琐

通过传入JavaBean创建实例

通过打包多个参数到一个JavaBean对象,来减少方法签名数

class DataBean {
    int var1;
    int var2;
    int var3;
}

class Demo {
    Demo(DataBean data){}
}

优点:

  • 多参数的构造中,可以减少参数的传递,减少调用者的负担

缺点:

  • 传入JavaBean的方式是先传入JavaBean,再获取实例,而该JavaBean的内容是可变的,这给实例带来了不确定性。

通过Builder构建器创建实例

通过创建Builder构建器,从而创建对象

class Demo {
    private int var1;
    private String var2;

    Demo(Builder builder) {
        this.var1 = builder.var1;
        this.var2 = builder.var2;
    }

    static class Builder {
        private int var1;
        private String var2;

        public Builder setVar1(int var){
            this.var1 = var;
            return this;
        }

        public Builder setVar2(String var) {
            this.var2 = var;
            return this;
        }

        public build(){
            //1. check the fileds in builder
            return new Demo(this);
        }
    }
}

优点:

  • 相比多个构造器,Builder模式可以实现多参数的自由组合,也有利于后期多参数扩展,可维护性更好
  • 相比JavaBean,传入的参数是当前对象可控可检查的,入参非法则不允许创建对象,更加安全
  • 链式调用,更符合函数式编程实现,对每个参数有强约束,更加易于阅读和操作

缺点:

  • 多构造器模式更加冗长,适用于大于4个参数的情况
  • 对性能开销有一定的负面影响

最优的单例实现方式

枚举单例和Atomic原子引用

//枚举法
public class EnumSingleton{
    private EnumSingleton(){}
    public static EnumSingleton getInstance(){
        return Singleton.INSTANCE.getInstance();
    }
    
    private static enum Singleton{
        INSTANCE;
        
        private EnumSingleton singleton;
        //JVM会保证此方法绝对只调用一次
        private Singleton(){
            singleton = new EnumSingleton();
        }
        public EnumSingleton getInstance(){
            return singleton;
        }
    }
}

//原子操作 + 自旋锁(乐观锁)
public class AtomicSingleton {  
    private static AtomicReference<AtomicSingleton> INSTANCE = new AtomicReference<>();
    private AtomicSingleton() {}
    public static AtomicSingleton getInstace(){
        while (true) {
            AtomicSingleton current = INSTANCE.get();
            if(current != null){ return current; }  
            current = new AtomicSingleton();
            if(INSTANCE.compareAndSet(null, current)) { return current; }
        }
    }
}

优点:

  • 多线程安全
  • 枚举:延迟加载,原子操作:乐观锁,保证了性能的最优化
  • 绝对的序列化安全和反射攻击安全

不可实例化类的声明

class Demo {
    // 私有化缺省构造,让调用者无法实例化
    private Demo(){
        //若有反射攻击,抛出一个异常
        throw new AssertionError();
    }
}

避免创建不必要的对象

//code 1
String s1 = new String("str");
//code 2
String s2 = "str";
  • code 1中,“str”已经为一个创建的实例,通过String的构造器再创建一个对象,这是一个无意义的操作
  • code2中,s2指向的不是一个String对象,而是常量池中的一个String地址,若“str”在常量池中已经存在相同字面值的一份,则引用直接指向它,否则才创建,这个做法类似于基本类型的创建机制
public static void main(String[] args) {
    Integer sum = 0;
    for (int i = 0; i <= Integer.MAX_VALUE; i++) {
        sum += i;
    }
    System.out.println(sum);
}
  • 上述代码中,由于装箱的存在,创建了2^{31}Integer对象,大大浪费了性能,所以尽量使用基础类型,避免无意识的装箱操作
  • 但不是说尽量避免创建对象,首先要保证业务的通畅和代码的可维护性、可阅读性。不要为了减少对象的创建,来做一些额外的操作(如维护对象池),若维护的对象是轻量级,这个操作反而降低了效率,并且增加了代码的复杂度。而对足够重量级的对象(如JDBC)进行这种操作才能带来收益

相关文章

  • 类的创建与销毁(一)

    优雅的创建实例 通过构造函数创建实例 通过多参数构造函数重载,提供多个实例化入口 优点: 简单的实现了多个实例化入...

  • 类的创建与销毁(二)

    销毁实例时注意事项 内存泄露 出现内存泄露的3种常见情况及应对原则 类中存在自我管理内存现象 类中存在容器(如数组...

  • 线程学习笔记(一)

    01 线程的创建与销毁 创建运行一般情况,线程的创建有两种方式1.实现Runnable接口,放入Thread类中执...

  • iOS文档补完计划--NSObject

    目录 NSObject类 类的初始化loadinitialize 创建、复制和销毁allocallocWithZo...

  • 设计模式-创建者模式总结

    创建者模式的特点及使用场景 《Effective Java》—— 创建与销毁对象 一章中有写道:当一个类中有大量的...

  • 进程的遍历、获取与销毁

    进程的遍历、获取与销毁 创建快照遍历 函数实现 进程的获取与销毁 exp:

  • (一)设计模式 单例模式

    设计模式代表了最佳的实践 引言 创建型模式。 主要特点:简单,样式多。 主要解决:一个全局使用的类频繁地创建与销毁...

  • 创建型模式-单例模式

    介绍 使用意图: 让一个类只有一个实例化对象。 主要解决: 一个全局使用的类频繁地创建与销毁。 何时使用: 想控制...

  • 07-单例模式与异常处理

    一、单例模式 单例模式主要是为了确保只有单个对象被创建,主要解决一个类的对象频繁地创建与销毁, 我们通过如下示例来...

  • UObject的创建与销毁

    1、使用NewObject进行创建 2、使用ConditionalBeginDestroy()销毁UObject

网友评论

    本文标题:类的创建与销毁(一)

    本文链接:https://www.haomeiwen.com/subject/ssgcrxtx.html