Car
类是不可变的,所有的参数默认值都在一个地方。builder
的 setter
方法返回builder
本身,这样就可以进行链式调用,从而生成一个流畅的 API
。public class Car {
/**
* id
*/
private String id;
/**
* 名称
*/
private String name;
/**
* 速度
*/
private float speed;
private Car(Builder builder) {
id = builder.id;
name = builder.name;
speed = builder.speed;
}
public static class Builder {
/**
* id
*/
private String id;
/**
* 名称
*/
private String name;
/**
* 速度
*/
private float speed;
public Builder() {
}
public Builder id(String id) {
this.id = id;
return this;
}
public Builder name(String name) {
this.name = name;
return this;
}
public Builder speed(float speed) {
this.speed = speed;
return this;
}
public Car build() {
return new Car(this);
}
}
// 省略 getset()
}
Builder
模式非常灵活。 单个 builder
可以重复使用来构建多个对象。Builder
模式是一个不错的选择,特别是许多参数是可选的或相同类型的。builder
模式客户端代码比使用伸缩构造方法(telescopingconstructors
)更容易读写,并且 builder
模式比 JavaBeans
更安全。builder
。虽然创建这个 builder
的成本在实践中不太可能被注意到,但在看中性能的场合下这可能就是一个问题。而且,builder
模式比伸缩构造方法模式更冗长,因此只有在有足够的参数时才值得使用它,比如四个或更多。工具类不是设计用来被实例化的,因为实例化对它没有任何意义。然而,在没有显式构造器的情况下,编译器提供了一个公共的、无参的默认构造器。对于用户来说,该构造器与其他构造器没有什么区别。
可以通过包含一个私有构造器来实现类的非实例化:
public class UtilityClass {
// 该类不能被实例化
private UtilityClass() {
throw new AssertionError();
}
}
每次需要时重用一个对象而不是创建一个新的相同功能对象通常是恰当的。重用可以更快更流行。
反例
// 语句每次执行时都会创建一个新的 String 实例,而这些对象的创建都不是必需的
String s = new String("hello");
// 使用单个 String 实例,而不是每次执行时创建一个新实例。
String s = "hello";
Boolean.valueOf(String)
比构造方法 Boolean(String)
更可取,'split()' could be replaced with compiled 'java.util.regex.Pattern' construct
方法内部为正则表达式创建一个 Pattern
实例,并且只使用它一次,之后它就有资格进行垃圾收集。 创建 Pattern
实例是昂贵的
// 反例
@Test
public void test2(){
String str = "Hello world";
String[] split = str.split("");
}
// 正例
private static final Pattern BLANK_PATTERN = Pattern.compile("");
@Test
public void test2(){
String str = "Hello world";
String[] split = BLANK_PATTERN.split(str);
}
private static long sum() {
// 可以使用 long 基本类型
Long sum = 0L;
for (long i = 0; i <= Integer.MAX_VALUE; i++)
sum += i;
return sum;
}
@Test
public void test1() throws IOException {
String path = "./test.txt";
BufferedReader br = new BufferedReader(new FileReader(path));
try {
String s = br.readLine();
logger.info(s);
} finally {
// 有可能会出错
br.close();
}
}
try-with-resources
@Test
public void test1() throws IOException {
String path = "./test.txt";
try (BufferedReader br = new BufferedReader(new FileReader(path))) {
String s = br.readLine();
logger.info(s);
}
}
要使用这个构造,资源必须实现 AutoCloseable
接口,该接口由一个返回为 void
的 close
组成。Java
类库和第三方类库中的许多类和接口现在都实现或继承了 AutoCloseable
接口。如果你编写的类表示
必须关闭的资源,那么这个类也应该实现 AutoCloseable
接口。
public abstract class Reader implements Readable, Closeable{}
// BufferedReader 实现 close 接口
public void close() throws IOException {
synchronized (lock) {
if (in == null)
return;
try {
in.close();
} finally {
in = null;
cb = null;
}
}
}
可以在 try-with-resources
语句中添加 catch
子句,就像在常规的 try-finally
语句中一样。这允许你
处理异常,而不会在另一层嵌套中污染代码。
使用 try-with-resources
语句替代 try-finally
语句。 生成的代码更简洁,更清晰,并且生成的异常更有用。 try-with-resources
语句在编写必须关闭资源的代码时会更容易,也不会出错,
equals
方法时,必须遵守它的通用约定。Object
的规范如下:equals
方法实现了一个等价关系,它有以下这些属性:
自反性: 对于任何非空引用 x
, x.equals(x)
必须返回 true
。
对称性: 对于任何非空引用 x
和 y
,如果且仅当 y.equals(x)
返回 true
时 x.equals(y)
必须
返回 true
。
传递性: 对于任何非空引用 x
、y
、z
,如果 x.equals(y)
返回 true
, y.equals(z)
返回 true
,
则 x.equals(z)
必须返回 true
。
一致性: 对于任何非空引用 x
和 y
,如果在 equals
比较中使用的信息没有修改,则
x.equals(y)
的多次调用必须始终返回 true
或始终返回 false
。
对于任何非空引用 x
, x.equals(null)
必须返回 false
。
contains
方法,如果违反了自反性会找不到这个对象boolean contains(Object o);
public int indexOf(Object o) {
if (o == null) {
for (int i = 0; i < size; i++)
if (elementData[i]==null)
return i;
} else {
for (int i = 0; i < size; i++)
if (o.equals(elementData[i]))
return i;
}
return -1;
}
public boolean equals(Object obj) {
return (this == obj);
}
import java.util.Objects;
public final class CaseInsensitiveString {
private final String s;
public CaseInsensitiveString(String s) {
this.s = Objects.requireNonNull(s);
}
@Override
public boolean equals(Object o) {
if (o instanceof CaseInsensitiveString)
return s.equalsIgnoreCase(((CaseInsensitiveString) o).s);
// 操作 String 字符串
if (o instanceof String)
return s.equalsIgnoreCase((String) o);
return false;
}
}
CaseInsensitiveString cis = new CaseInsensitiveString("Polish");
String s = "polish";
System.out.println(cis.equals(s)); // true
System.out.println(s.equals(cis)); // false
CaseInsensitiveString
类中的 equals
方法知道正常字符串,但 String
类中的 equals
方法却忽略了不区分大小写的字符串。 因此,s.equals(cis)
返回 false
,明显违反对称性。// Object equals 比较 两个对象是否具有相同的引用
public boolean equals(Object obj) {
return (this == obj);
}
public native int hashCode();
hashCode
和 equals
两个方法是用来协同判断两个对象是否相等的,采用这种方式的原因是可以提高程序插入和查询的速度,如果在重写 equals
时,不重写 hashCode
,就会导致在某些场景下,例如将两个相等的自定义对象存储在 Set
集合时,就会出现程序执行的异常,为了保证程序的正常执行,所以我们就需要在重写 equals
时,也一并重写 hashCode
方法才行。toString
通用约定「建议所有的子类重写这个方法」。public String toString() {
// 类名后跟一个「at」符号(@)和哈希码的无符号十六进制表示组成
return getClass().getName() + "@" + Integer.toHexString(hashCode());
}
toString
方法应该以一种美观的格式返回对象的简明有用的描述。final
。public static final String[] STR_ARR = {"1", "2", "3", "4"};
// 方法 1
private static final String[] STR_ARR = { ... };
public static final List<String> VALUES = Collections.unmodifiableList(Arrays.asList(STR_ARR));
// 方法 2
private static final String[] STR_ARR = {"1", "2", "3", "4"};
public static final String[] strArrayValues() {
return STR_ARR.clone();
}
尽可能地消除每一个未经检查的警告
每当使用 @SuppressWarnings(“unchecked”) 注解时,请添加注释,说明为什么是安全的。
catch
块会使异常达不到应有的目的catch
块中应该包含一条注释,说明为什么可以这么做,并且变量应该命名为 ignored
:@Test
public void test4() {
try {
String a = null;
method1();
logger.info(a);
} catch (RuntimeException ignored) {
// 捕获异常不做任何处理
}
logger.info("ok");
}
Override
注解,并且认为要重写父类声明,那么编译器可以保护免受很多错误的影响,但有一个例外。 在具体的类中,不需要注解标记你确信可以重写抽象方法声明的方法。