Java-类库-Guava-复写的Object常用方法
2016-07-02 17:27
597 查看
在Java中Object类是所有类的父类,其中有几个需要override的方法比如equals,hashCode和toString等方法。每次写这几个方法都要做很多重复性的判断, 很多类库提供了覆写这几个方法的工具类, Guava也提供了类似的方式。下面我们来看看Guava中这几个方法简单使用。
equals方法:
equals是一个经常需要覆写的方法, 可以查看Object的equals方法注释, 对equals有几个性质的要求:
1. 自反性reflexive:任何非空引用x,x.equals(x)返回为true;
2. 对称性symmetric:任何非空引用x和y,x.equals(y)返回true当且仅当y.equals(x)返回true;
3. 传递性transitive:任何非空引用x和y,如果x.equals(y)返回true,并且y.equals(z)返回true,那么x.equals(z)返回true;
4. 一致性consistent:两个非空引用x和y,x.equals(y)的多次调用应该保持一致的结果,(前提条件是在多次比较之间没有修改x和y用于比较的相关信息);
5. 对于所有非null的值x, x.equals(null)都要返回false。 (如果你要用null.equals(x)也可以,会报NullPointerException)。
当我们要覆写的类中某些值可能为null的时候,就需要对null做很多判断和分支处理。 使用Guava的Objects.equal方法可以避免这个问题, 使得equals的方法的覆写变得更加容易, 而且可读性强,简洁优雅。
hashCode方法:
当覆写(override)了equals()方法之后,必须也覆写hashCode()方法,反之亦然。这个方法返回一个整型值(hash code value),如果两个对象被equals()方法判断为相等,那么它们就应该拥有同样的hash code。Object类的hashCode()方法为不同的对象返回不同的值,Object类的hashCode值表示的是对象的地址。
hashCode的一般性契约(需要满足的条件)如下:
1.在Java应用的一次执行过程中,如果对象用于equals比较的信息没有被修改,那么同一个对象多次调用hashCode()方法应该返回同一个整型值。应用的多次执行中,这个值不需要保持一致,即每次执行都是保持着各自不同的值。
2.如果equals()判断两个对象相等,那么它们的hashCode()方法应该返回同样的值。
3.并没有强制要求如果equals()判断两个对象不相等,那么它们的hashCode()方法就应该返回不同的值。即,两个对象用equals()方法比较返回false,它们的hashCode可以相同也可以不同。但是,应该意识到,为两个不相等的对象产生两个不同的hashCode可以改善哈希表的性能。
写一个hashCode本来也不是很难,但是Guava提供给我们了一个更加简单的方法–Objects.hashCode(Object …), 这是个可变参数的方法,参数列表可以是任意数量,所以可以像这样使用Objects.hashCode(field1, field2, …, fieldn)。非常方便和简洁。
toString()方法:
因为每个类都直接或间接地继承自Object,因此每个类都有toString()方法。这个方法是用得最多的, 覆写得最多, 一个好的toString方法对于调试来说是非常重要的, 但是写起来确实很不爽。Guava也提供了toString()方法。
compare/compareTo方法:
CompareTo:compareTo(Object o)方法是java.lang.Comparable接口中的方法,当需要对某个类的对象进行排序时,该类需要实现 Comparable接口的,必须重写public int compareTo(T o)方法。java规定,若a,b是两个对象,当a.compareTo(b)>0时,则a大于b,a.compareTo(b)<0时,a
上面的compareTo方法,代码看上去并不是十分优雅,如果实体属性很多,数据类型丰富,代码可读性将会很差。在guava里, 对所有原始类型都提供了比较的工具函数来避免这个麻烦. 比如对Integer, 可以用Ints.compare()。利用guava的原始类型的compare,我们对上面的方法做一个简化,实现compare方
ComparisonChain是一个lazy的比较过程, 当比较结果为0的时候, 即相等的时候, 会继续比较下去, 出现非0的情况, 就会忽略后面的比较。ComparisonChain实现的compare和compareTo在代码可读性和性能上都有很大的提高。
equals方法:
equals是一个经常需要覆写的方法, 可以查看Object的equals方法注释, 对equals有几个性质的要求:
1. 自反性reflexive:任何非空引用x,x.equals(x)返回为true;
2. 对称性symmetric:任何非空引用x和y,x.equals(y)返回true当且仅当y.equals(x)返回true;
3. 传递性transitive:任何非空引用x和y,如果x.equals(y)返回true,并且y.equals(z)返回true,那么x.equals(z)返回true;
4. 一致性consistent:两个非空引用x和y,x.equals(y)的多次调用应该保持一致的结果,(前提条件是在多次比较之间没有修改x和y用于比较的相关信息);
5. 对于所有非null的值x, x.equals(null)都要返回false。 (如果你要用null.equals(x)也可以,会报NullPointerException)。
当我们要覆写的类中某些值可能为null的时候,就需要对null做很多判断和分支处理。 使用Guava的Objects.equal方法可以避免这个问题, 使得equals的方法的覆写变得更加容易, 而且可读性强,简洁优雅。
import org.junit.Test; import com.google.common.base.Objects; public class ObjectTest { @Test public void equalTest() { System.out.println(Objects.equal("a", "a")); System.out.println(Objects.equal(null, "a")); System.out.println(Objects.equal("a", null)); System.out.println(Objects.equal(null, null)); } @Test public void equalPersonTest() { System.out.println(Objects.equal(new Person("peida",23), new Person("peida",23))); Person person=new Person("peida",23); System.out.println(Objects.equal(person,person)); } } class Person { public String name; public int age; Person(String name, int age) { this.name = name; this.age = age; } }
true false false true false true
hashCode方法:
当覆写(override)了equals()方法之后,必须也覆写hashCode()方法,反之亦然。这个方法返回一个整型值(hash code value),如果两个对象被equals()方法判断为相等,那么它们就应该拥有同样的hash code。Object类的hashCode()方法为不同的对象返回不同的值,Object类的hashCode值表示的是对象的地址。
hashCode的一般性契约(需要满足的条件)如下:
1.在Java应用的一次执行过程中,如果对象用于equals比较的信息没有被修改,那么同一个对象多次调用hashCode()方法应该返回同一个整型值。应用的多次执行中,这个值不需要保持一致,即每次执行都是保持着各自不同的值。
2.如果equals()判断两个对象相等,那么它们的hashCode()方法应该返回同样的值。
3.并没有强制要求如果equals()判断两个对象不相等,那么它们的hashCode()方法就应该返回不同的值。即,两个对象用equals()方法比较返回false,它们的hashCode可以相同也可以不同。但是,应该意识到,为两个不相等的对象产生两个不同的hashCode可以改善哈希表的性能。
写一个hashCode本来也不是很难,但是Guava提供给我们了一个更加简单的方法–Objects.hashCode(Object …), 这是个可变参数的方法,参数列表可以是任意数量,所以可以像这样使用Objects.hashCode(field1, field2, …, fieldn)。非常方便和简洁。
import org.junit.Test; import com.google.common.base.Objects; public class ObjectTest { @Test public void hashcodeTest() { System.out.println(Objects.hashCode("a")); System.out.println(Objects.hashCode("a")); System.out.println(Objects.hashCode("a","b")); System.out.println(Objects.hashCode("b","a")); System.out.println(Objects.hashCode("a","b","c")); Person person=new Person("peida",23); System.out.println(Objects.hashCode(person)); System.out.println(Objects.hashCode(person)); } } class Person { public String name; public int age; Person(String name, int age) { this.name = name; this.age = age; } }
128 4066 4096 126145 19313256 19313256
toString()方法:
因为每个类都直接或间接地继承自Object,因此每个类都有toString()方法。这个方法是用得最多的, 覆写得最多, 一个好的toString方法对于调试来说是非常重要的, 但是写起来确实很不爽。Guava也提供了toString()方法。
import org.junit.Test; import com.google.common.base.Objects; public class ObjectTest { @Test public void toStringTest() { System.out.println(Objects.toStringHelper(this).add("x", 1).toString()); System.out.println(Objects.toStringHelper(Person.class).add("x", 1).toString()); Person person=new Person("peida",23); String result = Objects.toStringHelper(Person.class) .add("name", person.name) .add("age", person.age).toString(); System.out.print(result); } } class Person { public String name; public int age; Person(String name, int age) { this.name = name; this.age = age; } }
compare/compareTo方法:
CompareTo:compareTo(Object o)方法是java.lang.Comparable接口中的方法,当需要对某个类的对象进行排序时,该类需要实现 Comparable接口的,必须重写public int compareTo(T o)方法。java规定,若a,b是两个对象,当a.compareTo(b)>0时,则a大于b,a.compareTo(b)<0时,a
import org.junit.Test; public class ObjectTest { @Test public void compareTest(){ Person person=new Person("peida",23); Person person1=new Person("aida",25); Person person2=new Person("aida",25); Person person3=new Person("aida",26); Person person4=new Person("peida",26); System.out.println(person.compareTo(person1)); System.out.println(person1.compareTo(person2)); System.out.println(person1.compareTo(person3)); System.out.println(person.compareTo(person4)); System.out.println(person4.compareTo(person)); } } class Person implements Comparable<Person>{ public String name; public int age; Person(String name, int age) { this.name = name; this.age = age; } @Override public int compareTo(Person other) { int cmpName = name.compareTo(other.name); if (cmpName != 0) { return cmpName; } if(age>other.age){ return 1; } else if(age<other.age){ return -1; } return 0; } }
上面的compareTo方法,代码看上去并不是十分优雅,如果实体属性很多,数据类型丰富,代码可读性将会很差。在guava里, 对所有原始类型都提供了比较的工具函数来避免这个麻烦. 比如对Integer, 可以用Ints.compare()。利用guava的原始类型的compare,我们对上面的方法做一个简化,实现compare方
class Student implements Comparable<Student>{ public String name; public int age; public int score; Student(String name, int age,int score) { this.name = name; this.age = age; this.score=score; } @Override public int compareTo(Student other) { return ComparisonChain.start() .compare(name, other.name) .compare(age, other.age) .compare(score, other.score, Ordering.natural().nullsLast()) .result(); } } class StudentComparator implements Comparator<Student> { @Override public int compare(Student s1, Student s2) { return ComparisonChain.start() .compare(s1.name, s2.name) .compare(s1.age, s2.age) .compare(s1.score, s2.score) .result(); } } }
ComparisonChain是一个lazy的比较过程, 当比较结果为0的时候, 即相等的时候, 会继续比较下去, 出现非0的情况, 就会忽略后面的比较。ComparisonChain实现的compare和compareTo在代码可读性和性能上都有很大的提高。
import java.util.Comparator; import org.junit.Test; import com.google.common.base.Objects; import com.google.common.collect.ComparisonChain; import com.google.common.collect.Ordering; public class ObjectTest { @Test public void StudentTest(){ Student student=new Student("peida",23,80); Student student1=new Student("aida",23,36); Student student2=new Student("jerry",24,90); Student student3=new Student("peida",23,80); System.out.println("==========equals==========="); System.out.println(student.equals(student2)); System.out.println(student.equals(student1)); System.out.println(student.equals(student3)); System.out.println("==========hashCode==========="); System.out.println(student.hashCode()); System.out.println(student1.hashCode()); System.out.println(student3.hashCode()); System.out.println(student2.hashCode()); System.out.println("==========toString==========="); System.out.println(student.toString()); System.out.println(student1.toString()); System.out.println(student2.toString()); System.out.println(student3.toString()); System.out.println("==========compareTo==========="); System.out.println(student.compareTo(student1)); System.out.println(student.compareTo(student2)); System.out.println(student2.compareTo(student1)); System.out.println(student2.compareTo(student)); } } class Student implements Comparable<Student>{ public String name; public int age; public int score; Student(String name, int age,int score) { this.name = name; this.age = age; this.score=score; } @Override public int hashCode() { return Objects.hashCode(name, age); } @Override public boolean equals(Object obj) { if (obj instanceof Student) { Student that = (Student) obj; return Objects.equal(name, that.name) && Objects.equal(age, that.age) && Objects.equal(score, that.score); } return false; } @Override public String toString() { return Objects.toStringHelper(this) .addValue(name) .addValue(age) .addValue(score) .toString(); } @Override public int compareTo(Student other) { return ComparisonChain.start() .compare(name, other.name) .compare(age, other.age) .compare(score, other.score, Ordering.natural().nullsLast()) .result(); } } class StudentComparator implements Comparator<Student> { @Override public int compare(Student s1, Student s2) { return ComparisonChain.start() .compare(s1.name, s2.name) .compare(s1.age, s2.age) .compare(s1.score, s2.score) .result(); } } //=============运行输出=========================== ==========equals=========== false false true ==========hashCode=========== -991998617 92809683 -991998617 -1163491205 ==========toString=========== Student{peida, 23, 80} Student{aida, 23, 36} Student{jerry, 24, 90} Student{peida, 23, 80} ==========compareTo=========== 1 1 1 -1
相关文章推荐
- objective-C程序设计学习笔记3--分类和协议
- OpenGL Frame BufferObject(FBO)
- Can't get WebApplicationContext object from ContextRegistry.GetContext(): Resource handler for the 'web' protocol is not defined
- YYModel 源码解读(二)之NSObject+YYModel.h (5)
- Intellij IDEA 编译提示: "Test is already defined as object Test"
- POST请求报错:Error Domain=NSCocoaErrorDomain Code=3840 "No string key for value in object around charact
- java中Object转String
- objc runtime 动态增加属性
- 关于EnumerateObjectsUsingBlock和for-in之间的较量
- Objective-C编程之——发送消息Send Message
- java的API/Object
- 欢迎使用CSDN-markdown编辑器
- 创建一个Object()对象
- 创建一个Object()对象
- 创建一个Object()对象
- 创建一个Object()对象
- 创建一个Object()对象
- 创建一个Object()对象
- 创建一个Object()对象
- 创建一个Object()对象