51Testing软件测试论坛

标题: Junit高手请 [打印本页]

作者: xuj51test    时间: 2007-4-19 12:53
标题: Junit高手请
junit在断言判断时当需要比较是两个对象时除了重载该类的equals外有更好的方法吗?
作者: cwj007    时间: 2007-5-8 09:57
java的方法均可以适用,可以自己写来比较, 下面是一些java写比较的注意,我发给你
在JAVA中有两种方法比较对象是否相等.  那么首先如何判断一个对象是否相等?判断对象是否相等实际上是比较对象的内存地址,准确的来讲是比较对象的引用是否引用的同一个内存地址。

    “==”就是用来比较内存地址的运算符.  equals();方法就是比较两个对象的值是否相同.

     有以下代码:

String s1 = "abc";
String s2 = "abc";
那么 s1==s2 返回true. s1.equals(s2)也是返回true;
    因为String对象变量s1,s2 都是引用一个String对象"abc"的内存地址的,所以 s1==s2 为true, 又因为s1的值是"abc" ,s2 的值也是"abc", 所以s1.equlas(s2)返回的也是true.

     请看以下代码:

  String s1 = new String("abc");
  String s2 = new String("abc");
   那么s1==s2 返回的是false, 而s1.equals(s2)返回的是true;
    因为s1引用内存中的一个对象(此对象的值为"abc"), s2也引用内存中的一个对象(值也为"abc"),既然"=="号比较的是内存地址.那么这里s1==s2应该返回的是true才对啊, 如果你有这样的想法.那么绝对是由于你会觉得两句new String("abc")创建出来的对象是一样的! 其实不是一样的.为什么呢?? 如果是String s1 ="abc" ;的方式赋值的话,那么虚拟机会在他自己的常量池中先查找有没有"abc"的常量对象.如果有则赋值,如果没有则创建一个,但是 用 new String();语句实际上是在内存中创建一个String对象,那么两次调用new String("abc")方法会创建出不同的对象,所以s1==s2 返回false . equals()方法就不用说了..

   同样.你也可以利用Integer试试:

Integer i1 = new Integer(1);
Integer i2 = new Integer(1);i1 == i2 同样返回false. 但是i1.equals();方法会返回true.
作者: cwj007    时间: 2007-5-8 10:00
下面也是一些方法
比较原始类型的相等与比较两个对象相等是不同的。如果数值5存放在两个不同的int变量中,比较两个变量是否相等将产生结果为 boolean 值 true:
  public class TestIntComparison {
   public static void main(String[] args) {
   int x = 5, y = 5;
   System.out.println(
   "x == y yields " + (x == y));
   }
  }
  TestIntComparison 产生以下输出:
  
  D:\>java TestIntComparison
  x == y yields true
  相等操作符由于原始类型时比较的是它们的值。而用于对象时比较的是对象的引用而不是对象的实际内容。 您可能问:"这些引用都指向同一个对象么?" 为了说明清晰,请看下面只含有 tag 和 age 的 Dog 的又一版本:
  
   class Dog {
   int tag;
   int age;
   public void setTag(int t) {tag=t;}
   public void setAge(int a) {age=a;}
  }
  如果有两只狗(dog),即使它们的内容都相同,用 == 操作符时它们并不相等。下面代码段的输出表明在用 "==" 时 a 和 b 并不相等:
  
  Dog a = new Dog();
  a.setTag(23129);
  a.setAge(7);
  Dog b = new Dog();
  b.setTag(23129);
  b.setAge(7);
  if ( a==b ) {
   System.out.println("a is equal to b");
  }
  else {
   System.out.println("a is not equal to b");
  }
  那么,应该怎么比较两个对象的值而不是比较它们的引用呢?Java(TM) 编程语言有一个约定,方法 equals() 用来定义对象值相等。 类 Object 中定义了方法equals(),如果在其子类中没有被重载,那么默认使用的就是它。 为了比较两只狗(dog) a 和 b 的值,你应该重写上面的比较部分:
  
  if ( a.equals(b) ) {
   System.out.println("a is equals() to b");
  }
  else {
   System.out.println("a is not equals() to b");
  }
  上面的代码中,如果在 Dog 中没有重载 equals() 方法,两只狗依旧不等。因为 Object.equals() 实际模拟的是 == 操作符的功能。 Dog 中 equals() 的定义很好懂:
  
   class Dog {
   int tag;
   int age;
   public void setTag(int t) {tag=t;}
   public void setAge(int a) {age=a;}
   public boolean equals(Object o) {
   Dog d = (Dog)o;
   if ( tag==d.tag && age==d.age ) {
   return true;
   }
   return false;
   }
  }
  为什么 equals() 的参数类型是 Object 而不是 Dog 呢?因为你是在重载父类 Object 的方法 equals(),所以必须用相同的方法标记。但我们希望传进的参数是另一只Dog,所以为了能够访问参数的字段需将其类型转换为 Dog。
  但是,由于 equals() 是在 Dog 中定义的,你必须检查传入的对象是否是一只 Dog,因为有人可能这样用:
   fido.equals("blort");
  字符串 "blort" 也是一个 Object ,因此与 Dog 中 equals() 的标记是匹配的。equals() 的正确写法是:
  
   public boolean equals(Object o) {
   if ( o instanceof Dog ) {
   Dog d = (Dog)o;
   if ( tag==d.tag && age==d.age ) {
   return true;
   }
   }
   // false if not Dog or contents mismatched
   return false;
  }
  操作符 instanceof 询问 o 是否是 Dog (包括 Dog 的子类) 的实例。
  字符串的比较引入了对象比较的最后一个问题,那就是
  "abc"=="def"
  表达式的值为 true 还是 false 呢?是false,因为他们是本质都不同的对象(显而易见,他们的内容都不同)。但是,下面的表达式
  "abc"=="abc"
  是 true 还是 false 呢?不幸的是,这由编译器决定。如果编译器将对 "abc" 的两个引用优化为一个对象而不是两个对象,那么表达式的值为 true。但是,如果编译器不做这种优化,表达式的值则应为 false!
  如果你真的想判断两个字符串在物理上是不是同一个对象,请用 equals() 方法:
  boolean b = "abc".equals("def"); // false
  boolean c = "abc".equals("abc"); // true
作者: liulinzhu    时间: 2007-5-29 09:52

我是一窍不通
作者: a-niu-313    时间: 2007-7-16 09:36
sdlkfj2
作者: zzlovebird    时间: 2009-4-22 19:58
厉害
作者: haibinpark    时间: 2012-2-24 10:23
本帖最后由 haibinpark 于 2012-2-24 10:24 编辑

听明白了,谢谢。@cwj007说的很明白很透彻。




欢迎光临 51Testing软件测试论坛 (http://bbs.51testing.com/) Powered by Discuz! X3.2