C#构造函数问题,重载,重写,覆盖的区别

在C#当中,程序员写的每一个类在没写其它构造函数的情况下,都会默认有一个无参构造函数。

代码如下:

 class DemoClass
    {
        public string Name { set; get; }

    }

  DemoClass demoOb = new DemoClass();

如果程序员写了其它的构造函数,那类本身默认的无参构造函数会自动消失,所以更谈不上构造函数重载。

代码如下:

 class DemoClass
    {
        public string Name { set; get; }
        //添加一个string类型参数列表的构造函数
         public DemoClass(string strName)
        {
            this.Name = strName;
        }
    }

DemoClass Demoob1 = new DemoClass("小明");//正确

DemoClass demoOb2= new DemoClass();//错误,因为原来那个默认无参构造函数已经消失

这个时候,如果还想使用无参构造函数,必须重新写一个无参构造函数

也就是所谓的函数重载问题了,在这儿一定要记住,函数重载依靠的是参数类型,不是参数名

代码如下:

    class DemoClass
    {
        public string Name { set; get; }
        //添加一个string类型参数列表的构造函数
         public DemoClass(string strName)
        {
            this.Name = strName;
        }
        //无参构造函数
         public DemoClass()
         {
             Console.WriteLine("I am no Parameter Fun");
         }
         //再添加一个构造函数演示重载
         public DemoClass(string strName1, string strName2)
         {
             Console.WriteLine("Two Parameters");
         }
    }

 DemoClass Demoob1 = new DemoClass("小明");//正确
 DemoClass demoOb2 = new DemoClass();//正确
 DemoClass demoOb3 = new DemoClass("小刘", "小李");//正确

现在讲继承当中的构造函数问题:

首先:要理解这是一个递归的过程,子类会调用父类,父类也会根据情况调用爷爷类

在不做其它操作的情况下,子类的任何不同的构造函数都会先而且必须调用父类的无参构造函数,再调用实例化中所使用的本类构造函数,如果父类之上还有爷爷类,那么就是调用顺序:爷爷类->父类->本类

代码如下:

 class GrandParentClass
    {
        protected string name;
        public string Name
        {
            get { return name; }
            set { this.name = value; }
        }

        public GrandParentClass(string strName)
        {
            Console.WriteLine("GrandParentClass One parameter Constructor");
        }
        public GrandParentClass()
        {
            Console.WriteLine("GrandClass No parameter Constructor");
        }
    }
    class FatherClass :GrandParentClass
    {

        public FatherClass()
        {
            Console.WriteLine("ParentClass No parameter Constructor");
        }
        public FatherClass(string strName)
        {
            Console.WriteLine("FatherClass One parameter Constructor");
        }
    }
    class ChildClass : FatherClass
    {
        public ChildClass()
        {
            Console.WriteLine("ChildClass No parameter Constructor");
        }
        public ChildClass(string strName)
        {
            Console.WriteLine("ChildClass One parameter Constructor");
        }
    }

//会依次调用爷爷类,父亲类的无参构造函数 最后调用实例化中使用的本类的构造函数
 ChildClass childOb = new ChildClass();
 Console.WriteLine("******************************");
 //会依次调用爷爷类,父亲类的无参构造函数 最后调用实例化中使用的本类的构造函数
 ChildClass childOb2 = new ChildClass("wang");

运行结果如下:

记住,不管使用的什么构造函数,在不做处理的情况下,都会首先调用父类的无参构造函数,如果父类没有无参构造函数,那就会无法运行。

    class GrandParentClass
    {
        protected string name;
        public string Name
        {
            get { return name; }
            set { this.name = value; }
        }

        public GrandParentClass(string strName)
        {
            Console.WriteLine("GrandParentClass One parameter Constructor");
        }
        public GrandParentClass()
        {
            Console.WriteLine("GrandClass No parameter Constructor");
        }
    }
    class FatherClass :GrandParentClass
    {
        ///去掉了无参构造函数
        //public FatherClass()
        //{
        //    Console.WriteLine("ParentClass No parameter Constructor");
        //}

        public FatherClass(string strName)
        {
            Console.WriteLine("FatherClass One parameter Constructor");
        }
    }
    class ChildClass : FatherClass
    {
        public ChildClass()
        {
            Console.WriteLine("ChildClass No parameter Constructor");
        }
        public ChildClass(string strName)
        {
            Console.WriteLine("ChildClass One parameter Constructor");
        }
    }

这样就会提示:错误  “OverNewDemo.FatherClass”不包含采用“0”个参数的构造函数 E:\OverNewDemo\OverNewDemo\Program.cs 73 16 OverNewDemo

当然:如果我们把另外那个构造函数注释掉了,就可以运行,因为FatherClass里面就没有写任何构造函数了,它会默认有一个无参构造函数。

