dfoster@jarthur.Claremont.EDU (Derek R. Foster) (09/19/90)
A question from a very frustrated semi-novice C++ programmer: Is there any way to change the variable referenced by a reference variable after the reference variable has been initialized? In particular, I have a class which is passed a reference to a target variable which it is to modify. I would like to be able to simply declare a class member variable which would be a reference to the target variable, thus allowing my class member operations to access the target variable as if it was simply a member variable. In other words, I would like to be able to 'forget' that the target variable is actually external to the particular class object which has been told to modify it, except for in the routines which change to a new target. This seems like a perfectly legitimate use of a reference variable, and would work quite nicely if I was able to specify the target variable at the time a class object was constructed. Unfortunately, I can't. For example, class targetclass; class targetmodifyer { targetclass& targ; // a reference variable targetclass default; // a default 'internal' target targetmodifyer() : targ(default) {}; // initialize the variable to // point to the default // target modify() {targ.dosomething();}; // modify the target variable // in some way newtarget(targetclass& T) { ??????????? }; // but what do I do here? // I want to re-initialize // targ to now reference T, // but how? }; void main(void) { targetclass targ1; // a target targetmodifyer mod1(); // set up a target modifyer to use // an internal target mod1.modify(); // modify the internal target mod1.newtarget(targ1); // use the second target now mod1.modify(); // modify the second target } My question is this: what do I put in place of ???????? ? What I would _LIKE_ to be able to do is something like: ((targetclass &) targ) = newtarget; // set targ to point to a new target but this doesn't work due to C++'s semantic rules. (It is interpreted as '(targetclass) targ = newtarget' apparently, and results in a call to targetclass::operator = () which is not what I want.) I've tried all sorts of bizarre type casts to try to make the compiler understand that I want to change _what_ targ references instead of changing the value of the thing that targ references, but C++ steadfastly refuses to understand me. It also refuses to let me initialize targ a second time (i.e. 'targ(newtarget)') except in the context of a base class initializer (see kludge #2, below). I have tried all sorts of ways to do this and have come up with a few kludges but no really good way. I have scanned the Turbo C++ manual nearly from cover to cover and have found no reference of doing this, either "it can be done like so...", or "it can't be done." Incidentally, my specific application is that 'targetclass' is a string object (dynamically allocated, supports str1 = str2 + str3, etc.) and 'targetmodifyer' is a set of routines and variables which allow the user to edit a string on the screen, in color, performing validity checks, etc. Just for reference, the kludges I have found are: 1) Declare targ as a pointer to targetclass, then every time I reference what it points to, use indirection operators (ie. targ->dosomething()). There are two reasons I don't want to do this: (1) My class 'targetclass' has a bunch of overloaded operators, and it really looks ugly to have to call '(*targ)[something]' or '(*targ) += x', etc. when I know that I ought to be able to just say 'targ[something] and targ += x. (2) besides looking ugly, this makes my code almost unreadable. I went to a great deal of trouble to make my targetclass's operations simple and self-documenting so that they could be used in complex expressions and still be clear. If in my targetmodifyer I have to explicitly dereference them every time they are used, add extra parentheses, etc., I will turn relatively straightforward expressions into nightmarish tangles that I'm not sure even _I_ understand. 2) Declare a private dummy constructor which takes an argument of type (targetclass *) and which uses its base-class initialization list to set targ to reference the thing pointed to by the argument. Then call this constructor from newtarget(). This works, but seems quite kludgy since (1) I am calling a constructor from within a member function, (2) It isn't really a constructor. It is just phrased that way to make C++ happy, (3) Why do I need to make an extra function call anyway? (4) what if I need several variables like this? (5) What if I already have a constructor which takes a (targetclass *)? etc. Nonetheless, this is the method I am currently using, for want of a better one. Can anybody help? Is there a "nice" way to reroute a reference variable? Any and all help would be appreciated. Thanks! Derek Riippa Foster
JeanYves.Cras@cediag.bull.fr (Jean-Yves Cras) (09/19/90)
I currently have the same problem as you. However, not for the same purpose.
The point is that I want to define a class - say CLASS - and quite a lot
of functions that will receive arrays of references to CLASS as argument.
So I wish I could write:
class CLASS { ... };
void foo(CLASS&* p) { ... } // illegal; wonder why
main()
{
CLASS tab[10];
CLASS& tref[3];
for ( int i = 0; i < 3; i++)
tref[i] ??????? tab[i*2]; // re_referencing operator
f(tref);
}
This doesnot work because
1) Cannot declare pointers/references to references; and
2) lack a re_referencing operator.
Of course, using pointers fixes the problem but...
Anyway, I will certainly not refrain from using your #2 kludge until something cleaner is exhibited !