C#基础知识整理:基础知识(12) 超类Object

    面向对象三大特性:封装,继承,多态。那么类是从哪里继承呢?在面向对象语言中有基类或者也叫做超类的概念,也就是所有类都是从这个类继承得来的,这个超类叫object。.net中是这样描述object类的:
    支持 .net framework 类层次结构中的所有类,并为派生类提供低级别服务。这是 .net framework 中所有类的最终基类;它是类型层次结构的根。
既然是超类,object定义了一些关键的方法。如下:

Equals方法——用于比较两个实例是否相等。
public virtual bool Equals(Object obj),比较当前实例是否与obj相等;
public static bool Equals(Object objA,Object objB),比较指定两个实例是否相等。

Finalize 方法——允许 Object 在“垃圾回收”回收 Object 之前尝试释放资源并执行其他清理操作。

GetHashCode 方法——获取一个对象的Hash值。

GetType方法——获取当前实例的Type。

MemberwiseClone 方法——创建当前实例的浅表副本,也就是如果当前实例有值,新创建的实例中只获取值类型的值,引用类型是没有获取值。

ReferenceEquals 方法——比较两个实例是否相同,与static bool Equals(Object objA,Object objB)用法一样。

ToString 方法——这个平时用的比较多,用于返回当前实例的string。
Object是超类,所以C#中的所有类都具有这些方法。

下面着重介绍下Equals和ToString方法。
一、对象比较
C#中有值类型和引用类型,简单的理解就是值类型保存对象的值,而引用类型是实例的引用,类似于C语言的指针。因此在使用对象的比较的时候,值类型比较两个对象值是否相等,引用类型比较指定的引用是否引用同一个对象。当然有时也会比较引用类型指向的实例值是否相同。
下面是对象比较的代码:

