目录:

  • 建议10:创建对象时需要考虑是否实现比较器
  • 建议11:区别对待==与Equals
  • 建议12:重写Equals时也要重写GetHashCode

一、建议10:创建对象时需要考虑是否实现比较器

比较一下基本工资:

 class Salary : IComparable
{
public string Name { get; set; }
public decimal BaseSalary { get; set; }
public decimal Bonus { get; set; }
public int CompareTo(object obj)
{
Salary comparer = obj as Salary; if (BaseSalary > comparer.BaseSalary)
{
return ;
}
else if (BaseSalary == comparer.BaseSalary)
{
return ;
}
else
{
return -;
}
}
}

客户端调用:

 List<Salary> salaries = new List<Salary>();
salaries.Add(new Salary() { Name = "Sun", BaseSalary = });
salaries.Add(new Salary() { Name = "Yuan", BaseSalary = });
salaries.Add(new Salary() { Name = "Kun", BaseSalary = });
salaries.Add(new Salary() { Name = "Qun", BaseSalary = });
salaries.Add(new Salary() { Name = "Sun", BaseSalary = }); salaries.Sort(); foreach (var s in salaries)
{
Console.WriteLine("【Name】:{0},【BaseSalary】:¥{1}{2}", s.Name, s.BaseSalary,System.Environment.NewLine);
} Console.ReadKey();

运行:

如果不想用基本工资BaseSalary进行排序,而是以奖金Bonus进行排序,使用IComparer实现自定义比较器:

class BonusComparer : IComparer<Salary>
{
public int Compare(Salary x, Sarlary y)
{ return left.Bonus.CompareTo(right.Bonus);
}
}

客户端提供我们上面创建的比较器:

            List<Salary> salaries = new List<Salary>();
salaries.Add(new Salary() { Name = "Sun", BaseSalary = ,Bonus= });
salaries.Add(new Salary() { Name = "Yuan", BaseSalary = , Bonus = });
salaries.Add(new Salary() { Name = "Kun", BaseSalary = , Bonus = });
salaries.Add(new Salary() { Name = "Qun", BaseSalary = ,Bonus= });
salaries.Add(new Salary() { Name = "Dun", BaseSalary = ,Bonus= }); salaries.Sort(new BonusComparer()); foreach (var s in salaries)
{
Console.WriteLine("Name:【{0}】,BaseSalary:¥{1},Bonus:{2}{3}", s.Name, s.BaseSalary,s.Bonus, System.Environment.NewLine);
} Console.ReadKey();

输出:

二、建议11:区别对待==与Equals

两者都是指相等性,即:值相等性和引用相等性。

值类型:如果值类型相等,返回True。

引用类型:如果指向同一个引用,返回True。

很好理解,举个例子:

1、值类型:==与Equls()

            int x = ;

            int y = ;

            Console.WriteLine("int x=1;{0}int y=1;{0}", System.Environment.NewLine,System.Environment.NewLine);

            Console.WriteLine("x==y:{0}",x == y);

            Console.WriteLine("x.Equals(y):{0}{1}",x.Equals(y),System.Environment.NewLine);

            x = y;

            Console.WriteLine("x=y;{0}", System.Environment.NewLine);

            Console.WriteLine("x==y:{0}", x == y);

            Console.WriteLine("x.Equals(y):{0}", x.Equals(y));

            Console.ReadKey();

运行:

2、引用类型

class People
{
public String Name { get; set; }
}

客户端:

            People p1 = new People() { Name = "Sun" };

            People p2 = new People() { Name = "Yuan" };

            Console.WriteLine("People p1 = new People();{0}People p2 = new People();{1}", System.Environment.NewLine, System.Environment.NewLine);

            Console.WriteLine("p1==p2:{0}", p1 == p2);

            Console.WriteLine("p1.Equals(p2):{0}{1}", p1.Equals(p2), System.Environment.NewLine);

            Console.WriteLine("------------------------------------");

            p1 = p2;

            p1.Name = "Moon";

            Console.WriteLine("p1=p2;{0}", System.Environment.NewLine);

            Console.WriteLine("p1==p2:{0}", p1 == p2);

            Console.WriteLine("p1.Equals(p2):{0}", p1.Equals(p2));

