473,395 Members | 1,488 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,395 software developers and data experts.

Type "assurance" of derived classes

Hi All,

I'm coming from Java to C++ and this is one of the very last problems I
have so far... In Java, if I have, say, a class SISNode that extends
NetworkNode, I can have a function that returns a NetworkNode but I can
assure the compiler that it is in fact a SISNode and therefore call the
method getStatus() that only a SISNode has. Like

SISnode s,t;
NetworkNode n;
n =t;
n.getStatus();//won't work
s= (SISNode) n;
s.getStatus(); //will work
....
....

I'm now looking for some way to do this in C++. I do agent-based network
simulations, and I want to derive all kinds of agents from a generic
network node type. This network node is supposed to store his neighbours
in a std::list<GenericNetworkNodelist. Now in the derived classes I
can obtain the neighbours, but I cannot call their methods unless they
were already declared in the GenericNetworkNode declaration.

Anybody knows how to solve this problem? A hint in the right direction
(keyword) would be more than enough....

Thanks

Oliver
Aug 14 '08 #1
8 1618
Oliver Graeser <gr*****@phy.cuhk.edu.hkwrites:
Hi All,

I'm coming from Java to C++ and this is one of the very last problems
I have so far... In Java, if I have, say, a class SISNode that extends
NetworkNode, I can have a function that returns a NetworkNode but I
can assure the compiler that it is in fact a SISNode and therefore
call the method getStatus() that only a SISNode has. Like

SISnode s,t;
NetworkNode n;
n =t;
n.getStatus();//won't work
s= (SISNode) n;
s.getStatus(); //will work
No it won't.
...
...

I'm now looking for some way to do this in C++. I do agent-based
network simulations, and I want to derive all kinds of agents from a
generic network node type. This network node is supposed to store his
neighbours in a std::list<GenericNetworkNodelist. Now in the derived
classes I can obtain the neighbours, but I cannot call their methods
unless they were already declared in the GenericNetworkNode
declaration.

Anybody knows how to solve this problem? A hint in the right direction
(keyword) would be more than enough....
This is not possible, with what you wrote above and the assumed
declarations. You must realise that when you run n=t; you lose all
the information specific to a SISNode. There is no way to build it
back.

That's why I consider that C++ objects should always be allocated
dynamically and referend thru a pointer.

If you had written:

class SISNode:public NetworkNode {...};

SISNode* s=new SISNode();
NetworkNode* n=s;

then you could write:

SISNode* nAsSISNode=dynamic_cast<SISNode*>n;
if(nAsSISNode!=0){ nAsSISNode->getStatus(); }

--
__Pascal Bourguignon__
Aug 14 '08 #2
On 2008-08-14 12:44, Pascal J. Bourguignon wrote:
Oliver Graeser <gr*****@phy.cuhk.edu.hkwrites:
>Hi All,

I'm coming from Java to C++ and this is one of the very last problems
I have so far... In Java, if I have, say, a class SISNode that extends
NetworkNode, I can have a function that returns a NetworkNode but I
can assure the compiler that it is in fact a SISNode and therefore
call the method getStatus() that only a SISNode has. Like

SISnode s,t;
NetworkNode n;
n =t;
n.getStatus();//won't work
s= (SISNode) n;
s.getStatus(); //will work

No it won't.
>...
...

I'm now looking for some way to do this in C++. I do agent-based
network simulations, and I want to derive all kinds of agents from a
generic network node type. This network node is supposed to store his
neighbours in a std::list<GenericNetworkNodelist. Now in the derived
classes I can obtain the neighbours, but I cannot call their methods
unless they were already declared in the GenericNetworkNode
declaration.

Anybody knows how to solve this problem? A hint in the right direction
(keyword) would be more than enough....

This is not possible, with what you wrote above and the assumed
declarations. You must realise that when you run n=t; you lose all
the information specific to a SISNode. There is no way to build it
back.

That's why I consider that C++ objects should always be allocated
dynamically and referend thru a pointer.

If you had written:

class SISNode:public NetworkNode {...};

SISNode* s=new SISNode();
NetworkNode* n=s;

then you could write:

SISNode* nAsSISNode=dynamic_cast<SISNode*>n;
if(nAsSISNode!=0){ nAsSISNode->getStatus(); }
And, assuming that NetworkNode is derived from GenericNetworkNode you
can then use std::list<GenericNetworkNode*to store them. Though it
might be worth to follow Alf's advice and use some kind of smart pointer
(which would make it std::list<SmartPointerType<GenericNetworkNode).

