C#中 ==与equals的区别

using System;

internal class Person
{
    public Person(string name)
    {
        Name = name;
    }

    public string Name { get; set; }
}

internal class Program
{
    private static void Main()
    {
        var a = new string(new[] {‘h‘, ‘e‘, ‘l‘, ‘l‘, ‘o‘});
        var b = new string(new[] {‘h‘, ‘e‘, ‘l‘, ‘l‘, ‘o‘});
        Console.WriteLine("1.a == b:" + (a == b)); //True
        Console.WriteLine("2.a.Equals(b):" + a.Equals(b)); //True

        object g = a;
        object h = b;
        Console.WriteLine("3.g == h:" + (g == h)); //False
        Console.WriteLine("4.g.Equals(h):" + g.Equals(h)); //True

        var p1 = new Person("jia");
        var p2 = new Person("jia");
        Console.WriteLine("5.p1 == p2:" + (p1 == p2)); //False
        Console.WriteLine("6.p1.Equals(p2):" + p1.Equals(p2)); //False

        var p3 = new Person("jia");
        var p4 = p3;
        Console.WriteLine("7.p3 == p4:" + (p3 == p4)); //True
        Console.WriteLine("8.p3.Equals(p4):" + p3.Equals(p4)); //True

        Console.ReadLine();
    }
}

注意:在实际使用中,.NET把string弄成了值类型。所以不要把string当引用类型看。

因为值类型是存储在内存中的堆栈(以后简称栈),而引用类型的变量在栈中仅仅是存储引用类型变量的地址,而其本身则存储在堆中。

操作比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量在堆中存储的地址是否相同,即栈中的内容是否相同。

操作表示的两个变量是否是对同一个对象的引用,即堆中的内容是否相同。

而字符串是一个特殊的引用型类型,在C#语言中,重载了string 对象的很多方法方法(包括equals()方法),使string对象用起来就像是值类型一样。

因此在上面的例子中,第一对输出 ,字符串a和字符串b的两个比较是相等的。

第二对输出 object g = a 和object h = b , 在内存中两个不同的对象,所以在栈中的内容是不相同的,故不相等。而g.equals(h)用的是sting的equals()方法故相等(多态)。如果将字符串a和b作这样的修改: string a=“aa”; string b=“aa”; 则,g和h的两个比较都是相等的。这是因为系统并没有给字符串b分配内存,只是将“aa”指向了b.所以a和b指向的是同一个字符串(字符串在这种赋值的情况下做了内存的优化)。

对于p1和p2,也是内存中两个不同的对象,所以在内存中的地址肯定不相同,故p1==p2会返回false,又因为p1和p2又是对不同对象的引用,所以p1.equals(p2)将返回false.

对于p3和p4,p4=p3,p3将对对象的引用赋给了p4,p3和p4是对同一个对象的引用,所以两个比较都返回true.

中就有介绍啊:

下面的规则概括了 Equals 方法和等号运算符 (==) 的实现准则:

每次实现 Equals 方法时都实现 GetHashCode 方法。这可以使 Equals 和 GetHashCode 保持同步。 每次实现相等运算符 (==) 时,都重写 Equals 方法,使它们执行同样的操作。这样,使用 Equals 方法的基础结构代码(如 Hashtable 和 ArrayList)的行为就与用相等运算符编写的用户代码相同。 每次实现 IComparable 时都要重写 Equals 方法。 实现 IComparable 时,应考虑实现相等 (==)、不相等 (!=)、小于 ( <) 和大于 (>) 运算符的运算符重载。 不要在 Equals、GetHashCode 方法或相等运算符 (==) 中引发异常。 有关 Equals 方法的相关信息,请参见实现 Equals 方法。 在值类型中实现相等运算符 (==) 大多数编程语言中都没有用于值类型的默认相等运算符 (==) 实现。因此,只要相等有意义就应该重载相等运算符 (==)。 应考虑在值类型中实现 Equals 方法,这是因为 System::.ValueType 的默认实现和自定义实现都不会执行。 每次重写 Equals 方法时都实现相等运算符 (==)。 在引用类型中实现相等运算符 (==) 大多数语言确实为引用类型提供默认的相等运算符 (==) 实现。因此,在引用类型中实现相等运算符 (==) 时应小心。大多数引用类型(即使是实现 Equals 方法的引用类型)都不应重写相等运算符 (==)。 如果类型是 Point、String、BigNumber 等基类型,则应重写相等运算符 (==)。每当考虑重载加法 (+) 和减法 (-) 运算符时,也应该考虑重载相等运算符 (==)。