运行:

后面我们修改了p1里Name="Moon"的值,但是,p2的Name值也变成了Moon。所以,==与Equal()在比较引用类型时,引用地址一样,返回True

3、引用类型重载Equals()达到值类型比较效果

还有一点,有时我们需要我们的类型看上去和string类型类似,有值类型的感觉。所以说,我们的这个引用类型,需要重载==或者Equals()。

这里建议只重载Equals()来达到像值类型一样的比较效果。保留==,保留引用比较。例如:生活中我们认为身份证号码一样的是同一个人。

 class People
{
public String Name { get; set; }
public string IDCode { get; set; } public override bool Equals(object obj)
{ People p = obj as People; return p.IDCode == IDCode;
}
}

客户端:

 People p1 = new People() { IDCode="No1" };

            People p2 = new People() { IDCode = "No1" };

            Console.WriteLine("People p1 = new People();{0}People p2 = new People();{1}", System.Environment.NewLine, System.Environment.NewLine);

            Console.WriteLine("p1.IDCode={0}", p1.IDCode);
Console.WriteLine("p2.IDCode={0}", p2.IDCode);
Console.WriteLine(); Console.WriteLine("p1==p2:{0}【保留引用地址的对比】", p1 == p2); Console.WriteLine("p1.Equals(p2):{0}【重载比较IDCode,值类型比较效果】{1}", p1.Equals(p2), System.Environment.NewLine); Console.WriteLine("----------------------------------"); p1 = p2; p1.IDCode = "No2"; Console.ForegroundColor = ConsoleColor.Red; Console.WriteLine("p1.IDCode={0}", p1.IDCode);
Console.WriteLine("p2.IDCode={0}", p2.IDCode);
Console.WriteLine(); Console.ForegroundColor = ConsoleColor.White; Console.WriteLine("p1=p2;{0}", System.Environment.NewLine); Console.WriteLine("p1==p2:{0}", p1 == p2); Console.WriteLine("p1.Equals(p2):{0}", p1.Equals(p2)); Console.ReadKey();

运行:

还有,Object.ReferenceEquals方法比较实例是否相同。验证引用的相等性。

三、建议12:重写Equals时也要重写GetHashCode

当我们重写Equals时,编译器会提示一条警告:

为什么会有这样的提示?

因为在 System.Collections.Hashtable类型和System.Collections .Generic.Dictionary类型以及一些其他的集合类,要求两个对象相等,必须具有相同的哈希码。

所以在重写Equals时,也应该重写GetashCode,确保相等性的算法和对象哈希码算法一致。

添加:添加一个新的键值对时,首先会获取对象的哈希码,这个哈希码指出键值对应该存在哪一个哈希桶中。

查询:查询集合的一个键时,也获取指定键对象的哈希码,这个哈希码指定了我们查找键值对所存在哪一个哈希桶中。所以我们就去哈希桶中搜索与当前指定的键对象的哈希码。

看一下下面这个实例:

class Program
{
static Dictionary<People, string> ppdic = new Dictionary<People, string>(); static void AddPP()
{
People p = new People(){Name="Sun"}; ppdic.Add(p, "Sun");

        //Console.WriteLine(p.GetHashCode());          
Console.WriteLine(ppdic.ContainsKey(p)); } static void Main(string[] args)
{
AddPP(); People pp = new People() { Name = "Sun" };         //Console.WriteLine(pp.GetHashCode()); Console.WriteLine(ppdic.ContainsKey(pp)); Console.ReadKey();
}
}
class People
{
public String Name { get; set; }
public string IDCode { get; set; } public override bool Equals(object obj)
{ People p = obj as People; return p.IDCode == IDCode;
}
}