那么如果我们真的就是想在子类当中调父类的其它构造函数呢?

方法如下:

只需要在子类构造函数后面加上 “base:(实参数)”   这样就会先调用父类对应的构造函数,括号里的实参数就会传递给父类的构造函数参数。

代码如下:

 class GrandParentClass
    {
        protected string name;
        public string Name
        {
            get { return name; }
            set { this.name = value; }
        }

        public GrandParentClass(string strName)
        {
            Console.WriteLine("GrandParentClass One parameter Constructor");
        }
        public GrandParentClass()
        {
            Console.WriteLine("GrandClass No parameter Constructor");
        }
    }
    class FatherClass :GrandParentClass
    {

        public FatherClass()
        {
            Console.WriteLine("ParentClass No parameter Constructor");
        }

        public FatherClass(string strName)
        {
            Console.WriteLine("FatherClass One parameter Constructor");
        }
    }
    class ChildClass : FatherClass
    {
        public ChildClass()
        {
            Console.WriteLine("ChildClass No parameter Constructor");
        }
        public ChildClass(string strName):base(strName)
        {
            Console.WriteLine("ChildClass One parameter Constructor");
        }
    }

  ChildClass childOb2 = new ChildClass("wang");

这样就成功了,大家可以看到,子类的确调用了父类的有参构造函数,但是爷爷类还是调用的无参构造函数,这是因为这是一个递归的过程,父类的有参构造函数在不加 base的处理下,还是会调用爷爷类的有参构造函数,递归!!!这个时候把父类的无参构造函数注释掉也不会报错,因为它不会去调用,也不谈出错。

下面总结重载,重写和覆盖

1.重载很简单,就是多种不同参数列表甚至不同返回类型的同名函数,调用时依靠参数列表和参数类型进行区别, 需要注意:如果出现了相同参数列表但不同返回类型的方法,就会出错,因为不知道调用哪一个方法。

2. C# override重写,是指对父类中的虚方法(标记virtual)或抽象方法(标记为abstract)进行重写,实现新的功能,它必须与父类方法的签名完全一致,而且与父类方法的可访问性也必须一致.

注意:父类 对象名=new 父类名(); 仍然会使用原来的方法、

父类 对象名=new  子类名(); 会使用新方法

可以间接认为改动了父类方法

3.new隐藏(也叫做覆盖),是指在子类中重新定义一个签名与父类的方法相同的方法,这个方法可以不用new修饰,只是编译时会弹出一个警告信息:如果是有意隐藏,请使用关键字new。

注意:

父类 对象名=new 父类名(); 仍然会使用原来的方法、

父类 对象名=new  子类名(); 仍然会使用原来的方法、

代码如下:

 class Program
    {
        static void Main(string[] args)
        {
            FatherClassd fathOb1= new FatherClassd();
            fathOb1.Method1();//输出fath

            FatherClassd fathOb2 = new ChildClass();
            fathOb2.Method1();//输出child1

            FatherClassd fathOb3 = new ChildClass2();
            fathOb3.Method1();//输出fath

            ChildClass child1 = new ChildClass();
            child1.Method1();//输出child1

            ChildClass2 child2 = new ChildClass2();
            child2.Method1();//输出child2
            Console.ReadKey();
        }
    }
    class  FatherClassd
    {
        public virtual void Method1()
        {
            Console.WriteLine("fath");
        }
    }
    class ChildClass:FatherClassd
    {
        //重写
        public override void Method1()
        {
            Console.WriteLine("child1");
        }
    }
    class ChildClass2 : FatherClassd
    {
        //覆盖或者叫做隐藏
        public new void Method1()
        {
            Console.WriteLine("child2");
        }
    }
时间: 2024-12-28 16:00:13

C#构造函数问题,重载,重写,覆盖的区别的相关文章

重载与覆盖的区别

