对应《疯狂Java讲义(第5版)》7.3-7.4 章节
Object类
Object是所有类、数组、枚举类的父类,没有使用extends显式指定父类的类默认继承Object类
boolean equals(Object obj) 是否为同一对象
protected void finalize() 垃圾回收器清理该对象资源
Class<?> getClass() 返回运行时类
int hashCode() 默认根据地址计算,与System.identityHashCode(Object x)相同,但很多类改写了该方法
String toString() 对象的字符串表示,默认返回运行时类名@16进制hashCode
wait()、notify()、notifyAll() 控制线程的运行和暂停
protected Clone() 实现对象的“自我克隆”,得到一个完全隔离的副本
- “浅克隆”,只克隆该对象的所有成员变量值,不会对引用类型的成员变量引用的对象进行克隆
- “深克隆”需要开发者自己进行“递归克隆”
- 实现 Cloneable 接口(只是一个标记,接口里没有任何方法)
- 实现自定义的 clone() 方法
- 通过 super.clone() 调用父类的克隆方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
| class Address { String detail; public Address(String detail) { this.detail = detail; } }
class User implements Cloneable { int age; Address address; public User(int age) { this.age = age; address = new Address("广州天河"); } public User clone() throws CloneNotSupportedException { return (User)super.clone(); } } public class CloneTest { public static void main(String[] args) throws CloneNotSupportedException { User u1 = new User(29); User u2 = u1.clone(); System.out.println(u1 == u2); System.out.println(u1.address == u2.address); } }
|
操作对象的Objects工具类(Java 7)
提供的工具方法大都“空指针”安全,不会引发空指针异常,如toSting()将输出null
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| public class ObjectsTest { static ObjectsTest obj; public static void main(String[] args) { System.out.println(Objects.hashCode(obj)); System.out.println(Objects.toString(obj)); System.out.println(Objects.requireNonNull(obj , "obj参数不能是null!")); } }
|
1 2 3 4
| public Foo(Bar bar){ this.bar = Objects.requireNonNull(bar); }
|
String、StringBuffer、StringBuilder类(Java 9 改进)
String是不可变类,包含的字符序列不可修改
StringBuffer对象代表一个字符序列可变的字符串
StringBuilder和StringBuffer基本相同,但StringBuffer线程安全,StringBuilder没有实现线程安全,所以性能略高
都实现了CharSequence接口,字符串协议接口
Java 9 以前采用char[]数组保存字符, 每个字符占用2字节;Java 9 开始采用byte[]数组再加一个encoding-flag字段来保存字符,每个字符只占1字节
String
下面是 String 类支持的方法,更多详细,参看 Java String API 文档:
1 | char charAt(int index) 返回指定索引处的 char 值。 |
---|
2 | int compareTo(Object o) 把这个字符串和另一个对象比较。 |
3 | int compareTo(String anotherString) 按字典顺序比较两个字符串。 |
4 | int compareToIgnoreCase(String str) 按字典顺序比较两个字符串,不考虑大小写。 |
5 | String concat(String str) 将指定字符串连接到此字符串的结尾。 |
6 | boolean contentEquals(StringBuffer sb) 当且仅当字符串与指定的StringBuffer有相同顺序的字符时候返回真。 |
7 | static String copyValueOf(char[] data) 返回指定数组中表示该字符序列的 String。 |
8 | static String copyValueOf(char[] data, int offset, int count) 返回指定数组中表示该字符序列的 String。 |
9 | boolean endsWith(String suffix) 测试此字符串是否以指定的后缀结束。 |
10 | boolean equals(Object anObject) 将此字符串与指定的对象比较。 |
11 | boolean equalsIgnoreCase(String anotherString) 将此 String 与另一个 String 比较,不考虑大小写。 |
12 | byte[] getBytes() 使用平台的默认字符集将此 String 编码为 byte 序列,并将结果存储到一个新的 byte 数组中。 |
13 | byte[] getBytes(String charsetName) 使用指定的字符集将此 String 编码为 byte 序列,并将结果存储到一个新的 byte 数组中。 |
14 | void getChars(int srcBegin, int srcEnd, char[] dst, int dstBegin) 将字符从此字符串复制到目标字符数组。 |
15 | int hashCode() 返回此字符串的哈希码。 |
16 | int indexOf(int ch) 返回指定字符在此字符串中第一次出现处的索引。 |
17 | int indexOf(int ch, int fromIndex) 返回在此字符串中第一次出现指定字符处的索引,从指定的索引开始搜索。 |
18 | int indexOf(String str) 返回指定子字符串在此字符串中第一次出现处的索引。 |
19 | int indexOf(String str, int fromIndex) 返回指定子字符串在此字符串中第一次出现处的索引,从指定的索引开始。 |
20 | String intern() 返回字符串对象的规范化表示形式。 |
21 | int lastIndexOf(int ch) 返回指定字符在此字符串中最后一次出现处的索引。 |
22 | int lastIndexOf(int ch, int fromIndex) 返回指定字符在此字符串中最后一次出现处的索引,从指定的索引处开始进行反向搜索。 |
23 | int lastIndexOf(String str) 返回指定子字符串在此字符串中最右边出现处的索引。 |
24 | int lastIndexOf(String str, int fromIndex) 返回指定子字符串在此字符串中最后一次出现处的索引,从指定的索引开始反向搜索。 |
25 | int length() 返回此字符串的长度。 |
26 | boolean matches(String regex) 告知此字符串是否匹配给定的正则表达式。 |
27 | boolean regionMatches(boolean ignoreCase, int toffset, String other, int ooffset, int len) 测试两个字符串区域是否相等。 |
28 | boolean regionMatches(int toffset, String other, int ooffset, int len) 测试两个字符串区域是否相等。 |
29 | String replace(char oldChar, char newChar) 返回一个新的字符串,它是通过用 newChar 替换此字符串中出现的所有 oldChar 得到的。 |
30 | String replaceAll(String regex, String replacement) 使用给定的 replacement 替换此字符串所有匹配给定的正则表达式的子字符串。 |
31 | String replaceFirst(String regex, String replacement) 使用给定的 replacement 替换此字符串匹配给定的正则表达式的第一个子字符串。 |
32 | String[] split(String regex) 根据给定正则表达式的匹配拆分此字符串。 |
33 | String[] split(String regex, int limit) 根据匹配给定的正则表达式来拆分此字符串。 |
34 | boolean startsWith(String prefix) 测试此字符串是否以指定的前缀开始。 |
35 | boolean startsWith(String prefix, int toffset) 测试此字符串从指定索引开始的子字符串是否以指定前缀开始。 |
36 | CharSequence subSequence(int beginIndex, int endIndex) 返回一个新的字符序列,它是此序列的一个子序列。 |
37 | String substring(int beginIndex) 返回一个新的字符串,它是此字符串的一个子字符串。 |
38 | String substring(int beginIndex, int endIndex) 返回一个新字符串,它是此字符串的一个子字符串。 |
39 | char[] toCharArray() 将此字符串转换为一个新的字符数组。 |
40 | String toLowerCase() 使用默认语言环境的规则将此 String 中的所有字符都转换为小写。 |
41 | String toLowerCase(Locale locale) 使用给定 Locale 的规则将此 String 中的所有字符都转换为小写。 |
42 | String toString() 返回此对象本身(它已经是一个字符串!)。 |
43 | String toUpperCase() 使用默认语言环境的规则将此 String 中的所有字符都转换为大写。 |
44 | String toUpperCase(Locale locale) 使用给定 Locale 的规则将此 String 中的所有字符都转换为大写。 |
45 | String trim() 返回字符串的副本,忽略前导空白和尾部空白。 |
46 | static String valueOf(primitive data type x) 返回给定data type类型x参数的字符串表示形式。 |
47 | contains(CharSequence chars) 判断是否包含指定的字符系列。 |
48 | isEmpty() 判断字符串是否为空。 |
StringBuilder和StringBuffer
StringBuilder提供了一系列插入、追加、改变该字符串包含的字符序列的方法
length
和capacity
,length
代表字符序列长度,可以setLength(int len),capacity
表示StringBuilder的容量,通常大于length
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| public class StringBuilderTest { public static void main(String[] args) { StringBuilder sb = new StringBuilder(); sb.append("java"); sb.insert(0 , "hello "); sb.replace(5, 6, ","); sb.delete(5, 6); System.out.println(sb); sb.reverse(); System.out.println(sb); System.out.println(sb.length()); System.out.println(sb.capacity()); sb.setLength(5); System.out.println(sb); } }
|
Math类
提供更复杂的数学运算,构造器为 private ,方法均为类方法,以及两个类变量 PI 和 E
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
| public class MathTest { public static void main(String[] args) { System.out.println("Math.toDegrees(1.57):" + Math.toDegrees(1.57)); System.out.println("Math.toRadians(90):" + Math.toRadians(90)); System.out.println("Math.acos(1.2):" + Math.acos(1.2)); System.out.println("Math.asin(0.8):" + Math.asin(0.8)); System.out.println("Math.atan(2.3):" + Math.atan(2.3)); System.out.println("Math.cos(1.57):" + Math.cos(1.57)); System.out.println("Math.cosh(1.2 ):" + Math.cosh(1.2 )); System.out.println("Math.sin(1.57 ):" + Math.sin(1.57 )); System.out.println("Math.sinh(1.2 ):" + Math.sinh(1.2 )); System.out.println("Math.tan(0.8 ):" + Math.tan(0.8 )); System.out.println("Math.tanh(2.1 ):" + Math.tanh(2.1 )); System.out.println("Math.atan2(0.1, 0.2):" + Math.atan2(0.1, 0.2)); System.out.println("Math.floor(-1.2 ):" + Math.floor(-1.2 )); System.out.println("Math.ceil(1.2):" + Math.ceil(1.2)); System.out.println("Math.round(2.3 ):" + Math.round(2.3 )); System.out.println("Math.sqrt(2.3 ):" + Math.sqrt(2.3 )); System.out.println("Math.cbrt(9):" + Math.cbrt(9)); System.out.println("Math.exp(2):" + Math.exp(2)); System.out.println("Math.hypot(4 , 4):" + Math.hypot(4 , 4)); System.out.println("Math.IEEEremainder(5 , 2):" + Math.IEEEremainder(5 , 2)); System.out.println("Math.pow(3, 2):" + Math.pow(3, 2)); System.out.println("Math.log(12):" + Math.log(12)); System.out.println("Math.log10(9):" + Math.log10(9)); System.out.println("Math.log1p(9):" + Math.log1p(9)); System.out.println("Math.abs(-4.5):" + Math.abs(-4.5)); System.out.println("Math.copySign(1.2, -1.0):" + Math.copySign(1.2, -1.0)); System.out.println("Math.signum(2.3):" + Math.signum(2.3)); System.out.println("Math.max(2.3 , 4.5):" + Math.max(2.3 , 4.5)); System.out.println("Math.min(1.2 , 3.4):" + Math.min(1.2 , 3.4)); System.out.println("Math.nextAfter(1.2, 1.0):" + Math.nextAfter(1.2, 1.0)); System.out.println("Math.nextUp(1.2 ):" + Math.nextUp(1.2 )); System.out.println("Math.random():" + Math.random()); } }
|
ThreadLocalRandom (Java 7)与 Random
Random 类专门生成伪随机数,两个构造器,一个以当前时间为种子,另一个需显示传入 long 型证书作为种子
ThreadLocalRandom 在并发访问环境可以减少线程资源竞争,提高线程安全性,通过静态 current() 方法返回 ThreadLocalRandom 对象
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
| public class RandomTest { public static void main(String[] args) { Random rand = new Random(); System.out.println("rand.nextBoolean():" + rand.nextBoolean()); byte[] buffer = new byte[16]; rand.nextBytes(buffer); System.out.println(Arrays.toString(buffer)); System.out.println("rand.nextDouble():" + rand.nextDouble()); System.out.println("rand.nextFloat():" + rand.nextFloat()); System.out.println("rand.nextGaussian():" + rand.nextGaussian()); System.out.println("rand.nextInt():" + rand.nextInt()); System.out.println("rand.nextInt(26):" + rand.nextInt(26)); System.out.println("rand.nextLong():" + rand.nextLong()); } }
|
如果两个Random对象的种子相同,且方法的调用顺序相同,则会产生相同的数字序列(伪随机),推荐使用当前时间作为Random对象的种子 System.currentTimeMillis()
ThreadLoaclRandom用法:
1 2 3
| ThreadLocalRandom rand = ThreadLoaclRandom.current(); int val1 = rand.nextInt(4, 20); int val2 = rand.nextDouble(2.0, 10.0);
|
BigDecimal类
BigDecimal 类可以实现精确的浮点运算,float、double两种基本浮点类型会引起精度丢失
不推荐 new BigDecimal( 0.1 ) , 推荐 new BigDecimal( “0.1” ),因为浮点数的值不会正好等于0.1
如果使用浮点数作为参数,则使用 BigDecimal.valueOf(0.01);
提供了 add()、substract()、multiply()、divide()、pow()等精确浮点数运算方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| public class BigDecimalTest { public static void main(String[] args) { BigDecimal f1 = new BigDecimal("0.05"); BigDecimal f2 = BigDecimal.valueOf(0.01); BigDecimal f3 = new BigDecimal(0.05); System.out.println("使用String作为BigDecimal构造器参数:"); System.out.println("0.05 + 0.01 = " + f1.add(f2)); System.out.println("0.05 - 0.01 = " + f1.subtract(f2)); System.out.println("0.05 * 0.01 = " + f1.multiply(f2)); System.out.println("0.05 / 0.01 = " + f1.divide(f2)); System.out.println("使用double作为BigDecimal构造器参数:"); System.out.println("0.05 + 0.01 = " + f3.add(f2)); System.out.println("0.05 - 0.01 = " + f3.subtract(f2)); System.out.println("0.05 * 0.01 = " + f3.multiply(f2)); System.out.println("0.05 / 0.01 = " + f3.divide(f2)); } }
|
定义一个 Arith 工具类,提供double类型的精确计算
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48
| public class Arith { private static final int DEF_DIV_SCALE = 10; private Arith() {} public static double add(double v1,double v2) { BigDecimal b1 = BigDecimal.valueOf(v1); BigDecimal b2 = BigDecimal.valueOf(v2); return b1.add(b2).doubleValue(); } public static double sub(double v1,double v2) { BigDecimal b1 = BigDecimal.valueOf(v1); BigDecimal b2 = BigDecimal.valueOf(v2); return b1.subtract(b2).doubleValue(); } public static double mul(double v1,double v2) { BigDecimal b1 = BigDecimal.valueOf(v1); BigDecimal b2 = BigDecimal.valueOf(v2); return b1.multiply(b2).doubleValue(); } public static double div(double v1,double v2) { BigDecimal b1 = BigDecimal.valueOf(v1); BigDecimal b2 = BigDecimal.valueOf(v2); return b1.divide(b2 , DEF_DIV_SCALE , RoundingMode.HALF_UP).doubleValue(); } public static void main(String[] args) { System.out.println("0.05 + 0.01 = " + Arith.add(0.05 , 0.01)); System.out.println("1.0 - 0.42 = " + Arith.sub(1.0 , 0.42)); System.out.println("4.015 * 100 = " + Arith.mul(4.015 , 100)); System.out.println("123.3 / 100 = " + Arith.div(123.3 , 100)); } }
|
Date类(java.util.Date)
用于处理日期与时间,大部分方法、构造器已经过时
构造器
- Date() 底层调用System.currentTimeMillis() ,生成当前日期参数的Date对象
- Date( long date )
方法
- boolean after( date when )
- boolean before( date when )
- long getTime()
- void setTime()
1 2 3 4 5 6 7 8 9 10 11 12
| public class DateTest { public static void main(String[] args) { Date d1 = new Date(); Date d2 = new Date(System.currentTimeMillis() + 100); System.out.println(d2); System.out.println(d1.compareTo(d2)); System.out.println(d1.before(d2)); } }
|
Calender 类
Calander 是一个抽象类,是所有日历类的模板,本身不能实例化,Java提供了一个 GregorianCalendar 类,一个代表格里高利日历的子类。也可以创建自己的 Calendar 子类,作为 Calendar 对象使用(多态)
通过getInstance()方法获取Calendar对象,依据TimeZone和Local类
tips:Month范围 0~11,代表1~12月
void add(int field, int amount)
加减指定的时间量,上一级字段会进位,下一级字段会修正到变化最小的值 ,如 2003-8-31 =》 2004-2-29
int get(int field)
int getActualMaximum(int field)
int getActualMinimum(int field)
void roll(int field, int amount)
与add类似,但不会向上一字段进位
void set(int field, int value)
void set(int year, int month, int date)
void set(int year, int month, int date, int hourOfDay, int minute, int second)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52
| import java.util.*; import static java.util.Calendar.*; public class CalendarTest { public static void main(String[] args) { Calendar c = Calendar.getInstance(); System.out.println(c.get(YEAR)); System.out.println(c.get(MONTH)); System.out.println(c.get(DATE)); c.set(2003 , 10 , 23 , 12, 32, 23); System.out.println(c.getTime()); c.add(YEAR , -1); System.out.println(c.getTime()); c.roll(MONTH , -8); System.out.println(c.getTime());
Calendar cal1 = Calendar.getInstance(); cal1.set(2003, 7, 23, 0, 0 , 0); cal1.add(MONTH, 6); System.out.println(cal1.getTime());
Calendar cal2 = Calendar.getInstance(); cal2.set(2003, 7, 31, 0, 0 , 0); cal2.add(MONTH, 6); System.out.println(cal2.getTime());
Calendar cal3 = Calendar.getInstance(); cal3.set(2003, 7, 23, 0, 0 , 0); cal3.roll(MONTH, 6); System.out.println(cal3.getTime());
Calendar cal4 = Calendar.getInstance(); cal4.set(2003, 7, 31, 0, 0 , 0); cal4.roll(MONTH, 6); System.out.println(cal4.getTime()); } }
|
Calendar 的容错性
setLenient(false) 可以关闭容错性(non-lenient),进行严格的参数检查,超出允许范围将抛出异常
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| public class LenientTest { public static void main(String[] args { Calendar cal = Calendar.getInstance(); cal.set(MONTH , 13); System.out.println(cal.getTime()); cal.setLenient(false); cal.set(MONTH , 13); System.out.println(cal.getTime()); } }
|
set() 方法的延迟修改
日历字段 f 立即更改,但 Calendar 代表的时间不会立即修改,需等到下次 get()、add()等时才会重新计算日历的时间,防止多次set()时不必要的计算
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| public class LazyTest { public static void main(String[] args) { Calendar cal = Calendar.getInstance(); cal.set(2003 , 7 , 31); cal.set(MONTH , 8);
cal.set(DATE , 5); System.out.println(cal.getTime()); } }
|
新的日期、时间包(java.time)
Java 8 新增 java.time 包,包含了以下类:
- Clock(指定时区的当前日期、时间)
- Duration(持续时间,一段时间)
- Instant (具体时刻)
- LocalDate(不带时区的日期)2007-12-03
- LocalTime(不带时区的时间)10:15:30
- LocalDateTime 2007-12-03T10:15:30
- MonthDay(月日) –04-12
- Year 2014
- YearMonth(年月) 2014-04
- ZonedDateTime(时区化的日期、时间)
- ZoneId(代表一个时区)
- DayOfWeek (枚举类,周日到周六的枚举值)
- Month(枚举类,一月到十二月的枚举值)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
| import java.time.*; public class NewDatePackageTest { public static void main(String[] args) { Clock clock = Clock.systemUTC(); System.out.println("当前时刻为:" + clock.instant()); System.out.println(clock.millis()); System.out.println(System.currentTimeMillis()); Duration d = Duration.ofSeconds(6000); System.out.println("6000秒相当于" + d.toMinutes() + "分"); System.out.println("6000秒相当于" + d.toHours() + "小时"); System.out.println("6000秒相当于" + d.toDays() + "天"); Clock clock2 = Clock.offset(clock, d); System.out.println("当前时刻加6000秒为:" +clock2.instant()); Instant instant = Instant.now(); System.out.println(instant); Instant instant2 = instant.plusSeconds(6000); System.out.println(instant2); Instant instant3 = Instant.parse("2014-02-23T10:12:35.342Z"); System.out.println(instant3); Instant instant4 = instant3.plus(Duration .ofHours(5).plusMinutes(4)); System.out.println(instant4); Instant instant5 = instant4.minus(Duration.ofDays(5)); System.out.println(instant5); LocalDate localDate = LocalDate.now(); System.out.println(localDate); localDate = LocalDate.ofYearDay(2014, 146); System.out.println(localDate); localDate = LocalDate.of(2014, Month.MAY, 21); System.out.println(localDate); LocalTime localTime = LocalTime.now(); localTime = LocalTime.of(22, 33); System.out.println(localTime); localTime = LocalTime.ofSecondOfDay(5503); System.out.println(localTime); LocalDateTime localDateTime = LocalDateTime.now(); LocalDateTime future = localDateTime.plusHours(25).plusMinutes(3); System.out.println("当前日期、时间的25小时3分之后:" + future); Year year = Year.now(); System.out.println("当前年份:" + year); year = year.plusYears(5); System.out.println("当前年份再过5年:" + year); YearMonth ym = year.atMonth(10); System.out.println("year年10月:" + ym); ym = ym.plusYears(5).minusMonths(3); System.out.println("year年10月再加5年、减3个月:" + ym); MonthDay md = MonthDay.now(); System.out.println("当前月日:" + md); MonthDay md2 = md.with(Month.MAY).withDayOfMonth(23); System.out.println("5月23日为:" + md2); } }
|