Effective-Java-Chapter2

news2025/1/20 3:56:57

https://github.com/clxering/Effective-Java-3rd-edition-Chinese-English-bilingual/blob/dev/
在这里插入图片描述

准则一 考虑以静态工厂方法代替构造函数

优点
  • 静态工厂方法与构造函数相比的第一个优点,静态工厂方法有确切名称。
    知名见意,静态方法我们可以通过命名这个方法,知道这个方法返回什么对象。但是构造函数往往是通过参数来控制对象,需要我们对提供的构造函数有定的了解。
  • 静态工厂方法与构造函数相比的第二个优点,静态工厂方法不需要在每次调用时创建新对象。
    静态方法是共享的,不需要创建对象就能调用。
  • 静态工厂方法与构造函数相比的第三个优点,可以通过静态工厂方法获取返回类型的任何子类的对象。
  • 静态工厂的第四个优点是,返回对象的类可以随调用的不同而变化,作为输入参数的函数。
  • 静态工厂的第五个优点是,当编写包含方法的类时,返回对象的类不需要存在。
缺点
  • 仅提供静态工厂方法的主要局限是,没有公共或受保护构造函数的类不能被子类化。
  • 静态工厂方法的第二个缺点是程序员很难找到它们。
最佳实践

Boolean的valueOf

public static Boolean valueOf(boolean b) {
      return (b ? TRUE : FALSE);
  }

顺便提一嘴,布尔类型的变量是两个不可变的,实际上改变一个布尔类型值就是改变了引用:
在这里插入图片描述

Boolean sington = Boolean.valueOf(true);
sington = false;

通过字节码就看的比较清楚,sington = false;还是调用valueOf 这个静态方法
在这里插入图片描述

集合中的

public static <T> List<T> unmodifiableList(List<? extends T> list)
public static <T> Set<T> unmodifiableSet(Set<? extends T> s)
public static <K,V> Map<K,V> unmodifiableMap(Map<? extends K, ? extends V> m)

准则二 当构造函数有多个参数时,考虑改用构建器

优点

可伸缩构造函数模式可以工作,但是当有很多参数时,编写客户端代码是很困难的,而且读起来更困难。那么使用构造者模式可以在最后build的时候进行检验,也不需要写一堆Get,Set。但是通常来说构造对象发逻辑比较复杂时候也经常实用构造者模式。

缺点

为了创建一个对象,你必须首先创建它的构建器。虽然在实际应用中创建这个构建器的成本可能并不显著,但在以性能为关键的场景下,这可能会是一个问题。而且,建造者模式比可伸缩构造函数模式更冗长,因此只有在有足够多的参数时才值得使用,比如有 4 个或更多参数时,才应该使用它。

最佳实践

com.google.common.collect.ImmutableMap#Builder<K, V>

static final ImmutableMap<String, Integer> WORD_TO_INT = new ImmutableMap.Builder<String, Integer>()          .put("one", 1)          
.put("two", 2)          
.put("three", 3)          
.build();

准则三 使用私有构造函数或枚举类型实施单例属性

最佳实践

例如Java中的Runtime,可以看到构造函数是私有的,提供了一个静态方法返回单例的Runtime 对象。

public class Runtime {
    private static Runtime currentRuntime = new Runtime();

    /**
     * Returns the runtime object associated with the current Java application.
     * Most of the methods of class <code>Runtime</code> are instance
     * methods and must be invoked with respect to the current runtime object.
     *
     * @return  the <code>Runtime</code> object associated with the current
     *          Java application.
     */
    public static Runtime getRuntime() {
        return currentRuntime;
    }
    /** Don't let anyone else instantiate this class */
    private Runtime() {}
 }

准则四 用私有构造函数实施不可实例化

这个准则主要是由于,有的时候我们这类的作用仅仅是将一些通用的方法聚集在一起,比如我们这个方法是工具类。我们对这个类进行实例化是毫无意义的。

方式一
class NonInstantiionClass {
    // 构造函数私有化
    private NonInstantiionClass() {
    }
}
方式二
class NonInstantiionClass {
   public NonInstantiionClass() {
       throw new UnsupportedOperationException("不支持实例化");
   }
}
最佳实践

在Java总Arrays是一个工具类,里面大多数方法都是静态的,可以看到它的构造函数就被私有化了。
在这里插入图片描述

准则五 依赖注入优于硬连接资源

这个准则是什么意思呢?
首先来看一个反例,也是书上的例子:

public class SpellChecker {
    private static final Lexicon dictionary = ...;
    private SpellChecker() {} // Noninstantiable
    public static boolean isValid(String word) { ... }
    public static List<String> suggestions(String typo) { ... }
}