对于在同一个可访问区内被声明的几个具有不同参数列(参数的类型.个数.顺序不同)的同名函数,程序会根据不同的参数列表来确定具体调用哪个函数,这种机制叫重载(overload),overlaod不关心函数的返回值. 覆盖(override)是指派生类中存在重新定义的函数,其函数名.参数列.返回值类型必须同父类的相对应被覆盖的函数严格一致,覆盖函数和被覆盖函数只有函数体(花括号中的部分)不同,当派生类对象调用子类中同名函数时,会自动调用子类中的覆盖版本,而不是父类中的被覆盖版本. 成员函数被重载(ov

java的重写、重载、覆盖的区别

多态性 通过继承,一个类可以用作多种类型:可以用作它自己的类型.任何基类型,或者在实现接口时用作任何接口类型.这称为多态性 重载 每个类型成员都有一个唯一的签名.方法签名由方法名称和一个参数列表(方法的参数的顺序和类型)组成.只要签名不同,就可以在一种类型内定义具有相同名称的多种方法.当定义两种或多种具有相同名称的方法时,就称作重载.即重载时相同名称成员的参数列表是不相同的(参数顺序和类型). 继承,重写和隐藏成员 (重写 = 覆盖) 派生的类型继承其基本类型的所有成员:也就是说,这些成员是在派

类成员函数中重载/重写(覆盖)/重定义(隐藏)的区别

https://blog.csdn.net/ranxiaoxu123/article/details/52291385 https://www.cnblogs.com/renyuan/archive/2013/06/08/3127562.html 原文地址:https://www.cnblogs.com/Kiven5197/p/8934207.html

C++重载重写和多态区别

1.重载和重写的区别: 重载overload:在同一个类中,函数名相同,参数列表不同,编译器会根据这些函数的不同参数列表,将同名的函数名称做修饰,从而生成一些不同名称的预处理函数,未体现多态. 重写override:也叫覆盖,子类重新定义父类中有相同名称相同参数的虚函数,主要是在继承关系中出现的,被重写的函数必须是virtual的,重写函数的访问修饰符可以不同,尽管virtual是private的,子类中重写函数改为public,protected也可以,体现了多态. 重定义redefining

C++ Primer 学习笔记_28_操作符重载与转换(3)--成员函数的重载、覆盖与隐藏、类型转换运算符、*运算符重载、->运算符重载

C++ Primer 学习笔记_28_操作符重载与转换(3)--成员函数的重载.覆盖与隐藏.类型转换运算符.*运算符重载.->运算符重载 一.成员函数的重载.覆盖与隐藏 对于类层次的同名成员函数来说,有三种关系:重载.覆盖和隐藏,理清3种关系,有助于写出高质量的代码. 1.成员函数的重载 重载的概念相对简单,只有在同一类定义中的同名成员函数才存在重载关系,主要特点时函数的参数类型和数目有所不同:但不能出现函数参数的个数和类型均相同,仅仅依靠返回值类型不同来区分的函数,这和普通函数的重载是完全一致

重载和覆盖

说到java中的重载和覆盖呢,大家都很熟悉了吧,但是呢我今天就要写这个. 本文主题: 一.什么是重载 二.什么是覆盖 三.两者之间的区别 重载(overload): 在一个类中,如果出现了两个或者两个以上的同名函数,只要它们的参数的个数,或者参数的类型不同,即可称之为该函数重载了. 即当函数同名时,只看参数列表.和返回值类型没关系. 重载使用的时候需要注意: 1.在使用重载时只能通过不同的参数样式.例如,不同的参数类型,不同的参数个数,不同的参数顺序. 2.方法的异常类型和数目不会对重载造成影响

C++:重载、覆盖与隐藏

C++之中的重载.覆盖.隐藏 重载 覆盖 重载与覆盖的区别 相关代码 隐藏 重载 重载是指函数不同的参数表,对同名函数的名称做修饰,然后这些同名函数就成了不同的函数.在同一可访问区域内被声明的几个具有不同参数列表(参数类型.个数.顺序不同)的同名函数,程序会根据不同的参数列来确定具体调用哪个函数.对于重载函数的调用,在编译期间就已经确定,是静态的,它们的地址在编译期间就绑定了与多态无关. 注意:重载不关心函数的返回值类型 特征: (1)相同的范围(在同一个类中) (2)函数名字相同 (3)参数不

Java基础知识(重载和覆盖)

重载(overload): 在一个类中,如果出现了两个或者两个以上的同名函数,只要它们的参数的个数,或者参数的类型不同,即可称之为该函数重载了. 即当函数同名时,只看参数列表.和返回值类型没关系. 重载使用的时候需要注意: 1.在使用重载时只能通过不同的参数样式.例如,不同的参数类型,不同的参数个数,不同的参数顺序. 2.方法的异常类型和数目不会对重载造成影响. 3.对于父类来说,子类可以重载与继承父类中的方法同名的方法,如果某一方法在父类中是访问权限是priavte,那么就不能在子类对其进行重

重载-重写(覆盖)-隐藏的区别

重载和重写搞的差不多了,隐藏有有点混乱了,在此记一下,留作后看 成员函数被重载的特征(1)相同的范围(在同一个类中):(2)函数名字相同:(3)参数不同:(4)virtual 关键字可有可无.重写(覆盖)是指派生类函数覆盖基类函数,--这是c++多态的标志,虚函数赋零则变成虚基类(纯粹的接口申明类),特征是(1)不同的范围(分别位于派生类与基类):(2)函数名字相同:(3)参数相同:(4)基类函数必须有virtual 关键字.“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,规则如下(1)如果派