时间: 2024-10-10 14:18:24

C#中 ==与equals的区别的相关文章

java中==与equals的区别

值类型是存储在内存中的堆栈(以后简称栈),而引用类型的变量在栈中仅仅是存储引用类型变量的地址,而其本身则存储在堆中. ==操作比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量在堆中存储的地址是否相同,即栈中的内容是否相同. equals操作表示的两个变量是否是对同一个对象的引用,即堆中的内容是否相同. ==比较的是2个对象的地址,而equals比较的是2个对象的内容. 显然,当equals为true时,==不一定为true: 一.String中的equals和== 1. public

java 中 “==” 和 equals 的区别

在初学Java时,可能会经常碰到下面的代码: 1 String str1 = new String("hello"); 2 String str2 = new String("hello"); 3 4 System.out.println(str1==str2); 5 System.out.println(str1.equals(str2)); 为什么第4行和第5行的输出结果不一样?==和equals方法之间的区别是什么?如果在初学Java的时候这个问题不弄清楚,就

java中==和equals的区别详解

分析前基础了解: 一)JVM把内存划分成两种:一种是栈内存,一种是堆内存. ①在函数中定义的一些基本类型的变量和对象的引用变量(变量名)都在函数的栈内存中分配. ②当在一段代码块定义一个变量时,Java就在栈中为这个变量分配内存空间,当超过变量的作用域后,Java会自动释放掉为该变量所分配的内存空间,该内存空间可以立即被另作他用. ③堆内存用来存放由new创建的对象(包括由基本类型包装起来的类:Integer.String.Double,实际上每个基本类型都有他的包装类)和数组. 二)Objec

Java中==和equals的区别,equals和hashCode的区别

在java中: ==是运算符,用于比较两个变量是否相等. equals,是Objec类的方法,用于比较两个对象是否相等,默认Object类的equals方法是比较两个对象的地址,跟==的结果一样.Object的equals方法如下: public boolean equals(Object obj) { return (this == obj); } hashCode也是Object类的一个方法.返回一个离散的int型整数.在集合类操作中使用,为了提高查询速度.(HashMap,HashSet等

java中== 和 .equals()的区别

在java中,我们用来比较两个数据类型是否一样的时候,通常会用到 == 和 .equals(),然而这两者有什么区别呢.下面我们通过一些例子来看一下. package test; public class test { public static void main(String[] args) { // TODO Auto-generated method stub String a = "1"; String b = "1"; //int a = 1; //in

java 中 ==和equals 的区别

Java中equals和==的区别 java中的数据类型,可分为两类: 1.基本数据类型,也称原始数据类型.byte,short,char,int,long,float,double,boolean 他们之间的比较,应用双等号(==),比较的是他们的值. 2.复合数据类型(类) 当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false. JAVA当中所有的类都是继承于Object这个基类的,在Ob

[转]Java中==和equals的区别,equals和hashCode的区别

在java中: ==是运算符,用于比较两个变量是否相等. equals,是Objec类的方法,用于比较两个对象是否相等,默认Object类的equals方法是比较两个对象的地址,跟==的结果一样.Object的equals方法如下: [java] view plain copy public boolean equals(Object obj) { return (this == obj); } hashCode也是Object类的一个方法.返回一个离散的int型整数.在集合类操作中使用,为了提

C# 中 ==和equals的区别

不想说太多,直接上代码,这两个就没什么联系,有自己独立的规则.比较其实不利于记忆. 下面是测试代码 Console.WriteLine("--equals和==的区别--"); Console.WriteLine("1.对于值类型的数据"); Console.WriteLine(); Console.WriteLine("1.1.类型和数学上的值都相同的数"); Console.WriteLine($"(int)1 == (int)1

Java中==与equals的区别及理解

区别: “==” 比较的是两个引用在内存中指向的是不是同一对象(即同一内存空间),也就是说在内存空间中的存储位置是否一致.(引用类型) 如果两个对象的引用相同时(指向同一对象时),“==”操作符返回true,否则返回flase. 注:如果有对内存分配及变量存储位置(堆.栈.方法区常量池.方法区静态区)感兴趣的可以去看看这篇博客,里面写的很详细.对我还在学基础的人来说帮助很大,理解了很多内容,还有待消化. equals方法是由Object类提供的,可以由子类来进行重写 Object类默认的实现如下