这里,我们重写了People类的Equals方法,客户端中,首先调用了AddPP()方法,添加一个Name="Sun"的People对象.

紧跟着,我们也定义了同样一个Name="Sun"的People对象。因该说两次People对象一样,所以两次输出都应该为True.

(这里我们不重写Equals效果也是一样的,但这里的重点是说明:GetHashCode)

为什么相同的对象返回的结果不一样?其实上面说过了,CLR会为每个对象创建唯一的哈希码(在生存周期内),因为当前类没有重写GetHashCode方法,所以会调用Object的GetHashCode。

解开上面的两句注释,运行:我们看到两个实例对象(p、pp)的哈希码是不一样的。

如果我们定义的自定义类型会被用作字典等类型的Key值,那我们可能需要重写Equals的同时还要重写GetHashCode。以来正确地实现我们的需求。

 class People
{
public String Name { get; set; }
public string IDCode { get; set; } public override bool Equals(object obj)
{ People p = obj as People; return p.IDCode == IDCode;
} public override int GetHashCode()
{
if (IDCode != null)
return this.IDCode.GetHashCode();
return base.GetHashCode();
}
}

为了产生更好的哈希值的随机分布:

public override int GetHashCode()
{
if (IDCode != null)
return (System.Reflection.MethodBase.GetCurrentMethod().DeclaringType.FullName + "$" + this.IDCode).GetHashCode();
return base.GetHashCode();
}

当然最后我们也最好实现继承类型安全接口:IEquatable<People>

我们的身份证IDCode设计为只读属性,实例化时跟随一个身份证。

最终版本:

 class People:IEquatable<People>
{ public People(string idCode)
{
this._idCode = idCode;
}
public String Name { get; set; } private string _idCode;
public string IDCode { get; private set; } public override bool Equals(object obj)
{ People p = obj as People; return p.IDCode == IDCode;
} public override int GetHashCode()
{
if (IDCode != null)
return (System.Reflection.MethodBase.GetCurrentMethod().DeclaringType.FullName + "$" + this.IDCode).GetHashCode();
return base.GetHashCode();
} public bool Equals(People other)
{
return IDCode == other.IDCode;
}
}