--
Erik Wikström
Aug 14 '08 #3
On Aug 14, 8:03 pm, Erik Wikström <Erik-wikst...@telia.comwrote:
On 2008-08-14 12:44, Pascal J. Bourguignon wrote:
Oliver Graeser <grae...@phy.cuhk.edu.hkwrites:
I'm coming from Java to C++ and this is one of the very
last problems I have so far... In Java, if I have, say, a
class SISNode that extends NetworkNode, I can have a
function that returns a NetworkNode but I can assure the
compiler that it is in fact a SISNode and therefore call
the method getStatus() that only a SISNode has. Like
SISnode s,t;
NetworkNode n;
n =t;
n.getStatus();//won't work
s= (SISNode) n;
s.getStatus(); //will work
This is supposed to be Java, I suppose.

The exact equivalent in C++ would be:

SISnode* s ;
SISnode* t ;
NetworkNode* n ;
n = t ;
n->getStatus() ; // won't work
s = dynamic_cast< SISnode* >( n ) ;
s->getStatus() ; // will work./
No it won't.
I think his example is supposed to be Java. Otherwise, it won't
compile.
...
...
I'm now looking for some way to do this in C++. I do
agent-based network simulations, and I want to derive all
kinds of agents from a generic network node type. This
network node is supposed to store his neighbours in a
std::list<GenericNetworkNodelist. Now in the derived
classes I can obtain the neighbours, but I cannot call
their methods unless they were already declared in the
GenericNetworkNode declaration.
Anybody knows how to solve this problem? A hint in the
right direction (keyword) would be more than enough....
The two important points to remember are that C++ has value
semantics by default, you have to explicitly use pointers or
references to get reference semantics, and that the equivalent
to Java's cast operator in C++ is dynamic_cast.
This is not possible, with what you wrote above and the assumed
declarations. You must realise that when you run n=t; you lose all
the information specific to a SISNode. There is no way to build it
back.
If they above were C++, he'd not loose any type information.
But he'd get a new object (with a new type), which is probably
not what he wants.
That's why I consider that C++ objects should always be
allocated dynamically and referend thru a pointer.
That's valid for entity objects, but not for value objects.
If you had written:
class SISNode:public NetworkNode {...};
SISNode* s=new SISNode();
NetworkNode* n=s;
then you could write:
SISNode* nAsSISNode=dynamic_cast<SISNode*>n;
if(nAsSISNode!=0){ nAsSISNode->getStatus(); }
And, assuming that NetworkNode is derived from
GenericNetworkNode you can then use
std::list<GenericNetworkNode*to store them. Though it might
be worth to follow Alf's advice and use some kind of smart
pointer (which would make it
std::list<SmartPointerType<GenericNetworkNode).
If these are entity objects, as it would seem, I don't know of a
smart pointer that would really be appropriate.

--
James Kanze (GABI Software) email:ja*********@gmail.com
Conseils en informatique orientée objet/
Beratung in objektorientierter Datenverarbeitung
9 place Sémard, 78210 St.-Cyr-l'École, France, +33 (0)1 30 23 00 34
Aug 14 '08 #4
James Kanze wrote:
On Aug 14, 8:03 pm, Erik Wikström <Erik-wikst...@telia.comwrote:
>On 2008-08-14 12:44, Pascal J. Bourguignon wrote:
>>Oliver Graeser <grae...@phy.cuhk.edu.hkwrites:
I'm coming from Java to C++ and this is one of the very
last problems I have so far... In Java, if I have, say, a
class SISNode that extends NetworkNode, I can have a
function that returns a NetworkNode but I can assure the
compiler that it is in fact a SISNode and therefore call
the method getStatus() that only a SISNode has. Like
>>>SISnode s,t;
NetworkNode n;
n =t;
n.getStatus();//won't work
s= (SISNode) n;
s.getStatus(); //will work

This is supposed to be Java, I suppose.
Yes, it was supposed to be Java
>
The exact equivalent in C++ would be:

SISnode* s ;
SISnode* t ;
NetworkNode* n ;
n = t ;
n->getStatus() ; // won't work
s = dynamic_cast< SISnode* >( n ) ;
s->getStatus() ; // will work./
>>No it won't.
Thanks a lot for that! I will consider what Alf Steinbach wrote above on
how to do it with a boost::shared_ptr, but in any case this is exactly
what I was trying to do. The word "dynamic_cast" is actually the key, it
is really hard to google something when you don't know its name ;)
I think his example is supposed to be Java. Otherwise, it won't
compile.
>>>...
...
>>>I'm now looking for some way to do this in C++. I do
agent-based network simulations, and I want to derive all
kinds of agents from a generic network node type. This
network node is supposed to store his neighbours in a
std::list<GenericNetworkNodelist. Now in the derived
classes I can obtain the neighbours, but I cannot call
their methods unless they were already declared in the
GenericNetworkNode declaration.
>>>Anybody knows how to solve this problem? A hint in the
right direction (keyword) would be more than enough....

