引用相等

介绍

在 C# 中可以通过 object.ReferenceEquals(obj1, obj2) 方法来判断两个变量引用的是不是同一个地址,如果是,那么就是引用相等。

引用相等是针对引用类型变量来说的,因为值类型变量存储在栈内存,不存在引用情况。

普通引用类型

1、有如下实体类:

class Person
{
    private int id;
    private string name;

    public int Id
    {
        get { return id; }
        set { id = value; }
    }

    public string Name
    {
        get { return name; }
        set { name = value; }
    }
}

2、编写如下代码运行:

var p1 = new Person();
p1.Id = ;
p1.Name = "张三";

var p2 = p1;

var p3 = new Person();
p3.Id = ;
p3.Name = "张三";

Console.WriteLine(object.ReferenceEquals(p1,p2)); // true
Console.WriteLine(object.ReferenceEquals(p1,p3)); // false
Console.WriteLine(p1 == p2); // true
Console.WriteLine(p1 == p3); // false

可以看到,类的实例用 obj1 == obj2 比较与 object.ReferenceEquals(obj1, obj2) 的结果是一样的。

结论:对于普通引用类型来使用 == 来比较两个引用类型变量也是比较它们是不是引用同一个地址。

不一样的引用类型-string

为什么说 string 不一样呢?看如下示例:

 string s1 = "hello";
 string s2 = "hello";
 string s3 = new string(new char[] {'h', 'e', 'l', 'l', 'o'});
 Console.WriteLine(object.ReferenceEquals(s1,s2)); // true
 Console.WriteLine(object.ReferenceEquals(s1,s3)); // false
 Console.WriteLine(s1 == s2); // true
 Console.WriteLine(s1 == s3); //true

由于字符串拘留池的原因, s1 和 s2 肯定是引用同一个地址的,所以第 4 行和第 6 行结果都为 true 。

再看第 5 行和第 7 行,因为 s3 是手动 new 的一个对象,所以 s1 与 s3 肯定不是同一个引用,所以第 5 行结果为 false 。但是再看第 7 行, s1 == s3 的结果为 true 。

结论:对于 string 类型,使用 == 比较的是字符串的实际内容,而不是它们的引用地址。可以理解为, string 类对 == 运算符方法进行了重载。

字符串拘留池:也叫字符串暂存池、缓冲池。字符串是引用类型,程序中常会存在大量的字符串对象,如果每次都创建一个字符串对象,会比较浪费内存、性能低下,因此 CLR 做了字符串拘留池,在一些情况下对于字符串对象进行了重用。

相关面试题

下面的代码一共创建了几个字符串对象?

string s1 = "hello";
string s2 = "hello";
string s3 = new string(new char[] {'h', 'e', 'l', 'l', 'o'});

答案:创建了两个对象, s1 和 s2 引用的是同一个对象 "hello" ,另一个是 s3 指向的对象。

运算符重载

介绍

C# 允许用户定义的类型通过使用 operator 关键字定义静态成员函数来重载运算符。注意必须用 public 修饰且必须是类的静态的方法。但并非所有内置运算符都可以被重载,详见下表:

运算符 可重载性
 +、-、!、~、++、--、true、false  可以重载这些一元运算符, true和false运算符必须成对重载
 +、-、*、/、%、&、|、^、<<、>>  可以重载这些二元运算符
 ==、!=、<、>、<=、>=  可以重载比较运算符,必须成对重载
 &&、||  不能重载条件逻辑运算符,但可以使用能够重载的&和|进行计算
 []  不能重载数组索引运算符,但可以定义索引器
 ()  不能重载转换运算符,但可以定义新的转换运算符(请参见 explicit 和 implicit)
 +=、-=、*=、/=、%=、&=、|=、^=、<<=、>>=  不能显式重载赋值运算符,在重写单个运算符如+、-、%时,它们会被  隐式重写
 =、.、?:、->、new、is、sizeof、typeof  不能重载这些运算符

示例

 internal class Program
 {
     public static void Main(string[] args)
     {
         var p1 = new Person();
         p1.Id = ;
         p1.Name = "张三";

         var p2 = p1;

         var p3 = new Person();
         p3.Id = ;
         p3.Name = "张三";

         Console.WriteLine(object.ReferenceEquals(p1,p2)); // true
         Console.WriteLine(object.ReferenceEquals(p1,p3)); // false
         Console.WriteLine(p1 == p2); // true
         Console.WriteLine(p1 == p3); // true
     }

     class Person
     {
         // == 与 != 必须成对重载
         public static bool operator ==(Person left, Person right)
         {
             return left.Id == right.Id && left.Name == right.Name;
         }

         public static bool operator !=(Person left, Person right)
         {
             return left.Id != right.Id || left.Name != right.Name;
         }

         private int id;
         private string name;

         public int Id
         {
             get { return id; }
             set { id = value; }
         }

         public string Name
         {
             get { return name; }
             set { name = value; }
         }
     }
 }

可以看到第 16 行的结果为 false ,说明 p1 和 p3 引用的不是同一个对象。而第 18 行的执行结果为 true ,说明此时通过 == 比较是执行了我们重载的运算符方法。

更多可参考微软官方文档

相关面试题

在 C# 中, obj1 == obj2 的比较结果是否与 obj1.Equals(obj2) 相同?

