Java资源网

| JAVA基础 | 环境配置 | JDBC | 线程技术 | Socket编程 | JavaMail | JAVA与XML | 设计模式 | 技术新闻 | Java认证 | 程序人生 软件下载
| JSP&Servlet | Spring | Struts | Hibernate | JBuilder | Eclipse | WebService | EJB技术 | J2ME开发 | 应用服务器 | JXTA | Ajax
Articles search文章搜索
   关键字:
   类 别:
       
New download 最新下载
· [组件]HTML Parser 1.5
· [教程]WebSphere Studio应用教程
· [组件]JDom 1.0
· [工具]Junit3.8.1
· [教程]EJB编程及J2EE系统架构和设计
· [教程]EJB教程
· [教程]J2EE Tutorial中文版
· [教程]Java编程思想2(英文)
· [教程]java编程思想(完整版)
· [教程]Java网络编程
New articles 最新文章
· 设计移动 Web 服务
· 解析XML的时候完全忽略DTD
· 理解XML Schema XML Schema 初步
· 标签库的深入研究
· 提升JSP应用程序的七大绝招
· 如何使用JDOM对XML文件进行操作
· 处理XML字符串中特殊字符
· 利用Digester把XML转换成为Java对象
· 使用WebService 和RMI远程协作
· 使用Axis开发Web Service程序
Articles top 热门文章
· Eclipse基础--plugin插件安装(6644)
· eclipse+tomcat+lomboz的安装配置说明(4774)
· Java程序员就业前景(4584)
· Windows下JAVA环境变量的设置祥解(3788)
· Tomcat下JSP、Servlet和JavaBean环境的配置(3716)
· 使用links方式安装Eclipse插件(3698)
· 一个老程序员的心理话(3533)
· linux下jdk的安装与配置(3459)
· 初学者入门:Structs中基本配置入门(3334)
· Eclipse 运行命令行参数大全(3084)
您的位置:首页>>JAVA基础>>你会重载Equals么?about how to override Equals
你会重载Equals么?about how to override Equals
2005-08-18   来源:CSDN  作者:shouldbe
 

首先你必须了解Equals方法的作用。

默认的Object.Equals方法是比较两个应用是否指向同一对象:

 class A
 {
  public int a;
 }

A a1 = new A ();

a1.a = 10;

A a2 = new A ();

a2.a = 10;

这时,a1.Equals (a2)为False。

默认的ValueType.Equals方法是比较两个struct对象数据结构是否相同:

 struct A
 {
  public int a;
 }

A a1 = new A ();

a1.a = 10;

A a2 = new A ();

a2.a = 10;

这时,a1.Equals (a2)为True。

很多情况下,默认的Equals已经够用。

但各种情况下,你需要重载Equals,那么请看清一下条款在写Equals。

1。重载Equals必须重载GetHashCode

这是一条编译器的rule。如果你重载了Equals不重载GetHashCode,编译器会发出警报。

GetHashCode需要返回一个Int32值,这个值得规律很简单:

      如果两个对象相同,那么对象返回的HashCode必须相同。

      如果两个对象返回的HashCode不相同,那么这两个对象必定不相同。

这样就使得Equals的效率提高很多。因为GetHashCode通常成本比Equals小得多。

打个比方,如果要比较两个collection的对象是否一样(不分顺序),那么就要逐一比对他们内部包含的对象。这是一个费时的比对过程。GetHashCode在这个时候就显得尤为重要。

  public override int GetHashCode ()
  {
   int n = 0;
   foreach (object inner in this)
    n ^= inner.GetHashCode ();
   return n;
  }

  public override bool Equals (object obj)
  {
   if (obj.GetHashCode () != this.GetHashCode ())
    return false;
   // TODO:以下具体实现就不写了
  }

以上的Equals碰到不相同的情况很快就能排除掉,直接就return false了,碰到HashCode相同,那么两个对象可能相同,但不能就依此return true了,还要具体分析。

2。Equals碰到null时的情形

一般人在重写Equals的时候并未考虑碰到null时的情形,一旦被比对对象或者比对对象中一个为null,运行时错误就会产生。

其实解决方法很简单,把a.Equals (b)写成Object.Equals (a, b)。

Object的静态方法Equals会帮你处理null的对象。