The two important points to remember are that C++ has value
semantics by default, you have to explicitly use pointers or
references to get reference semantics, and that the equivalent
to Java's cast operator in C++ is dynamic_cast.
>>This is not possible, with what you wrote above and the assumed
declarations. You must realise that when you run n=t; you lose all
the information specific to a SISNode. There is no way to build it
back.

If they above were C++, he'd not loose any type information.
But he'd get a new object (with a new type), which is probably
not what he wants.
Yup. Actually this was a problem I encountered before when I had an
object a, created a new object b, said b=a, manipulated b and was
stunned that a was still the same.... I'm sorry, I think the question
was probably not really well asked, I tried to keep it concise.
>>That's why I consider that C++ objects should always be
allocated dynamically and referend thru a pointer.

That's valid for entity objects, but not for value objects.
>>If you had written:
>>class SISNode:public NetworkNode {...};
>>SISNode* s=new SISNode();
NetworkNode* n=s;
>>then you could write:
>>SISNode* nAsSISNode=dynamic_cast<SISNode*>n;
if(nAsSISNode!=0){ nAsSISNode->getStatus(); }
>And, assuming that NetworkNode is derived from
GenericNetworkNode you can then use
std::list<GenericNetworkNode*to store them. Though it might
be worth to follow Alf's advice and use some kind of smart
pointer (which would make it
std::list<SmartPointerType<GenericNetworkNode).

If these are entity objects, as it would seem, I don't know of a
smart pointer that would really be appropriate.
Ayah. Smart pointer, entity objects, value objects - enough for today to
read on. Thanks a lot, everyone!
--
James Kanze (GABI Software) email:ja*********@gmail.com
Conseils en informatique orientée objet/
Beratung in objektorientierter Datenverarbeitung
9 place Sémard, 78210 St.-Cyr-l'École, France, +33 (0)1 30 23 00 34
Aug 15 '08 #5
On 15 Aug, 09:16, Oliver Graeser <grae...@phy.cuhk.edu.hkwrote:
>Oliver Graeser <grae...@phy.cuhk.edu.hkwrites:
I'm coming from Java to C++ and this is one of the very
last problems I have so far... In Java, if I have, say, a
class SISNode that extends NetworkNode, I can have a
function that returns a NetworkNode but I can assure the
compiler that it is in fact a SISNode and therefore call
the method getStatus() that only a SISNode has. Like
>>SISnode s,t;
NetworkNode n;
n =t;
n.getStatus();//won't work
s= (SISNode) n;
s.getStatus(); //will work
This is supposed to be Java, I suppose.

Yes, it was supposed to be Java
The exact equivalent in C++ would be:
* * SISnode* s ;
* * SISnode* t ;
* * NetworkNode* n ;
* * n = t ;
* * n->getStatus() ; * // won't work
* * s = dynamic_cast< SISnode* >( n ) ;
* * s->getStatus() ; * // will work./
>No it won't.

Thanks a lot for that! I will consider what Alf Steinbach wrote above on
how to do it with a boost::shared_ptr, but in any case this is exactly
what I was trying to do. The word "dynamic_cast" is actually the key, it
is really hard to google something when you don't know its name ;)
I'd just like to mention that if the dynamic type of *s happened to be
something else (not a SISnode), then this would not work. To ensure
that the cast succeeded, check the pointer against 0.

s = dynamic_cast<SISnode*>(n);
if(!s)
{
// the cast failed, handle error...
}

or:

if( !(s = dynamic_cast<SISnode*>(n)))
{
// handle error...
}

I prefer the former.