using System;using System.Collections.Generic;using System.Text;namespace YYS.CSharpStudy.MainConsole.AboutObject{    public class Int32Value : ICloneable    {        //字段,定义字段最好赋上初值        private int intValue = 0;        ///         /// 属性        ///         public int IntValue        {            get            {                return this.intValue;            }            set            {                this.intValue = value;            }        }        ///         /// 定义一个无参构造器        ///         public Int32Value() { }        ///         /// 带参数的构造器        ///         public Int32Value(int value)        {            this.intValue = value;        }        /////         ///// 实现ICloneable接口        /////         public object Clone()        {            return this.MemberwiseClone();        }    }}

调用:

using System;using YYS.CSharpStudy.MainConsole.AboutObject;namespace YYS.CSharpStudy.MainConsole{    class Program    {        static void Main(string[] args)        {            //声明一个Int32Value类型的引用value1,指向一个实例            Int32Value value1 = new Int32Value(30);            //声明value2,指向value1,此时两个引用是指向一个实例的            Int32Value value2 = value1;            //使用==比较            Console.WriteLine(string.Format("value1和value2 {0}", value1 == value2 ? "相同" : "不相同"));//相同            //调用Clone,复制一个value1的副本,赋值给value2            //此时是两个实例了            value2 = (Int32Value)value1.Clone();            //使用==比较            Console.WriteLine(string.Format("value1和value2 {0}", value1 == value2 ? "相同" : "不相同"));//不相同            //将value1赋给value2,此时他们指向同一个实例            value2 = value1;            //使用Equals比较            Console.WriteLine(string.Format("value1和value2 {0}", value1.Equals(value2) ? "相同" : "不相同"));//相同            //调用Clone,复制一个value1的副本,赋值给value2            //此时是两个实例了            value2 = (Int32Value)value1.Clone();            //使用Equals比较            Console.WriteLine(string.Format("value1和value2 {0}", value1.Equals(value2) ? "相同" : "不相同"));//不相同            Console.ReadLine();        }    }}

结果:
C#基础知识整理:基础知识(12) 超类Object

由代码可以看出:

a、对于引用类型,=运算符将引用从一个变量传递到另一个变量,所以=两边的变量引用同一个对象;
 
b、对于引用相同的两个变量,使用==运算符返回true;

c、对象的Clone方法产生一个新实例,返回该实例的引用,该实例的所有字段值和原对象相同,即Clone方法得到对象的一个副本。从Object类继承下来的保护方法MemberwiseClone返回当前对象的副本;

d、对于Clone方法返回的对象,其引用和原对象不同,==运算符返回false;

e、从Object继承下来的Equals方法结果和==运算符相同,只是比较当前引用(this)和参数保存的引用是否引用到了同一个对象上。
在上述代码的基础上,重写Equals方法。

using System;using System.Collections.Generic;using System.Text;namespace YYS.CSharpStudy.MainConsole.AboutObject{    public class Int32Value : ICloneable    {        //字段,定义字段最好赋上初值        private int intValue = 0;        ///         /// 属性        ///         public int IntValue        {            get            {                return this.intValue;            }            set            {                this.intValue = value;            }        }        ///         /// 定义一个无参构造器        ///         public Int32Value() { }        ///         /// 带参数的构造器        ///         public Int32Value(int value)        {            this.intValue = value;        }        /////         ///// 实现ICloneable接口        /////         public object Clone()        {            return this.MemberwiseClone();        }        ///         /// 覆盖Equals方法        ///         ///         ///         public override bool Equals(object obj)        {            bool isEquals = Object.ReferenceEquals(obj, this);            if (isEquals == false)            {                Int32Value value = obj as Int32Value;                if (value != null)                {                    isEquals = value.intValue == this.intValue;                }            }            return isEquals;        }    }}

调用

using System;using YYS.CSharpStudy.MainConsole.AboutObject;namespace YYS.CSharpStudy.MainConsole{    class Program    {        static void Main(string[] args)        {            //声明一个Int32Value类型的引用value1,指向一个实例            Int32Value value1 = new Int32Value(30);            //声明value2,指向value1,此时两个引用是指向一个实例的            Int32Value value2 = value1;            //使用==比较            Console.WriteLine(string.Format("value1和value2 {0}", value1 == value2 ? "相同" : "不相同"));//相同            //调用Clone,复制一个value1的副本,赋值给value2            //此时是两个实例了            value2 = (Int32Value)value1.Clone();            //使用==比较            Console.WriteLine(string.Format("value1和value2 {0}", value1 == value2 ? "相同" : "不相同"));//不相同            //将value1赋给value2,此时他们指向同一个实例            value2 = value1;            //使用Equals比较            Console.WriteLine(string.Format("value1和value2 {0}", value1.Equals(value2) ? "相同" : "不相同"));//相同            //调用Clone,复制一个value1的副本,赋值给value2            //此时是两个实例了            value2 = (Int32Value)value1.Clone();            //使用Equals比较            Console.WriteLine(string.Format("value1和value2 {0}", value1.Equals(value2) ? "相同" : "不相同"));//相同            Console.ReadLine();        }    }}

结果:
C#基础知识整理:基础知识(12) 超类Object
修改后的代码:
覆盖了Equals方法,程序运行结果则不同了:Equals不再比较两个变量保存的对象引用是否相同,而是比较两个变量所引用的对象是否具有相同的属性值。
覆盖后的Equals执行流程是这样的:
我们看一下覆盖后Equals方法的具体执行流程:

a、使用Object静态方法ReferenceEquals比较参数obj和当前对象引用(this)是否相同,如果引用相同,则必为相同对象;
b、如果变量obj和当前引用不相同,则使用as运算符,尝试将obj类型转换和当前对象相同的类型,转换成功表示obj变量引用着和当前对象类相同的对象,as运算符转换后对象的引用,否则返回null,只有同类型的对象才需要比较,不同类型的对象必然是不同的对象;
c、如果as运算符转换对象成功,则进一步比较当前对象引用(this)和参数obj引用对象的字段值是否相等。
由上面两段代码可以看出:
==和Object类的Equals方法都是比较对象的引用是否相同,并不比较对象的字段是否相等;而Equals方法可以覆盖,以求让其对对象的字段进行等值比较或者其它的比较的需求。
因此,如果需要比较两个C#引用类型变量,除非确定需要比较的就是对象引用时才可以使用==运算符,否则应该使用对象的Equals方法,防止对象所属的类覆盖了这个方法。

    上面是引用类型的比较,相对来说,值类型的比较就比较纯粹。值类型是不存在引用的,所以值类型就是比较两个对象值是否相等。当然如果覆盖了Equals方法就除外了。不过一般情况下,我们无需覆盖Equals方法。对于引用类型来说,等值比较就是比较对象的引用,引用不同的两个对象应该就是不同的对象;对于值类型来说,比较的就是所有字段的值,字段值不同的两个对象是不同的对象。只有在特殊情况下,才需要覆盖Equals方法,已定义某个类特殊的对象比较方法。
对于覆盖Equals方法,要注意:
注意:理论上可以用任意代码覆盖Equals方法,但要保证如下原则:
a、Equals只能用于对象比较,不能做其它任何用途;
b、对于两个已存在的对象,在对象未作任何改变的情况下,无论任何时候调用Equals方法,返回的结果应该是一样的;
c、对于对象a和b,则a.Equalse(b)和b.Equals(a)返回的结果应该是一样的;
d、覆盖了Equals方法,同时要覆盖GetHashCode(方法),这个后面再详细说明。

二、ToString()
代码:

   public class Int32Value : ICloneable    {        //字段,定义字段最好赋上初值        private int intValue = 0;        ///         /// 属性        ///         public int IntValue        {            get            {                return this.intValue;            }            set            {                this.intValue = value;            }        }        ///         /// 定义一个无参构造器        ///         public Int32Value() { }        ///         /// 带参数的构造器        ///         public Int32Value(int value)        {            this.intValue = value;        }        /////         ///// 实现ICloneable接口        /////         public object Clone()        {            return this.MemberwiseClone();        }             ///         /// 重写ToString方法        ///         ///         public override string ToString()        {            return this.intValue.ToString();            //return "重写ToString";        }    }

 调用:

     class Program    {        static void Main(string[] args)        {            Int32Value value = new Int32Value(30);            Console.WriteLine(value.ToString());//30            //Console.WriteLine(value.ToString());//重写ToString            Console.ReadLine();        }    }

可以看出,ToString方法可以用任意代码覆盖,只要返回string即可,但是在覆盖时也要返回符合需求的返回值。

以上就是C#基础知识整理:基础知识(12) 超类Object 的内容,更多相关内容请关注PHP中文网(www.php.cn)!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1432201.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 06:21:43
下一篇 2025年12月13日 22:50:01

相关推荐

  • C#基础知识整理:基础知识(11) 值类型,引用类型

    c#是面向对象的语言,在面向对象的思想中,只有对象,所有事物都可以用类描述。所以比如这些,int,bool,char,string,double,long等都是类,那么像,30,2.5,”test”都是对应类的一个对象。 static void Main(string[] a…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:基础知识(13) 异常

        往往我们在写代码的时候,总会在运行中遇到某些问题而导致程序崩溃。这并不是编程人员的水平不行,而是由业务逻辑,操作系统,或者电脑等其它设备出现问题而造成,比如在c#中经常用到user32.dll里的一些方法,假如这个文件被删掉了,你的程序照样运行不了。当然作为一个有水平的程序员总会在写程序时是…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:基础知识(14) 数组

    无论哪种语言,肯定会有集合的概念。而最简单,最直观的集合应该就是数组了,数组是在内存中连续的一段空间。看看c#中数组 的定义。1、int[] intArry ; intArry= new int[6]; 这里声明了一个int数组类型变量intArry,并保存一个具有6个单元的int数组对象; int…

    2025年12月17日
    000
  • C#基础知识整理 基础知识(18) 值类型的装箱和拆箱(一)

    仔细了解装箱和拆箱其实是很有趣的,首先来看为什么会装箱和拆箱呢?看下面一段代码: class Program { static void Main(string[] args) { ArrayList array = new ArrayList(); Point p;//分配一个 for (int …

    好文分享 2025年12月17日
    000
  • C#基础知识整理:C#类和结构(3)

    1、静态类和静态成员有哪些功能特性? 实现代码?    静态类和静态成员是指使用static关键字定义的类或成员,凡是static类的成员一定是static成员,不然会报错的。静态类和成员的一大特点就是它们是独一无二的。如果是static类,那么它是不能实例化的,而且加载在内存中只有一个;如果是st…

    2025年12月17日
    000
  • C#基础知识整理:C#类和结构(4)

    1、什么是接口? 功能特性? 实现代码?    接口就是使用interface关键字定义的,由类的成员的组合组成的,描述一些功能的一组规范。在c#中可以看到,系统的一些接口都是这样命名的:icomparable(类型的比较方法)、icloneable(支持克隆)、idisposable(释放资源)等…

    2025年12月17日
    000
  • C#基础知识整理:基础知识(1) Main方法

    千里之行始于足下,掌握某一项技能,都是要从最基本的东西开始。温故而知新,一件小事当你在不同的境遇,不同的时间下去做,总能获得意想不到的体验和收获。那么首先就从main方法开始吧, //using关键字在该处的作用是引入一个“外部程序集”,System是微软提供的.net平台最基本程序集,所有.net…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:基础知识(2) 类

        类,是面向对象语言的基础。类的三大特性:封装、继承、多态。最基本的特性就是封装性。程序员用程序描述世界,将世界的所有事物都看成对象,怎么描述这个对象?那就是类了。也就是用类来封装对象。用书上的话说,类是具有相同属性和行为的对象的抽象。宝马汽车、别克汽车、五菱之光汽车… 基本具有相…

    2025年12月17日
    000
  • C#基础知识整理:基础知识(3) 类的构造

    我们定义了yschool,yteacher类,实例化对象时: YSchool shool1 = new YSchool(); shool1.ID = 1; shool1.Name = “清华附中”; YSchool school2 = new YSchool(); school2.ID = 2; s…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:基础知识(4) 继承

        前面提到过,面向对象的三大特性:封装性、继承性和多态性。封装在前面类的定义里也了解的差不多透彻了,现在看看继承的特性。继承实际是一个类对另一个类的扩展,后者称之为基类,前者称之为子类。继承就是子类拥有基类的一切属性和方法,子类还可以增加属性和方法。但是子类不能去掉父类的属性和方法。    当…

    2025年12月17日
    000
  • C# 快速排序

    c# 快速排序 using System;using System.Collections.Generic;using System.Linq;using System.Text;namespace Sort{ class QuickSorter { private static int[] myA…

    2025年12月17日
    000
  • C# 归并排序

     c# 归并排序 using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace Sort { class MergeSorter { /// /// 归并排序之归:归并排…

    2025年12月17日
    000
  • C# 插入排序

    c#  ,插入排序 using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace Sort { class InsertSorter { public static in…

    2025年12月17日
    000
  • C# 堆排序

    c#  堆排序 using System; using System.Collections; namespace Sort { public class HeapSorter { public static int[] Sort(int[] sortArray) { BuildMaxHeap(so…

    好文分享 2025年12月17日
    000
  • 鸡尾酒排序(双冒泡排序、搅拌排序或涟漪排序)

     鸡尾酒排序(双冒泡排序、搅拌排序或涟漪排序) using System; using System.Collections.Generic; using System.Linq; using System.Text;namespace Sort { class CockTailSorter { p…

    好文分享 2025年12月17日
    000
  • C# 冒泡排序

     c# 冒泡排序 using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace Sort { class BubbleSorter { public static int…

    好文分享 2025年12月17日
    000
  • 排序算法测试程序入口

     排序算法测试程序入口 using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Collections;using System.Diagnostics;namesp…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:.NET知识

    1、什么是.NET Framework    所谓.NET FrameWork就是一个平台,它的目的是为了跨操作系统编程。它包含了很多模块,例如有windows应用程序构件,Web开发的模块等,而不同的操作系统根据自己的特性,支持其中部分模块。NET框架是采用虚拟机运行的编程平台,以通用语言运行时(…

    好文分享 2025年12月17日
    000
  • C#数据类型

        c#的数据类型可以分为3类:数值类型,引用类型,指针类型。指针类型仅在不安全代码中使用。    值类型包括简单类型(如字符型,浮点型和整数型等),集合类型和结构型。引用类型包括类类型,接口类型,代表类型和数组类型。     值类型和引用类型的不同之处是值类型的变量值直接包含数据,而引用类型的…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:C#基础(1)

    c#基本结构 using System;using System.Collections.Generic;using System.Text;namespace ConsoleApplication1{ class MyFirstClass { static void Main(string[] a…

    好文分享 2025年12月17日
    000

发表回复

登录后才能评论
关注微信