*:first-child {
margin-top: 0 !important; }
body > *:last-child {
margin-bottom: 0 !important; }
a {
color: #4183C4; }
a.absent {
color: #cc0000; }
a.anchor {
display: block;
padding-left: 30px;
margin-left: -30px;
cursor: pointer;
position: absolute;
top: 0;
left: 0;
bottom: 0; }
h1, h2, h3, h4, h5, h6 {
margin: 20px 0 10px;
padding: 0;
font-weight: bold;
-webkit-font-smoothing: antialiased;
cursor: text;
position: relative; }
h1:hover a.anchor, h2:hover a.anchor, h3:hover a.anchor, h4:hover a.anchor, h5:hover a.anchor, h6:hover a.anchor {
background: url() no-repeat 10px center;
text-decoration: none; }
h1 tt, h1 code {
font-size: inherit; }
h2 tt, h2 code {
font-size: inherit; }
h3 tt, h3 code {
font-size: inherit; }
h4 tt, h4 code {
font-size: inherit; }
h5 tt, h5 code {
font-size: inherit; }
h6 tt, h6 code {
font-size: inherit; }
h1 {
font-size: 28px;
color: black; }
h2 {
font-size: 24px;
border-bottom: 1px solid #cccccc;
color: black; }
h3 {
font-size: 18px; }
h4 {
font-size: 16px; }
h5 {
font-size: 14px; }
h6 {
color: #777777;
font-size: 14px; }
p, blockquote, ul, ol, dl, li, table, pre {
margin: 15px 0; }
hr {
background: transparent url() repeat-x 0 0;
border: 0 none;
color: #cccccc;
height: 4px;
padding: 0;
}
body > h2:first-child {
margin-top: 0;
padding-top: 0; }
body > h1:first-child {
margin-top: 0;
padding-top: 0; }
body > h1:first-child + h2 {
margin-top: 0;
padding-top: 0; }
body > h3:first-child, body > h4:first-child, body > h5:first-child, body > h6:first-child {
margin-top: 0;
padding-top: 0; }
a:first-child h1, a:first-child h2, a:first-child h3, a:first-child h4, a:first-child h5, a:first-child h6 {
margin-top: 0;
padding-top: 0; }
h1 p, h2 p, h3 p, h4 p, h5 p, h6 p {
margin-top: 0; }
li p.first {
display: inline-block; }
li {
margin: 0; }
ul, ol {
padding-left: 30px; }
ul :first-child, ol :first-child {
margin-top: 0; }
dl {
padding: 0; }
dl dt {
font-size: 14px;
font-weight: bold;
font-style: italic;
padding: 0;
margin: 15px 0 5px; }
dl dt:first-child {
padding: 0; }
dl dt > :first-child {
margin-top: 0; }
dl dt > :last-child {
margin-bottom: 0; }
dl dd {
margin: 0 0 15px;
padding: 0 15px; }
dl dd > :first-child {
margin-top: 0; }
dl dd > :last-child {
margin-bottom: 0; }
blockquote {
border-left: 4px solid #dddddd;
padding: 0 15px;
color: #777777; }
blockquote > :first-child {
margin-top: 0; }
blockquote > :last-child {
margin-bottom: 0; }
table {
padding: 0;border-collapse: collapse; }
table tr {
border-top: 1px solid #cccccc;
background-color: white;
margin: 0;
padding: 0; }
table tr:nth-child(2n) {
background-color: #f8f8f8; }
table tr th {
font-weight: bold;
border: 1px solid #cccccc;
margin: 0;
padding: 6px 13px; }
table tr td {
border: 1px solid #cccccc;
margin: 0;
padding: 6px 13px; }
table tr th :first-child, table tr td :first-child {
margin-top: 0; }
table tr th :last-child, table tr td :last-child {
margin-bottom: 0; }
img {
max-width: 100%; }
span.frame {
display: block;
overflow: hidden; }
span.frame > span {
border: 1px solid #dddddd;
display: block;
float: left;
overflow: hidden;
margin: 13px 0 0;
padding: 7px;
width: auto; }
span.frame span img {
display: block;
float: left; }
span.frame span span {
clear: both;
color: #333333;
display: block;
padding: 5px 0 0; }
span.align-center {
display: block;
overflow: hidden;
clear: both; }
span.align-center > span {
display: block;
overflow: hidden;
margin: 13px auto 0;
text-align: center; }
span.align-center span img {
margin: 0 auto;
text-align: center; }
span.align-right {
display: block;
overflow: hidden;
clear: both; }
span.align-right > span {
display: block;
overflow: hidden;
margin: 13px 0 0;
text-align: right; }
span.align-right span img {
margin: 0;
text-align: right; }
span.float-left {
display: block;
margin-right: 13px;
overflow: hidden;
float: left; }
span.float-left span {
margin: 13px 0 0; }
span.float-right {
display: block;
margin-left: 13px;
overflow: hidden;
float: right; }
span.float-right > span {
display: block;
overflow: hidden;
margin: 13px auto 0;
text-align: right; }
code, tt {
margin: 0 2px;
padding: 0 5px;
white-space: nowrap;
border: 1px solid #eaeaea;
background-color: #f8f8f8;
border-radius: 3px; }
pre code {
margin: 0;
padding: 0;
white-space: pre;
border: none;
background: transparent; }
.highlight pre {
background-color: #f8f8f8;
border: 1px solid #cccccc;
font-size: 13px;
line-height: 19px;
overflow: auto;
padding: 6px 10px;
border-radius: 3px; }
pre {
background-color: #f8f8f8;
border: 1px solid #cccccc;
font-size: 13px;
line-height: 19px;
overflow: auto;
padding: 6px 10px;
border-radius: 3px; }
pre code, pre tt {
background-color: transparent;
border: none; }
sup {
font-size: 0.83em;
vertical-align: super;
line-height: 0;
}
* {
-webkit-print-color-adjust: exact;
}
@media screen and (min-width: 914px) {
body {
width: 854px;
margin:0 auto;
}
}
@media print {
table, pre {
page-break-inside: avoid;
}
pre {
word-wrap: break-word;
}
}
-->
方法
第38条:检查参数的有效性
- 公有方法:要用Javadoc的
@throws
标签在文档中说明违反参数值限制时抛出的异常。
/**
* Returns a BigInteger whose value is {@code (this mod m}). This method
* differs from {@code remainder} in that it always returns a
* <i>non-negative</i> BigInteger.
*
* @param m the modulus.
* @return {@code this mod m}
* @throws ArithmeticException {@code m} ≤ 0
* @see #remainder
*/
public BigInteger mod(BigInteger m) {
if (m.signum <= 0)
throw new ArithmeticException("BigInteger: modulus not positive");
BigInteger result = this.remainder(m);
return (result.signum >= 0 ? result : result.add(m));
}
- 非公有方法:使用断言检查参数
private static void sort(long array[],int offset,int length){
//assert 断言
assert array!=null;
assert offset>=0&&offset<=array.length;
assert length>=0&&length<= array.length-offset;
}
sort(null,0,0);
//Exception in thread "main" java.lang.AssertionError
断言默认是关闭的
- Java编译中启用断言:
-enableassertions
,简写为-ea
- IDEA启用断言:Run-->Edit Configuration-->VM Options-->添加
-ea
第39条:必要时进行保护性拷贝
保护性地设计程序(假设类的客户端会尽其所能破坏类的约束条件)
//下面的类声称可以表示一段不可变的时间周期
public final class Period {
private final Date start;
private final Date end;
public Period(Date start,Date end){
if(start.compareTo(end)>0)
throw new IllegalArgumentException(start+"after"+end);
this.start = start;
this.end = end;
}
public Date getStart(){
return start;
}
public Date getEnd() {
return end;
}
}
上面那个Period类表面上看是不可变类,并且加了约束条件:周期的起始时间不能再结束时间之后。但由于Date类是可变的,很容易就违反这个约束条件。如下。
public static void main(String[] args) {
Calendar calendar = Calendar.getInstance();
calendar.set(2008,Calendar.JULY,8);
Date start = calendar.getTime();
calendar.set(2016,Calendar.JULY,3);
Date end = calendar.getTime();
Period period = new Period(start,end);
start.setYear(2017);//修改了Period类
Date start1 = period.getStart();
System.out.println(start1.getYear());//2017
}
为避免类的实例的内部信息受到攻击,可对构造器中的每个可变参数进行保护性拷贝。
public Period(Date start,Date end){
// if(start.compareTo(end)>0)
// throw new IllegalArgumentException(start+"after"+end);
// this.start = start;
// this.end = end;
this.start = new Date(start.getTime());
this.end = new Date(end.getTime());
if(this.start.compareTo(this.end)>0)
throw new IllegalArgumentException(start+"after"+end);
}
虽然替换构造方法能避免上述的攻击,但仍然可以改变Period实例。它的访问方法提供了对其可变内部成员的访问能力。
period.getEnd().setYear(1998);//修改了Period实例
对于后一种攻击,只需修改方法,使它返回可变内部域的保护性拷贝即可:
public Date getStart(){
return new Date(start.getTime());
}
public Date getEnd() {
return new Date(end.getTime());
}
参数的保护性拷贝:
- 不可变类
- 客户提供的对象进入内部数据结构中(编写方法或构造器时)
下面的例子就是使用可变对象作为Map的键,导致Map的约束条件被破坏。
Calendar calendar = Calendar.getInstance();
calendar.set(2008,Calendar.JULY,8);
Date start = calendar.getTime();
Map<Date,String> map = new HashMap<>();
map.put(start,start.getYear()+"");
start.setYear(1999);
String s = map.get(start);
System.out.println(s);
//修改后,使用Date.getTime()返回的long值作为Map内部时间的表示方法
Map<Long,String> newMap = new HashMap<>();
newMap.put(start.getTime(),start.getYear()+"");
结论:
- 最好使用不可变对象作为对象内部的组件。
- 类具有从客户端得到或返回客户端的可变组件,类就必须保护性地拷贝这些组件。
- 拷贝的成本受到限制,且类信任它的客户端不会不恰当地修改组件,就可在文档中指明客户端的职责是不得修改受到影响的组件。
第40条:谨慎设计方法签名
- 谨慎选择方法的名称
可参看Google Java Style中的方法命名标准
- 不要过于追求提供便利的方法
- 避免过长的参数列表
目标是4个参数,或者更少。 相同类型的长参数序列格外有害。 容易弄错顺序,但程序仍可以编译和运行。
缩短过长的参数列表的三种办法:
- 1.分解成多个方法
- 2.创建辅助类
- 3.Builder模式
对于参数类型,优先使用接口而不是类。
methodA(Map<K,V> map);
//methodA(HashMap<K,V> map);
对于boolean参数,优先使用两个元素的枚举类型。
public enum TemperatureScale{
F,C
}
Thermometer.newInstance(TemperatureScale.C);
慎用重载
public class CollectionClassifier {
public static String classify(Set<?> set){
return "Set";
}
public static String classify(List<?> list){
return "List";
}
public static String classify(Collection<?> collection){
return "Unknown Collection";
}
public static void main(String[] args) {
Collection<?> [] collections = {
new HashSet<String>(),
new ArrayList<BigInteger>(),
new HashMap<String,Object>().values()
};
for(Collection collection:collections){
System.out.println(classify(collection));
}
}
}
//
//Unknown Collection
//Unknown Collection
//Unknown Collection
classify方法被重载(overloaded)了,而调用哪个重载方法是在编译时做出决定的。for循环中的三次循环,参数的编译时类型都是相同的:Collection<?>
。即使每次循环的运行时类型都是不同的。
重载方法(overloaded method)的选择是静态的,被覆盖的方法(overridden method)的选择是动态的。调用哪个被覆盖的方法是在运行时做出决定的。
public class Overriding {
public static void main(String[] args) {
Wine [] wines = {
new Wine(),
new SparklingWine(),
new Champagne()
};
for(Wine wine:wines){
System.out.println(wine.name());
}
}
}
class Wine{
String name(){
return "wine";
}
}
class SparklingWine extends Wine{
@Override
String name() {
return "sparkling wine";
}
}
class Champagne extends SparklingWine{
@Override
String name() {
return "champagne";
}
}
//output:
//wine
//sparkling wine
//champagne