How can I tell the inheriting class to not call its base class' parameter-less constructor?

前端 未结 7 2754
礼貌的吻别
礼貌的吻别 2021-02-19 23:32

I was surprised to find out that the parameter-less constructor of my base class is called any time I call any constructor in a derived class. I thought that is what

相关标签:
7条回答
  • 2021-02-20 00:13

    When you instantiate a class all the classes in the inheritance hierarchy are instantiated starting from the topmost Parent to the lowermost child, stopping at the type you instantiated. So if you instantiate System.Text.StringBuilder(), first you call the default constructor of System.Object, then StringBuilder(). You can use the base keyword to call a different constructor (one with equal or fewer params), but not more params. If you do not specify base, the default constructor is called, thus the reason this is happening to you. You can also use the base keyword within the context of an instance method to call base implementations, such as in the case of the decorator pattern. If you do not want to call the base class private constructors then you should set the children with private default constructors and also explicitly call base(params) on your other constructors.

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    
    namespace ConsoleApplication2
    {
        class ProcessCaller
        {
            static void Main()
            {
                MyDerived md1 = new MyDerived(1);
                object o = new System.Text.StringBuilder();//This will implicitly instantiate the classes in the inheritance hierarchy:  object, then stringbuilder
            }
        }
    
    public class MyBase
    {
       int num;
    
       public MyBase() 
       {
          Console.WriteLine("in MyBase()");
       }
    
       public MyBase(int i )
       {
          num = i;
          Console.WriteLine("in MyBase(int i)");
       }
    
       public virtual int GetNum()
       {
          return num;
       }
    }
    
    public class MyDerived: MyBase
    {
       //set private constructor.  base(i) here makes no sense cause you have no params
       private MyDerived()
       {
    
       }
    
        // Without specifying base(i), this constructor would call MyBase.MyBase()
       public MyDerived(int i) : base(i)
       {
       }
       public override int GetNum()
       {
           return base.GetNum();//here we use base within an instance method to call the base class implementation.  
       }
    }
    
    }
    
    0 讨论(0)
  • 2021-02-20 00:20

    You could make the default base constructor protected, then have only non-default constructors for the base and its child.

    edit

    You could give the base a protected constructor with a different signature (such as a protected enum type), and put your initialization code in there, while the default constructor, also protected, does not particular initialization.

    0 讨论(0)
  • 2021-02-20 00:28

    The only way is to explicitly tell it which other base ctor you want it to call; which of course means you must choose some base ctor to call.

    You can't have it call no base ctor at all - conceptually, constructing a Customer is done by first constructing a Person, and then doing Customer specific construction on top of it. For example, suppose Person had private fields - how would these be correctly constructed if construction of a Customer was allowed to not first construct a Person?

    0 讨论(0)
  • As others have pointed out, a derived instance must call call one of its base class' constructors.

    If you want to control the execution of a base class' initialization logic, remove its default constructor and replace it with something like this:

    public class Base {
        // Base has no default constructor
        public Base(bool initialize) {
            if (initialize) {
                // ... logic here 
            }
        }    
    }
    

    And the derived constructors look like this:

    // Derived1 executes the initialization logic
    public Derived1(): base(true) {}
    
    // Derived2 does not
    public Derived2(): base(false) {}
    
    0 讨论(0)
  • 2021-02-20 00:31

    In .NET, every object constructor in an object hierarchy will be called regardless if you call :base() or not.

    :base() is implicitly called if you don't explicitly call it.

    :base() can be used if you want to call a different contructor on a parent object rather than the default constructor.

    If you have code in the parent constructor that should not be called everytime, it may be better to move that code to it's own method that will need to be called explicitly after the object is constructed. Or, create a parameterized constructor on the parent object and use the parameter to determine if the code should be executed or not.

    For example:

    :base(true) - This executes your code.
    :base(false) - This does not execute your code.
    
    0 讨论(0)
  • 2021-02-20 00:35

    Your requirement to only need to call the constructor for some derived objects is not in line with the object-oriented principles. If it's a Person, then it needs to be constructed as such.

    If you need shared initialization for some objects, then you should consider creating an Initialize method or adding a parameterized constructor that will be called by the specific derived objects.

    The parameterized constructor approach feels a little awkward:

    public abstract class Person
    {
        protected Person()
        {
        }
    
        protected Person( bool initialize )
        {
            if (initialize)
            {
                Initialize();
            }
        }
    
        // ...
    }
    
    public class Customer : Person
    {
        public Customer(string firstName, string lastName)
        {
           // ...
        }
    }
    
    public class Employee : Person
    {
        public Customer(string firstName, string lastName) : base(true)
        {
           // ...
        }
    }
    
    0 讨论(0)
提交回复
热议问题