所以重载运算符==,!=时也就要用到静态的Equals:

  public static bool operator == (T t1, T t2)
  {
   return Object.Equals (t1, t2);
  }
  public static bool operator != (T t1, T t2)
  {
   reutrn !Object.Equals (t1, t2);
  }


3。Equals对软件复用带来的影响

先来看一个例子:

 class A
 {
  public int a;
  public override int GetHashCode ()
  {
   return a.GetHashCode ();
  }
  public override bool Equals (object obj)
  {
   if (obj.GetHashCode () != this.GetHashCode ())
    return false;
   if (obj is A && (obj as A).a == a)
    return true;
   return false;
  }
 }
 class B
 {
  public long b;
  public override int GetHashCode ()
  {
   return b.GetHashCode();
  }
  public override bool Equals (object obj)
  {
   if (obj.GetHashCode () != this.GetHashCode ())
    return false;
   if (obj is B && (obj as B).b == b)
    return true;
   if (obj is A && (obj as A).a == b)
    return true;
   return false;
  }
 }

 class Program
 {
  static void Main (string[] args)
  {
   A a1 = new A ();
   a1.a = 20;
   B b1 = new B ();
   b1.b = 20;
   Console.WriteLine (a1.Equals (b1));
   Console.WriteLine (b1.Equals (a1));
  }
 }

在这个例子中A只认得自己人,新来的类B不仅认得自己人,还认得以前就有的A。

然后就出现了不该有的一幕:a1.Equals (b1)返回False,而b1.Equals (a1)返回了True。

这可如何是好。系统应该听那个呢?

这就是软件添加新类后遇到的麻烦。你必须改动已有的类来让他认识新来的。但如果已有的类无法修改呢?

办法也是有的。那就是建立一个EqualManager:

public class TwoTypes
{
  Type t1, t2;
  public TwoTypes (Type t1, Type t2)
  {
    this.t1 = t1; this.t2 = t2;
  }
  public override int GetHashCode ()
  {
    reutrn t1.GetHashCode() ^ t2.GetHashCode(); // 不分顺序
  }
  public override bool Equals (object obj)
  {
    if (obj.GetHashCode() != this.GetHashCode())
        return false;
    if (obj is TwoTypes)
    {
       Type tt1 = (obj as TwoTypes).t1, tt2 = (obj as TwoTypes).t2;
       if (tt1 == t1 && tt2 == t2)
         return true;
       if (tt1 == t2 && tt2 == t1)
         return true;
    }
    return false;
  }
}
public class EqualManager : IComparer  // singleton
{
  Hashtable ht = new Hashtable ();
  private EqualManager ()
  {
  }
  public readonly EqualManager Instance = new EqualManager ();
  public int Compare (object o1, object o2)
  {
    TwoTypes tt = new TwoTypes (o1.GetType(), o2.GetType());
    if (ht.ComtainsKey (tt))
    {
      return (ht[tt] as IComparer).Compare (o1, o2);
    }
    else
      return false;
  }
  public void RegisterComparer (Type t1, Type t2, IComparer comp)
  {
    this.ht.Add (new TwoTypes (t1, t2), comp);
  }
}

看看代码就知道具体实现了,需要比对服务的类只要这样重写Equals:
public override bool Equals (object obj)
{
  return EqualManager.Instance.Compare (this, obj) == 0;
}
然后具体的实现就要依赖一个IComparer了。这样既实现了程序的可扩展性。

总结语:
Equals对于较小的项目确实无所谓,但对于较大的项目就显得重要了,毕竟这是底层的东西。幸好它比较简单,各位只要掌握以上三点,我想Equals应该不在话下了把。

只是无聊时写的无聊文章,如有不对的地方敬请更正。

  --相关文章--
· J2EE全面介绍(二) (2007-04-13)
· 项目经验二则:读取war包中的文件及Ant使用中的OutOfMemoryError解决 (2007-04-13)
· 走向J2EE,漫长的道路 (2007-04-13)
· 步入J2EE架构和过程(2) (2007-04-13)
· 步入J2EE架构和过程(1) (2007-04-13)
· 方兴未艾的CORBA (2007-04-13)

版权所有©2005-2006 JAVA资源网 渝ICP备05007591号 虚拟主机 | 关于我们 | 联系方式 | 广告业务 | 网站地图 | 友情链接