比如说这个类,我们要检查单词拼写的正确性,我们需要一个参考标准,而这个标准就是Lexicon这个字典,这样写代码好像貌似完全可行。但是有个缺陷,那如果我们需要引入新的字典怎么办,或者这个字典并不是非常的全面我们需要替换字典怎么办呢?

public class SpellChecker {
    /**
     * 注入字典的抽象 通过注入的方式是不是灵活性更强了呢?
     * 如果我们要替换字典只需通过set注入改变引用即可
     */
    private Dictionary dictionary;
    public SpellChecker() {}
    public static boolean isValid(String word) {
        return Boolean.FALSE;
    }
    public static List<String> suggestions(String typo) {
        return Collections.emptyList();
    }
    public void setDictionary(Dictionary dictionary) {
        this.dictionary = dictionary;
    }
}

我个人的理解就是这样,这也是为什么Java编程中有个准则叫依赖倒置原则,我们常用的Spring IOC容器也是这个思想。

准则六 避免创建不必要的对象

下面这两个语句到底有什么区别?

String s = new String("bikini"); 
String s = "bikini";

在这里插入图片描述
在这里插入图片描述
通过对比字节码我们可以看到第二条语句没有new 而是直接使用了LDC(Push item from run-time constant pool),根据JVM规范是把这个对方放到常量池中了,所以说第一条语句其实创建了多余的对象。其实这里也是经常说的字符串常量池,在直接使用双引号申明字符串的时候会放到字符串常量池中。
书中还有一个例子,关于原生类型和包装类型,看到这个例子之前我经常用包装类型,以后得改变这个习惯:

private static long sum1() {
     Long sum = 0L;
     for (long i = 0; i <= Integer.MAX_VALUE; i++)
         sum += i;
     return sum;
 }
 private static long sum2() {
     long sum = 0L;
     for (long i = 0; i <= Integer.MAX_VALUE; i++)
         sum += i;
     return sum;
 }

 public static void main(String[] args) {
     System.out.println("===========================");
     long cur = System.currentTimeMillis();
     sum1();
     System.out.println("sum1 cost time : " + (System.currentTimeMillis() - cur));
     System.out.println("-----------------------------");
     cur = System.currentTimeMillis();
     sum2();
     System.out.println("sum2 cost time : " + (System.currentTimeMillis() - cur));
 }

差距既然如此之大,原因就在于使用包装类创建了很多不必要的对象。
在这里插入图片描述

另外书中给我们举了这个例子:

static boolean isRomanNumeral(String s) {
    return s.matches("^(?=.)M*(C[MD]|D?C{0,3})" + "(X[CL]|L?X{0,3})(I[XV]|V?I{0,3})$");
}

这个例子有什么问题,问题就在于每一次调用都会去实例化"^(?=.)M*(C[MD]|D?C{0,3})" + “(X[CL]|L?X{0,3})(I[XV]|V?I{0,3})$” 这个正则的Pattern对象,但是这个Pattern其实完全可以复用。如果说我们这个函数调用的比较频繁,就会创建大量的对象,可能会增加monitor GC的频率,影响性能。怎么优化呢?

// 这样子是不是就是复用了这个对象呢? 这里还有一个思考,就是这种写法其实是饿汉式的写法,一上来就创建,我们是不是可以考虑延迟初始化呢?这个就需要结合具体的场景了
private final static Pattern MATCH_PATTERN = Pattern.compile("^(?=.)M*(C[MD]|D?C{0,3})\" + \"(X[CL]|L?X{0,3})(I[XV]|V?I{0,3})$");
最佳实践

这个准则的最佳实践其实就是我们常用的数据库连接池、线程池,用设计模式的角度来看待的话就是享元模式。这个准则更适合应用在我们使用的对象可以复用并且创建它的成本是比较大的,如果成本可以忽略不计也没有必要。

准则七 排除过时的对象引用避免内存泄漏

常见的场景
  • 一个类管理它自己的内存时,程序员应该警惕内存泄漏。当释放一个元素时,该元素中包含的任何对象引用都应该被置为 null。
import java.util.Arrays;
import java.util.EmptyStackException;

// Can you spot the "memory leak"?
public class Stack {
    private Object[] elements;
    private int size = 0;
    private static final int DEFAULT_INITIAL_CAPACITY = 16;

    public Stack() {
        elements = new Object[DEFAULT_INITIAL_CAPACITY];
    }

    public void push(Object e) {
        ensureCapacity();
        elements[size++] = e;
    }
    // 这里就存在内存泄漏,因为这个数据是这个类进行管理的,我们并没有释放对象(只是改变了数组的大小),它依然被数组引用
    public Object pop() {
        if (size == 0)
            throw new EmptyStackException();
        return elements[--size];
    }

