美文网首页
第二章 创建和销毁对象

第二章 创建和销毁对象

作者: wozhenshihaoren | 来源:发表于2018-09-11 15:56 被阅读0次

    1.考虑用静态工厂方法代替构造器

    静态工厂方法与构造器不同的优势在于:

    1.他们有名称(容易阅读)

    2.不用每次调用创建一个新的对象

    3.返回原返回类型的任何子类型的对象(服务提供者框架)

    4.在创建参数化类型实例的时候,使代码变得更简洁

    例: Map<String,List<String>> map = new HashMap<String,List<String>>();

    例如提供静态工厂方法: public static <K, V> HashMap<K, V> newInstance { return new HashMap<K, V>(); }

    变成: Map> map = HashMap.newInstance(); 

    缺点在于:

    1.类如果不含公有的或者受保护的构造器,就不能被子类化 (推荐使用组合,而不是继承)

    2.与其他静态方法没有区别(不容易识别),但是有一些命名习惯表示 valueOf() of() getInstance() newInstance() getType() newType()

    思想上就是提供公有构造器的时候 先想到静态工厂

    2.遇到多个构造器参数的时候考虑构建器

    构造器多个参数的问题:

    1. 直接传参,参数多的时候难于编写、阅读,容易出错

    例: public NutritionFacts(int servingSize, int servings, int calories, int fat, int sodium, int carbohydrate){ }

    2. 使用javaBeans 模式,就是创建一个java bean 将参数set进去,然后将javabean作为参数,弥补了问题1,但是缺点在于:

    java bean 可能处于不一致的状态,需要确保它的线程安全

    3. 构造器模式

    public class NutritionFacts {

        private final int servingSize;

        private final int servings;

        public static class Builder{

            private final int servingSize;

            private final int servings;

            public Builder(int servingSize){ this.servingSize = servingSize; }

            public Builder setServings(int servings){

                this.servings = servings;

            }

            public NutritionFacts build(){

                return new NutritionFacts(this);

            }

        }

        private NutritionFacts(Builder builder){

            servingSize = builder.servingSize;

            servings = builder.servings;

        }

    }

    调用: NutritionFacts cocaCola = new NutritionFacts.Builder(240).setServings(8).builder();

    3.用私有构造器或者枚举类型强化Singleton属性

    单例模式:

    1.构造器私有化(但是可以通过反射调用,在创建第二个实例时抛出异常)

    public class Elvis {

        public static final Elvis INSTANCE = new Elvis();

        private Elvis(){}

    }

    2.公有的成员是个静态工厂方法

    public class Elvis {   

        private static final Elvis INSTANCE = new Elvis();    

        private Elvis(){}

        public static Elvis getInstance(){

            return INSTANCE;

        }

        private Objcet readResolve(){

            return INSTANCE;

        }

    }

    在序列化时会创建新的实例,需要加入readResolve方法,反序列化时如果有readResolve方法就会调用

    3.枚举(最佳)

    public enum Elvis{

        INSTANCE;

        public void leaveTheBuilding(){}

    }

    4.通过私有构造器强化不可实例化的能力

    有些工具类不希望被实例化,实例对它没有意义 java.lang.Math ,最好加上注释

    public class UtilityClass {

        // 阻止实例化

        private UtilityClass(){

            throw new AssertionError();

        }

    }

    5.避免创建不必要的对象

    例:String s = new String("hello"); // 错误

    String s = "hello";

    当你应该重用现有对象的时候,请不要创建新的对象。当你创建新的对象的时候,请不要重用现有对象

    重用对象付出的代价要远远大于因创建重复对象而付出的代价

    6.消除过期的对象引用

    过期引用:永远也不会被解除的引用

    修复方法:清空引用  引用 = null;

    一般而言,只要类是自己管理内存,就应该警惕内存泄露问题。一旦元素被释放掉,则该元素包含的任何对象引用都应该被清空。

    内存泄露的另一个常见来源是缓存,容易遗忘掉。解决,使用WeakHashMap或者创建一个清除工作的后台线程,java.util.LinkedHashMap.removeEldestEntry删除旧条目

    第三个常见来源是监听器和其他回调,实现了一个API,客户端在这个API中注册,没有显示的取消注册,就会聚集。确保回调立即被当做垃圾的最佳方法是保存他们的弱引用,例如:只讲他们保存成WeakHashMap的键

    7.避免使用终结方法

    终结方法( finalizer ) 通常是不可预测的,也是很危险的,一般情况下是不必要的。使用终结方法会导致行为不稳定、降低性能,以及可移植性问题。

    终结问题缺点在于不能保证会被及时的执行。从一个对象变得不可达开始,到它的终结方法被执行,所花费的时间是任意长的。

    java语言规范不仅不保证终结方法被及时的执行,而且不保证它们会被执行。

    使用终结方法有一个非常严重的性能损失

    解决:提供一个显示的终止方法 ,显示的终止方法通常与try-finally结构结合起来使用。在finally子句内部调用显示的终止方法。

    终结方法2个用途:第一种:当忘记调用显示终结方法时,可以充当安全网。

    第二种:与对象的本地对等体有关,本地对等体是一个本地对象,普通对象通过本地方法委托给一个本地对象。因为本地对象不是一个普通对象,所以垃圾回收器不会知道它,当它的java对等体被回收的时候,它不会被回收。在本地对等体并不用于关键资源的前提下,终止方法正式执行这项任务最合适的工具。

    相关文章

      网友评论

          本文标题:第二章 创建和销毁对象

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