读改善c#代码157个建议:建议10~12的更多相关文章

  1. 读改善c#代码157个建议:建议4~6

    目录: 建议4:TryParse比Parse好 建议5:使用int?确保值类型也可以为null 建议6:区别 readonly 和 const 的用法 一.建议4:TryParse比Parse 好 T ...

  2. 读改善c#代码157个建议:建议13~15

    目录: 建议13:为类型输出格式化字符串 建议14:正确实现浅拷贝和深拷贝 建议15:使用dynamic来简化反射实现 一.建议13:为类型输出格式化字符串 有些类型需要我们根据业务需求提供字符串的格 ...

  3. 读改善c#代码157个建议:建议7~9

    目录: 建议7:将0值作为枚举的默认值 建议8:避免给枚举类型的元素提供显示的值 建议9:习惯运算符重载 一.建议7:将0值作为枚举的默认值 允许使用的枚举类型有:byte.sbyte.short.u ...

  4. 读改善c#代码157个建议:建议1~3

    目录: 建议一:正确操作字符串 建议二:使用默认转型方法 建议三:区别对待强制转型和as 建议一.正确操作字符串 1.确保尽量少的装箱 static void Main(string[] args) ...

  5. 编写高质量代码改善程序的157个建议:第87个建议之区分WPF和WinForm的线程模型

    今天有时间了,继续<编写高质量代码改善程序的157个建议>的阅读,当我阅读到建议87的时候,里面的一些代码示例和文中所说的不一致了,是不是我现在用的是NetFramework 4.0的缘故 ...

  6. 编写高质量代码改善java程序的151个建议——导航开篇

    2014-05-16 09:08 by Jeff Li 前言 系列文章:[传送门] 下个星期度过这几天的奋战,会抓紧java的进阶学习.听过一句话,大哥说过,你一个月前的代码去看下,慘不忍睹是吧.确实 ...

  7. 编写高质量代码:改善Java程序的151个建议 --[117~128]

    编写高质量代码:改善Java程序的151个建议 --[117~128] Thread 不推荐覆写start方法 先看下Thread源码: public synchronized void start( ...

  8. Github即将破百万的PDF:编写高质量代码改善JAVA程序的151个建议

    在通往"Java技术殿堂"的路上,本书将为你指点迷津!内容全部由Java编码的最佳 实践组成,从语法.程序设计和架构.工具和框架.编码风格和编程思想等五大方面,对 Java程序员遇 ...

  9. 每周一书-编写高质量代码:改善C程序代码的125个建议

    首先说明,本周活动有效时间为2016年8月28日到2016年9月4日.本周为大家送出的书是由机械工业出版社出版,马伟编著的<编写高质量代码:改善C程序代码的125个建议>. 编辑推荐 10 ...

随机推荐

  1. SQL SERVER FOR 多列字符串连接 XML PATH 及 STUFF

    原文:SQL SERVER FOR 多列字符串连接 XML PATH 及 STUFF 本来用 Writer 写一篇关于一列多行合并的博客来的,结果快写完了时候,在一个插入代码时候,崩了,重新打开,居然 ...

  2. jquery autocomplete ajax获取动态数据,兼容各浏览器,支持中文

    jquery.autocomplete.js经过改动,支持各种浏览器.支持中文输入! 1.效果图例如以下 2.HTML和ajax代码 <!DOCTYPE html> <html xm ...

  3. 【OC加强】NSDate的使用方法——日期时间在实际开发中比較有用

    (1)日期的最主要知识点就是日期转换成字符串格式化输出,相反就是依照某个格式把字符串转换成日期. (2)一般关于时区的设置非常少用到,仅仅要了解就可以. #import <Foundation/ ...

  4. SOHO路由器的静态路由的不同

    网络拓扑如下,其中RA与RB皆为TP-LINK家用路由器 最终在TP-LINK官网的官网上找到这么一段话 静态路由是在路由器中手工设置的固定的路由条目.我司路由器静态路由是基于ICMP重定向原理,与其 ...

  5. 2-07. 素因子分解(20) (ZJUPAT 数学)

    题目链接:http://pat.zju.edu.cn/contests/ds/2-07 给定某个正整数N,求其素因子分解结果,即给出其因式分解表达式 N = p1^k1 * p2^k2 *-*pm ^ ...

  6. u-boot TFTP: &#39;Access violation&#39; (2)

    今天做tftp下载时间会遇到以下问题. --->8--- Load address: 0x20000000 Loading: * TFTP error: 'Access violation' ( ...

  7. PHP图像处理:3D图纸、缩放、回转、剪下、水印(三)

    来源:http://www.ido321.com/887.html 5.PHP对图像的旋转 1: <div> 2: <h4>旋转之前</h4> 3: <img ...

  8. 猫学习IOS(三)UI纯代码UI——图片浏览器

    猫分享.必须精品 看看效果 主要实现相似看新闻的一个界面,不用拖拽,纯代码手工写. 首先分析app能够非常easy知道他这里有两个UILabel一个UIImageView还有两个UIButton 定义 ...

  9. C# Windows Phone 8 WP8 开发,将WebClient的DownloadStringCompleted事件改成非同步的awiat方法。

    原文:C# Windows Phone 8 WP8 开发,将WebClient的DownloadStringCompleted事件改成非同步的awiat方法. 一般我们在撰写Windows Phone ...

  10. Chromium Graphics: GPUclient的原理和实现分析之间的同步机制-Part II

    摘要:Part I探析GPUclient之间的同步问题,以及Chromium的GL扩展同步点机制的基本原理.本文将源码的角度剖析同步点(SyncPoint)机制的实现方式. 同步点机制的实现主要涉及到 ...