    /**
     * Ensure space for at least one more element, roughly
     * doubling the capacity each time the array needs to grow.
     */
    private void ensureCapacity() {
        if (elements.length == size)
            elements = Arrays.copyOf(elements, 2 * size + 1);
    }
}
  • 缓存,要有淘汰策略,或者设置软引用,或者后台线程定期清理。
  • 内存泄漏的第三个常见来源是侦听器和其他回调。 如果你实现了一个 API,其中客户端注册回调,但不显式取消它们,除非你采取一些行动,否则它们将累积。确保回调被及时地垃圾收集的一种方法是仅存储对它们的弱引用,例如,将它们作为键存储在 WeakHashMap 中。

准则八 避免使用终结器和清除器

准则九 使用 try-with-resources 优于 try-finally

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1978824.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

编程新手到大师:大学生学编程的高效学习路径

​​​​​​​ 目录 ​​​​​​​​​​​​​​ 编程新手到大师&#xff1a;大学生学编程的高效学习路径 编程小白如何成为大神&#xff1f;大学新生的最佳入门攻略 一、确定学习目标 1.1、我接受想贯彻互联网思维的洗礼 1.2、我想提升在专业中的竞争力 1.3、我觉得…

(STM32笔记)九、RCC时钟树与时钟 第三部分

我用的是正点的STM32F103来进行学习&#xff0c;板子和教程是野火的指南者。 之后的这个系列笔记开头未标明的话&#xff0c;用的也是这个板子和教程。 九、RCC时钟树与时钟 九、RCC时钟树与时钟3、使用固件库的函数来配置时钟bsp_rccclkconfig.c思路配置HSE时钟把RCC寄存器复位…

OpenCV Python 图像相加与透明色转换

将两幅图添加起来&#xff0c;构成一幅新的图像&#xff0c;并尝试将一个PNG的透明背景转换为特定的颜色。 生成纯背景 以下代码生成一个纯色背景JPG&#xff0c;颜色为照片底板的蓝&#xff08;R:60 G:140 B:220&#xff09;。在用OpenCV创建图像时&#xff0c;颜色按BGR。 …

必须知道的国内linux镜像下载网址,建议收藏

linux镜像下载 一、阿里云开源镜像站下载可用的镜像 二、网易开源镜像下载可用的镜像 三、搜狐开源镜像下载可用的镜像 一、阿里云开源镜像站下载 https://developer.aliyun.com/mirror/ 阿里云开源镜像站是阿里云提供的一个公共服务&#xff0c;旨在为开发者提供快速、稳定…

4. 最长公共前缀

4. 最长公共前缀 题目题目分析 题目 题目分析 首先要对字符串数组进行分析&#xff0c;字符串数组元素的最长公共前缀肯定不会超过最小元素长度&#xff0c;并如存在公共前缀则需遍历整个字符串元素&#xff0c;有点像二维数组&#xff0c;最后加上截取字符串加上判空操作就完…

apache一台服务器如何通过不同端口映射不同网页服务入口

一、背景 由于服务器资源受限&#xff08;IP资源或硬件资源&#xff09;&#xff0c;有时候希望一台服务器IP能有部署多个网页服务入口。 传统都是80端口映射为默认服务入口&#xff0c;当需要部署多个网页服务入口时&#xff0c;可以启用其他端口&#xff0c;不同端口映射到…

Javascript常见算法详解

在JavaScript&#xff08;JS&#xff09;中&#xff0c;常见的算法涵盖了多个领域&#xff0c;从基础的数组操作到更复杂的排序、搜索和数据结构算法。下面是一些在JS中常见的算法示例&#xff1a; 1. 排序算法 Java排序算法-CSDN博客 冒泡排序&#xff08;Bubble Sort&#x…

【C++从小白到大牛】多态那些事儿(上)

一、多态的概念 1.1概念: 通俗来说&#xff0c;就是多种形态&#xff0c;具体点就是去完成某个行为&#xff0c;当不同的对象去完成时会产生出不同的状态。 二、 多态的定义及实现 2.1多态的构成条件 多态是在不同继承关系的类对象&#xff0c;去调用同一函数&#xff0c;产…

flutter开发windows应用程序(.exe)配置环境搭建 以及 visual studio配置flutter windows所需环境

flutter 开发 windows 应用程序&#xff08;.exe&#xff09;配置环境搭建 安装 android studio 开发工具 并配置相关所需环境 参考该文章&#xff1a;我还没写完你上百度上搜一下吧我帮你付钱了&#x1f602; 控制台运行 flutter doctor 命令 查看当前 flutter 开发环境 报…