答案:不一定,因为 == 比较实际上是执行默认的运算符方法,除非 obj1 和 obj2 所属类型的 == 运算符方法和 Equals 方法的比较规则相同它们的结果才相同。

C#基础加强(6)之引用相等与运算符重载的更多相关文章

  1. 《挑战30天C++入门极限》C++运算符重载函数基础及其值返回状态

        C++运算符重载函数基础及其值返回状态 运算符重载是C++的重要组成部分,它可以让程序更加的简单易懂,简单的运算符使用可以使复杂函数的理解更直观. 对于普通对象来说我们很自然的会频繁使用算数运 ...

  2. (转)2.4.1 基础知识--添加服务引用与Web引用的区别

    <Web服务开发学习实录>第2章构建ASP.NET Web服务,本章我们将学习创建Web服务的各种方法,并重点对使用Visual Studio创建ASP.NET Web服务和修改Web服务 ...

  3. 小猪猪C++笔记基础篇(六)参数传递、函数重载、函数指针、调试帮助

    小猪猪C++笔记基础篇(六) ————参数传递.函数重载.函数指针.调试帮助 关键词:参数传递.函数重载.函数指针.调试帮助 因为一些事情以及自己的懒惰,大概有一个星期没有继续读书了,已经不行了,赶紧 ...

  4. C++学习6-面向对象编程基础(运算符重载、类的派生与继承、命名空间)

    运算符重载 重载的运算符是具有特殊名字的函数:它们的名字由关键字operator和其后要定义的运算符号共同组成.重载的运算符是遵循函数重载的选择原则,根据不同类型或不同参数来选择不同的重载运算符. 运 ...

  5. C++基础学习笔记----第十三课(操作符重载-下)

    本节主要讲使用成员函数重载操作符,包括[],=,(),->四种操作符的重载以及&&和||的问题. 类的成员函数进行操作符重载 基本概念 类的成员函数也可以进行操作符的重载.类的普 ...

  6. C++_基础_运算符重载

    内容: (1)输入输出运算符重载 (2)友元类和友元函数 (3)双目运算符重载 (4)单目运算符重载 (5)不能被重载的运算符 (6)只能定义为成员形式的运算符 1.输入输出运算符重载如: int n ...

  7. C++基础——运算符重载友元函数示例

    一.前言 其实本人学习C++的目的,只是为了体会OOP设计思想,并为利用System Verilog验证复杂设计做准备.如果想要真正做点软件方面项目级的东西,还需要掌握其他高级语言和库.框架等知识.因 ...

  8. C++ 基础语法 快速复习笔记(3)---重载函数,多态,虚函数

    1.重载运算符和重载函数: C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载. 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它 ...

  9. 前端学习笔记--js概述与基础语法、变量、数据类型、运算符与表达式

    本篇记录js的概述与基础语法.变量.数据类型.运算符与表达式 1.概述与基础语法 2.变量 举例: 3.数据类型 4.运算符与表达式

随机推荐

  1. Python中的 @staticmethod@classmethod方法

    python类中有三种方法,常见的是实例方法,另外两种是staticmethod装饰的静态方法,和classmethod装饰的类方法. 1.对比 流畅的python里,用一个例子进行了对比: (1)两 ...

  2. openwrt MT7628 编译前更改为DHCP,root 密码、ssid、时区、主机名

    一.设置为DHCP动态获取ip地址 在:/home/OpenWrt/openwrt_CC_mt76xx_zhuotk_source/ 目录下,新建文件名/files/etc/config. 将配置好的 ...

  3. 《剑指offer》数组中出现一半次数的数字

    本题来自<剑指offer> 反转链表 题目: 思路: C++ Code: Python Code: 总结:

  4. 倒影问题(reflect:below)

    这个例子灵感来源于实现一个登录框下方的倒影: .box { width: 300px; height: 200px; border: 1px solid #1f637b; -webkit-box-re ...

  5. 详解 CAP 定理 Consistency(一致性)、 Availability(可用性)、Partition tolerance(分区容错性)

    CAP原则又称CAP定理,指的是在一个分布式系统中,Consistency(一致性). Availability(可用性).Partition tolerance(分区容错性),三者不可得兼. 分布式 ...

  6. Python2还是Python3

    Python2还是Python3 相信很多新接触Python的人都会纠结这一个问题,学Python2还是Python3? 不像Java一样每个新版本基本都是基本兼容以前的版本的.Python2和Pyt ...

  7. 课堂小记---JavaScript(1)

    day01 1.数据类型  number string boolean undefined object function 加号具有两种功能,数字相加 和 字符串拼接.加号两边只要碰见字符串,则执行字 ...

  8. Innodb与Myisam引擎的区别与应用场景

    1. 区别: (1)事务处理: MyISAM是非事务安全型的,而InnoDB是事务安全型的(支持事务处理等高级处理): (2)锁机制不同: MyISAM是表级锁,而InnoDB是行级锁: (3)sel ...

  9. CommonJs规范详解---【XUEBIG】

     CommonJS是服务器模块的规范,Node.js采用了这个规范   1.CommonJs规范的出发点:JS没有模块系统.标准库较少.缺乏包管理工具:为了让JS可以在任何地方运行,以达到Java.C ...

  10. flash上传头像,截取图像 组件演示

    效果图如下: HTML页面代码: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http:// ...