473,394 Members | 1,642 Online
Bytes | Software Development & Data Engineering Community
Post Job

Home Posts Topics Members FAQ

Join Bytes to post your question to a community of 473,394 software developers and data experts.

Constructor firing Inheritance in C#

I'm having a problem, and here is a simplified example of code that
demonstrates it:
public class BizObj
{
public string TableName="";
private DataSet oData;

public BizObj()
{
LoadData();
}

public void LoadData()
{
SqlConnection oConn = new SqlConnection("...");
SqlDataAdapter oAdapter = new SqlDataAdapter("SELECT * FROM
"+this.TableName,oConn);
oAdapter.Fill(oData);

}
}

public class AuthorBizObj: BizObj
{
public AuthorBizObj()
{
this.TableName="Authors";
}
}

Now, because the base class constructor fires first, LoadData is called
before the TableName can be set to "Authors".

So I try to override the TableName field in the following manner, hoping
that it will be set before the constructor methods fire:
public class AuthorBizObj: BizObj
{
public string TableName="Authors";

public AuthorBizObj()
{
}
}

But I get a compile warning: "The keyword new is required on
AuthorBizObj.TableName because it hides inherited member BizObj.TableName."

I'm not trying to hide the member, I'm trying to override it.

How can I make sure that the TableName field has the value "Authors" before
the BizObj constructor fires?

<b>This question is strictly about inheritance.</b>

You might say "Pass a value for TableName to the constructor" , but I think
that the Authors bizobj should know its table, not have its name passed into
it.

You might ask "Why do you want to call LoadData in the constructor?" or say
"Call LoadData after you have instantiated AuthorBizObj, not in the
constructor" - I'm actually not creating bizobjs, this is just some sample
code that demonstrates the problem, I'm trying to override a member field,
not figure out where LoadData should be called.

Thanks,
--
Rick Hodder
Jan 20 '06 #1
11 2383
What about something like this? Here name gets set before the WriteLine is called.

class BaseObject
{
protected string name = string.Empty;

protected BaseObject()
{
Init();
}

protected virtual void Init()
{
System.Diagnostics.Debug.WriteLine(name);
}
}

class DerivedObject : BaseObject
{
public DerivedObject()
{
}

protected override void Init()
{
name = "MyTable";
base.Init();
}
}
RickHodder wrote:
I'm having a problem, and here is a simplified example of code that
demonstrates it:
public class BizObj
{
public string TableName="";
private DataSet oData;

public BizObj()
{
LoadData();
}

public void LoadData()
{
SqlConnection oConn = new SqlConnection("...");
SqlDataAdapter oAdapter = new SqlDataAdapter("SELECT * FROM
"+this.TableName,oConn);
oAdapter.Fill(oData);

}
}

public class AuthorBizObj: BizObj
{
public AuthorBizObj()
{
this.TableName="Authors";
}
}

Now, because the base class constructor fires first, LoadData is called
before the TableName can be set to "Authors".

So I try to override the TableName field in the following manner, hoping
that it will be set before the constructor methods fire:
public class AuthorBizObj: BizObj
{
public string TableName="Authors";

public AuthorBizObj()
{
}
}

But I get a compile warning: "The keyword new is required on
AuthorBizObj.TableName because it hides inherited member BizObj.TableName."

I'm not trying to hide the member, I'm trying to override it.

How can I make sure that the TableName field has the value "Authors" before
the BizObj constructor fires?

<b>This question is strictly about inheritance.</b>

You might say "Pass a value for TableName to the constructor" , but I think
that the Authors bizobj should know its table, not have its name passed into
it.

You might ask "Why do you want to call LoadData in the constructor?" or say
"Call LoadData after you have instantiated AuthorBizObj, not in the
constructor" - I'm actually not creating bizobjs, this is just some sample
code that demonstrates the problem, I'm trying to override a member field,
not figure out where LoadData should be called.

Thanks,

Jan 20 '06 #2
I'm not trying to hide the member, I'm trying to override it.
Well you can't override fields.

You might say "Pass a value for TableName to the constructor" ,
Exactly.

but I think
that the Authors bizobj should know its table, not have its name passed into
it.


I don't see how that prevents you from passing it to a BizObj
constructor. Wouldn't this work for you?

public class BizObj
{
public string TableName;

protected BizObj(string tableName)
{
TableName = tableName;
LoadData();
}

...
}

public class AuthorBizObj : BizObj
{
public AuthorBizObj() : base("Authors")
{
}
}
Mattias

--
Mattias Sjögren [C# MVP] mattias @ mvps.org
http://www.msjogren.net/dotnet/ | http://www.dotnetinterop.com
Please reply only to the newsgroup.
Jan 20 '06 #3
> I'm having a problem, and here is a simplified example of code that
demonstrates it:

public class BizObj
{
public string TableName="";
private DataSet oData;
public BizObj()
{
LoadData();
}
public void LoadData()
{
SqlConnection oConn = new SqlConnection("...");
SqlDataAdapter oAdapter = new SqlDataAdapter("SELECT * FROM
"+this.TableName,oConn);
oAdapter.Fill(oData);
}
}
public class AuthorBizObj: BizObj
{
public AuthorBizObj()
{
this.TableName="Authors";
}
}
Now, because the base class constructor fires first, LoadData is
called before the TableName can be set to "Authors".

So I try to override the TableName field in the following manner,
hoping that it will be set before the constructor methods fire:

public class AuthorBizObj: BizObj
{
public string TableName="Authors";
public AuthorBizObj()
{
}
}
But I get a compile warning: "The keyword new is required on
AuthorBizObj.TableName because it hides inherited member
BizObj.TableName."

I'm not trying to hide the member, I'm trying to override it.

How can I make sure that the TableName field has the value "Authors"
before the BizObj constructor fires?

<b>This question is strictly about inheritance.</b>

You might say "Pass a value for TableName to the constructor" , but I
think that the Authors bizobj should know its table, not have its name
passed into it.

You might ask "Why do you want to call LoadData in the constructor?"
or say "Call LoadData after you have instantiated AuthorBizObj, not in
the constructor" - I'm actually not creating bizobjs, this is just
some sample code that demonstrates the problem, I'm trying to override
a member field, not figure out where LoadData should be called.

Thanks,


I think "overriding" or "hiding" a field is bad news. How about this.

public abstract class BizObj
{
private readonly string TableName = string.Empty;

protected BizObj(string tableName)
{
this.TableName = tableName;
Console.WriteLine("BizObj Ctor");
LoadData();
}

protected void LoadData()
{
Console.WriteLine("TableName: {0}", TableName);
}
}

public class AuthorBizObj : BizObj
{
public AuthorBizObj() : base("Author")
{
Console.WriteLine("AuthorBizObj Ctor.");
}
}
Jan 20 '06 #4
Hi Darren,

Thanks for the reply.

I am trying to override a field: something that should be available in an
object-oriented language, IMO.

For example: If I create a Vehicle class with a WheelCount property, and
then I create a subclass Car, I dont think that I should have to set up a
special method just so that I can set the WheelCount to 4, I should just be
able to initialize WheelCount to 4.

And what's more annoying is that you can override a property - another
developer suggested that I should override the property:

public string TableName
{
get
{
return "Authors";
}
}

But that seems like it circumvents inheritance.

I'm trying to understand the thought process.
--
Rick Hodder

Jan 20 '06 #5
RickHodder <Ri********@discussions.microsoft.com> wrote:
Thanks for the reply.

I am trying to override a field: something that should be available in an
object-oriented language, IMO.
It's not been available in any of the languages I've worked in. I don't
see it's necessary, either.
For example: If I create a Vehicle class with a WheelCount property, and
then I create a subclass Car, I dont think that I should have to set up a
special method just so that I can set the WheelCount to 4, I should just be
able to initialize WheelCount to 4.
You could have a virtual *property*, as you say below.
And what's more annoying is that you can override a property - another
developer suggested that I should override the property:

public string TableName
{
get
{
return "Authors";
}
}

But that seems like it circumvents inheritance.

I'm trying to understand the thought process.


In what way does that circumvent inheritance? I'd suggest making it
abstract in the base class, even.

I prefer the idea of passing the name of the table to the base
constructor though. The business object knows the table it's part of
that way, but the business object class itself doesn't know any
specific tables. Sounds ideal to me.

--
Jon Skeet - <sk***@pobox.com>
http://www.pobox.com/~skeet Blog: http://www.msmvps.com/jon.skeet
If replying to the group, please do not mail me too
Jan 20 '06 #6
Hi Mattias,

Thanks for replying

I hope what I say doesnt come across as an attack - I'm just trying to
understand.
Well you can't override fields.
What's the thinking behind that - why shouldnt fields be overridable?
Properties are, but they are public, what if I dont want it to be public?
You might say "Pass a value for TableName to the constructor" ,

Exactly.


That seems like a workaround - I'm trying to approach this totally from an
OOP point of view. Why have inheritance of fields if you are going to merely
pass all of the values into the constructor - and as the number of fields in
the base class(es) increase, the signature of the constructor must keep
growing.

I realize that part of the problem here is that I'm calling a method from a
constructor.
I don't see how that prevents you from passing it to a BizObj
constructor. Wouldn't this work for you?


If I know when I create a class what value a field should have, why should I
have to pass it in?

Bizobj was just an example of a class: I'm not trying to argue whether the
tablename should be passed into the constructor when I'm designing a Bizobj
class. But if I know that AuthorBizobj will have a table name "Authors" and
it will never change, why should I have to pass it in to the class?

--
Thanks,
Rick Hodder

Jan 20 '06 #7
RickHodder <Ri********@discussions.microsoft.com> wrote:
Well you can't override fields.
What's the thinking behind that - why shouldnt fields be overridable?


Fields aren't callable members - there's no vtable for them. I don't
think I even know what it would mean to be able to override a field.
Properties are, but they are public, what if I dont want it to be public?
Properties aren't always public. They can be private, protected,
internal etc.
You might say "Pass a value for TableName to the constructor" ,

Exactly.


That seems like a workaround - I'm trying to approach this totally from an
OOP point of view. Why have inheritance of fields if you are going to merely
pass all of the values into the constructor - and as the number of fields in
the base class(es) increase, the signature of the constructor must keep
growing.


Well, only the fields for which the base class needs information from
the derived classes need to be initialised in the constructor.
I realize that part of the problem here is that I'm calling a method from a
constructor.
Partly. Calling overridable methods in the constructor is almost always
a no-no. You need to clearly indicate in the documentation that the
method will be called before the object is fully constructed. For
instance, at that point there may be invariants which don't yet hold.
I don't see how that prevents you from passing it to a BizObj
constructor. Wouldn't this work for you?


If I know when I create a class what value a field should have, why should I
have to pass it in?


So that Bizobj doesn't need to know about the Author class at all.
Bizobj was just an example of a class: I'm not trying to argue whether the
tablename should be passed into the constructor when I'm designing a Bizobj
class. But if I know that AuthorBizobj will have a table name "Authors" and
it will never change, why should I have to pass it in to the class?


Well you need to make the data available *somewhere*. I don't see why
passing it in the constructor is worse than having it anywhere else -
it's simple and it works.

--
Jon Skeet - <sk***@pobox.com>
http://www.pobox.com/~skeet Blog: http://www.msmvps.com/jon.skeet
If replying to the group, please do not mail me too
Jan 21 '06 #8
Hi Jon,

Thanks for replying
I prefer the idea of passing the name of the table to the base
constructor though. The business object knows the table it's part of
that way, but the business object class itself doesn't know any
specific tables. Sounds ideal to me.


As I said in my post, I'm asking a more general question, the bizobj just
happened to be the example. I was trying to steer away from an app-specific
answer. Passing the name of the table to the bizobj speaks to the developers
thoughts on bizobjs. It works well if the hierarchy is more geared towards a
generic bizobj that won't necessarily be subclassed - you pass in the
tablename and the queries are structured programmatically.

I guess the best way to re-state my request is "I have code in a constructor
that calls out to another method . When creating subclasses of this class,
how can I override the values of fields (using initializers maybe??) so that
when the base class calls out to the method, the field is initialized to the
value appropriate for the subclass"

Thanks,
Rick
Jan 23 '06 #9
> Hi Jon,

Thanks for replying
I prefer the idea of passing the name of the table to the base
constructor though. The business object knows the table it's part of
that way, but the business object class itself doesn't know any
specific tables. Sounds ideal to me.

As I said in my post, I'm asking a more general question, the bizobj
just happened to be the example. I was trying to steer away from an
app-specific answer. Passing the name of the table to the bizobj
speaks to the developers thoughts on bizobjs. It works well if the
hierarchy is more geared towards a generic bizobj that won't
necessarily be subclassed - you pass in the tablename and the queries
are structured programmatically.

I guess the best way to re-state my request is "I have code in a
constructor that calls out to another method . When creating
subclasses of this class, how can I override the values of fields
(using initializers maybe??) so that when the base class calls out to
the method, the field is initialized to the value appropriate for the
subclass"

Thanks,
Rick


You've already heard the correct answer. You don't "override" fields that
the method depends on. You have to pass the variables into the base constructor.
Jan 23 '06 #10
RickHodder wrote:

<snip>
I guess the best way to re-state my request is "I have code in a constructor
that calls out to another method . When creating subclasses of this class,
how can I override the values of fields (using initializers maybe??) so that
when the base class calls out to the method, the field is initialized to the
value appropriate for the subclass"


You *could* have an Init method which could be overridden in derived
classes. However, my best advice would be to try to avoid that design
pattern in the first place.

Jon

Jan 23 '06 #11
It is best to keep as much of the initialization code in the base as possible.
That way, if your base class needs to change there is less impact on derived classes.
Jon Skeet [C# MVP] wrote:
RickHodder wrote:

<snip>
I guess the best way to re-state my request is "I have code in a constructor
that calls out to another method . When creating subclasses of this class,
how can I override the values of fields (using initializers maybe??) so that
when the base class calls out to the method, the field is initialized to the
value appropriate for the subclass"


You *could* have an Init method which could be overridden in derived
classes. However, my best advice would be to try to avoid that design
pattern in the first place.

Jon

Jan 25 '06 #12

This thread has been closed and replies have been disabled. Please start a new discussion.

Similar topics

7
by: Robin Forster | last post by:
I have two classes: aule_gl_window (parent class) and aule_button (sub class) I want to call the super class (parent) constructor code from the sub class constructor.
0
by: Lefevre | last post by:
Hello I recently had troubles with a class inheritance hierarchy. I solved it, but it didn't satisfied me. I found the solution using this forum :) Actualy i found the following message...
10
by: jeffc | last post by:
When compiling the following program, I get an error on the line specified: The base class "B" cannot be initialized because it does not have a default constructor. If I remove the "virtual"...
3
by: Jun | last post by:
I have following script <script> var Animal = function(name){ this.name = name; } Animal.prototype.eat = function (food) {
45
by: Ben Blank | last post by:
I'm writing a family of classes which all inherit most of their methods and code (including constructors) from a single base class. When attempting to instance one of the derived classes using...
4
by: Francisco Amaro | last post by:
Hi all, Have question about inheriting a class that has parameters in the constructor such as : Public MustInherit Class MyParentClass Public mystring As String Public Sub New(ByVal...
3
by: kk_oop | last post by:
Hi. I have base class Base and derived class Derived. My derived class constructor is dependent upon attributes set in the base class constructor. Is this okay? Does ANSI C++ guarantee that a...
4
by: ingoweiss | last post by:
Hi, I am having trouble passing parameters of a Javascript subclass constructor through to it's superclass constructor. I am trying all sorts of things, including the below, but nothing...
3
by: hurcan solter | last post by:
Consider the code fragment; class A { public: A(){} A(int prm):mprm(prm){} int mprm; }; class B:public A {
0
by: Charles Arthur | last post by:
How do i turn on java script on a villaon, callus and itel keypad mobile phone
0
by: ryjfgjl | last post by:
If we have dozens or hundreds of excel to import into the database, if we use the excel import function provided by database editors such as navicat, it will be extremely tedious and time-consuming...
0
by: emmanuelkatto | last post by:
Hi All, I am Emmanuel katto from Uganda. I want to ask what challenges you've faced while migrating a website to cloud. Please let me know. Thanks! Emmanuel
1
by: nemocccc | last post by:
hello, everyone, I want to develop a software for my android phone for daily needs, any suggestions?
1
by: Sonnysonu | last post by:
This is the data of csv file 1 2 3 1 2 3 1 2 3 1 2 3 2 3 2 3 3 the lengths should be different i have to store the data by column-wise with in the specific length. suppose the i have to...
0
marktang
by: marktang | last post by:
ONU (Optical Network Unit) is one of the key components for providing high-speed Internet services. Its primary function is to act as an endpoint device located at the user's premises. However,...
0
Oralloy
by: Oralloy | last post by:
Hello folks, I am unable to find appropriate documentation on the type promotion of bit-fields when using the generalised comparison operator "<=>". The problem is that using the GNU compilers,...
0
by: Hystou | last post by:
Overview: Windows 11 and 10 have less user interface control over operating system update behaviour than previous versions of Windows. In Windows 11 and 10, there is no way to turn off the Windows...
0
tracyyun
by: tracyyun | last post by:
Dear forum friends, With the development of smart home technology, a variety of wireless communication protocols have appeared on the market, such as Zigbee, Z-Wave, Wi-Fi, Bluetooth, etc. Each...

By using Bytes.com and it's services, you agree to our Privacy Policy and Terms of Use.

To disable or enable advertisements and analytics tracking please visit the manage ads & tracking page.