傅里叶级数的数学描述

目录 概述 1 傅里叶级数 1.1 概念 1.2 表示形式 2 傅里叶变换 2.1 概念 2.2 数学描述 2.3 应用 3 傅里叶级数的数学推论 3.1 三角函数的正交性 3.1.1 正交性介绍 3.1.2 正交性证明 3.1.3 相同函数乘积积分 3.2 理论介绍 3.3 傅里叶级数的表述 概述 傅里叶级数是…

sais复杂推理能力评估笔记(一):baseline简介

赛题流程 初赛数据集为逻辑推理数据&#xff0c;其中训练集中包含500条训练数据&#xff0c;测试集中包含500条测试数据。每个问题包括若干子问题&#xff0c;每个子问题为单项选择题&#xff0c;选项不定&#xff08;最多5个&#xff09;。目标是为每个子问题选择一个正确答案…

前端需要了解的数据库基础与身份认证

目录 数据库与身份认证数据库的基本概念常见数据库和分类传统型数据库的数据组织结构安装并配置MySQL MySQL的基本使用使用SQL管理数据库SQL中的SELECT语句SQL中的INSERT INTO语句SQL中的UPDATE语句SQL中的DELETE语句SQL中的WHERE子句SQL中的AND和OR运算符SQL中的ORDER BY子句S…

Pip 使用报错及解决

pip install 是Python 包管理器命令&#xff0c;常用参数&#xff1a; -r&#xff1a;从一个需求文件中安装所有的包。-U 或 --upgrade&#xff1a;升级一个已经安装的包到最新版本。-I 或 --ignore-installed&#xff1a;即使包已经安装&#xff0c;也重新安装。--no-cache-d…

奥运会Ⅴ--具有混合模型的 Transformer 架构

Transformer 架构的变革性影响和混合模型的未来&#xff0c;将 Transformer 与其他架构相结合&#xff0c;突破 AI 能力的界限。 Transformer 架构从根本上重塑了自然语言处理 (NLP) 和机器学习的格局。这种序列建模和传导任务的创新方法不仅超越了之前最先进的模型&#xff0c…

每日OJ_牛客WY15 幸运的袋子

目录 牛客HJ62 查找输入整数二进制中1的个数 解析代码 牛客HJ62 查找输入整数二进制中1的个数 查找输入整数二进制中1的个数_牛客题霸_牛客网 解析代码 本题是计算一个数二进制表示中1的个数&#xff0c;通过&#xff08;n >> i) & 1可以获取第i位的二进制值&…

阿里云文件上传之客户端上传

阿里云文件上传之前一直是使用服务端上传,但一直存在上传不稳定问题,三兆以上的文件上传经常出现上传超时问题.究其原因客户端将文件上传到业务服务器&#xff0c;然后业务服务器将文件上传到OSS。在这个过程中&#xff0c;一份数据需要在网络上传输两次&#xff0c;会造成网络…

Vuforia AR篇(九)— AR塔防下篇

目录 前言一、搭建UI二、创建脚本 前言 在增强现实&#xff08;AR&#xff09;技术快速发展的今天&#xff0c;Vuforia作为一个强大的AR开发平台&#xff0c;为开发者提供了许多便捷的工具和功能。在本篇博客中&#xff0c;我们将介绍如何使用Vuforia在Unity中创建一个简单的塔…

46 class添加与颜色分配47 区域规则介绍与添加48 走线修线介绍49 复制、改变、删除操作

46 class添加与颜色分配&&47 区域规则介绍与添加&&48 走线修线介绍&&49 复制、改变、删除操作 第一部分 46 class添加与颜色分配创建网络类CLS创建网络组net-group NG颜色分配**填充类型设置****高亮关闭****修改颜色面板的颜色**从其它已有PCB设计中导…

用Manim实现——计算和绘制图形下方区域

用Manim实现——计算和绘制图形下方区域 get_area 函数 get_area是一个用于计算和绘制图形下方区域的函数&#xff0c;常用于图形动画库&#xff08;如 Manim&#xff09; get_area(graph, x_rangeNone, color(ManimColor(#58C4DD),ManimColor(#83C167)), opacity0.3, bounde…

市场主流 AI 视频生成技术的迭代路径

AI视频生成技术的迭代路径经历了从GANVAE、Transformer、Diffusion Model到Sora采用的DiT架构&#xff08;TransformerDiffusion&#xff09;等多个阶段&#xff0c;每个阶段的技术升级都在视频处理质量上带来了飞跃性的提升。这些技术进步不仅推动了AI视频生成领域的快速发展&…