DP
Aug 15 '08 #6
Triple-DES wrote:
On 15 Aug, 09:16, Oliver Graeser <grae...@phy.cuhk.edu.hkwrote:
>>>>Oliver Graeser <grae...@phy.cuhk.edu.hkwrites:
>I'm coming from Java to C++ and this is one of the very
>last problems I have so far... In Java, if I have, say, a
>class SISNode that extends NetworkNode, I can have a
>function that returns a NetworkNode but I can assure the
>compiler that it is in fact a SISNode and therefore call
>the method getStatus() that only a SISNode has. Like
>SISnode s,t;
>NetworkNode n;
>n =t;
>n.getStatus();//won't work
>s= (SISNode) n;
>s.getStatus(); //will work
This is supposed to be Java, I suppose.
Yes, it was supposed to be Java
>>The exact equivalent in C++ would be:
SISnode* s ;
SISnode* t ;
NetworkNode* n ;
n = t ;
n->getStatus() ; // won't work
s = dynamic_cast< SISnode* >( n ) ;
s->getStatus() ; // will work./
No it won't.
Thanks a lot for that! I will consider what Alf Steinbach wrote above on
how to do it with a boost::shared_ptr, but in any case this is exactly
what I was trying to do. The word "dynamic_cast" is actually the key, it
is really hard to google something when you don't know its name ;)

I'd just like to mention that if the dynamic type of *s happened to be
something else (not a SISnode), then this would not work. To ensure
that the cast succeeded, check the pointer against 0.
Thanks for the hint. I'm actually not worried about having nodes with
the wrong type since every network only has one kind of node, I'm going
through this ordeal only because I want to keep all reporting methods in
the same network class.

I used you hint though to work out another problem. I originally stored
all my nodes in an array, such as

GenericNetworkNode * nodeList;//declaration
......

nodeList = new SISNetworkNode[size];//network constructor
......
SISNetworkNode *snn;
snn=dynamic_cast <SISNetworkNode*(&nodeList[i]);//here comes the disaster

I managed to find out that it is something about the array, i.e. if I
instead use an array of pointers, create a node for each pointer, and
then do the dynamic cast for nodeList[i] instead of &nodeList[i], it
works. Is there any particular reason why dynamic casts doesn't work
with arrays?

>
s = dynamic_cast<SISnode*>(n);
if(!s)
{
// the cast failed, handle error...
}

or:

if( !(s = dynamic_cast<SISnode*>(n)))
{
// handle error...
}

I prefer the former.

DP
Aug 18 '08 #7
On 18 Aug, 04:02, Oliver Graeser <grae...@phy.cuhk.edu.hkwrote:
I used you hint though to work out another problem. I originally stored
all my nodes in an array, such as

GenericNetworkNode * nodeList;//declaration
.....

nodeList = new SISNetworkNode[size];//network constructor
.....
SISNetworkNode *snn;
snn=dynamic_cast <SISNetworkNode*(&nodeList[i]);//here comes the disaster

I managed to find out that it is something about the array, i.e. if I
instead use an array of pointers, create a node for each pointer, and
then do the dynamic cast for nodeList[i] instead of &nodeList[i], it
works. Is there any particular reason why dynamic casts doesn't work
with arrays?
Yes. When you assign an array of SISNetworkNode to a
GenericNetworkNode*, the array decays to a pointer to the first
element. This pointer is then converted to a GenericNetworkNode*. The
fact that this pointer actually points to an array of SISNetworkNode
is "lost" to the compiler.

When you later write nodeList[i], the compiler will compute the memory
location as if you had an array of GenericNetworkNode, not
SISNetworkNode. Since SISNetworkNode is almost surely larger than
GenericNetworkNode you will end up referencing some random memory
location within the array.

As you have already figured out, arrays of pointers solves the
problem. But it would be even better to use std::vector instead and
avoid arrays.

DP
Aug 18 '08 #8
HL
On Aug 18, 10:02*am, Oliver Graeser <grae...@phy.cuhk.edu.hkwrote:
Triple-DES wrote:
On 15 Aug, 09:16, Oliver Graeser <grae...@phy.cuhk.edu.hkwrote:
>>>Oliver Graeser <grae...@phy.cuhk.edu.hkwrites:
I'm coming from Java to C++ and this is one of the very
last problems I have so far... In Java, if I have, say, a
class SISNode that extends NetworkNode, I can have a
function that returns a NetworkNode but I can assure the
compiler that it is in fact a SISNode and therefore call
the method getStatus() that only a SISNode has. Like
SISnode s,t;
NetworkNode n;
n =t;
n.getStatus();//won't work
s= (SISNode) n;
s.getStatus(); //will work
This is supposed to be Java, I suppose.
Yes, it was supposed to be Java
>The exact equivalent in C++ would be:
* * SISnode* s ;
* * SISnode* t ;
* * NetworkNode* n ;
* * n = t ;
* * n->getStatus() ; * // won't work
* * s = dynamic_cast< SISnode* >( n ) ;
* * s->getStatus() ; * // will work./
No it won't.
Thanks a lot for that! I will consider what Alf Steinbach wrote above on
how to do it with a boost::shared_ptr, but in any case this is exactly
what I was trying to do. The word "dynamic_cast" is actually the key, it
is really hard to google something when you don't know its name ;)
I'd just like to mention that if the dynamic type of *s happened to be
something else (not a SISnode), then this would not work. To ensure
that the cast succeeded, check the pointer against 0.

Thanks for the hint. I'm actually not worried about having nodes with
the wrong type since every network only has one kind of node, I'm going
through this ordeal only because I want to keep all reporting methods in
the same network class.

I used you hint though to work out another problem. I originally stored
all my nodes in an array, such as

GenericNetworkNode * nodeList;//declaration
.....

nodeList = new SISNetworkNode[size];//network constructor
.....
SISNetworkNode *snn;
snn=dynamic_cast <SISNetworkNode*(&nodeList[i]);//here comes the disaster

I managed to find out that it is something about the array, i.e. if I
instead use an array of pointers, create a node for each pointer, and
then do the dynamic cast for nodeList[i] instead of &nodeList[i], it
works. Is there any particular reason why dynamic casts doesn't work
with arrays?


s = dynamic_cast<SISnode*>(n);
if(!s)
{
* // the cast failed, handle error...
}
or:
if( !(s = dynamic_cast<SISnode*>(n)))
{
* // handle error...
}
I prefer the former.
DP- Hide quoted text -

- Show quoted text -- Hide quoted text -

- Show quoted text -
Oliver, your question is exactly about polymorphism.
polymorphism in C++ is implemented with Pointers and References. You
can't expect polymorphic behavior on objects.
And, since they SisNode and NetworkNode are in the same class
hierarchy, static_cast is supposed to use instead of dynamic_cast. The
latter is usually used in multi-inheritence casting a base to its
sibling. But it's so possible to be fail that Triple-DEC mentioned to
check the result of casting.
Aug 18 '08 #9

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

Similar topics

1
by: Ahmad Noori | last post by:
I have a form and in the form, I have a drop down box. Based on what the user selects, i want to display different input boxes. Here is my drop down box: <td> <select name="reptype"...
17
by: Torbjørn Pettersen | last post by:
I've got a table where I want some of the cells to use a background image. The cells have variable height, so I am using an image with a rather small height to fill up the background of the cells,...
18
by: Leslaw Bieniasz | last post by:
Cracow, 28.10.2004 Hello, I have a program that intensively allocates and deletes lots of relatively small objects, using "new" operator. The objects themselves are composed of smaller...
4
by: yf | last post by:
A KB article "http://support.microsoft.com/default.aspx?scid=kb;en-us;209599" tells that the maximum number of records that a table may hold if the PRIMARY key data type is set to AUTONUMBER is...
188
by: infobahn | last post by:
printf("%p\n", (void *)0); /* UB, or not? Please explain your answer. */
6
by: s99999999s2003 | last post by:
hi i come from a non OO environment. now i am learning about classes. can i ask, in JAva, there are things like interface. eg public interface someinterface { public somemethod (); .... ... }...
21
by: T.A. | last post by:
I understand why it is not safe to inherit from STL containers, but I have found (in SGI STL documentation) that for example bidirectional_iterator class can be used to create your own iterator...
41
by: Stuart Golodetz | last post by:
Hi all, Just wondering whether there's any reason why exception specifications are enforced at runtime, rather than at compile-time like in Java? (This was prompted by reading an article on...
33
by: phfle1 | last post by:
Hi, I want to display the cursor as an hourglass on the click of a button in the form and use the following line to achieve that Me.btnRecherche.Attributes.Add("onclick",...
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: ryjfgjl | last post by:
In our work, we often receive Excel tables with data in the same format. If we want to analyze these data, it can be difficult to analyze them because the data is spread across multiple Excel files...
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
0
BarryA
by: BarryA | last post by:
What are the essential steps and strategies outlined in the Data Structures and Algorithms (DSA) roadmap for aspiring data scientists? How can individuals effectively utilize this roadmap to progress...
1
by: nemocccc | last post by:
hello, everyone, I want to develop a software for my android phone for daily needs, any suggestions?
0
by: Hystou | last post by:
There are some requirements for setting up RAID: 1. The motherboard and BIOS support RAID configuration. 2. The motherboard has 2 or more available SATA protocol SSD/HDD slots (including MSATA, M.2...
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
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.