clindh@Stride.COM (Christer Lindh) (06/10/88)
Does anyone know this? I can't find this
information in the C++ book, but I may have missed something:
1)
I want to define overloads to a function Print (used for a interface
to curses), it can either take;
a) one character,
b) a character pointer, or
c) a character pointer *and* a variable number of other arguments
(because I don't want to call the function that uses varargs
unless it needs to be printed with printf)
I came up with this:
// class-definitions and stuff
public:
void Print(char c) { waddch (w,c); }
void Print(char* s) { waddstr(w,s); }
void Print(char* ...); // Function that uses varargs
// end of class
But GNU C++ can't distinguish between the two last declarations, it
thinks that is ambigious. I tried it on AT&T C++ as well, and this
example compiles, but if I remove the first line ( Print(char c) ),
it fails with "two declarations of Print()"! Bug or feature??
The definition for ellipis (r.8.4, C++ Programming Language):
"the number of arguments is known to be *equal to or
greater* than the number of argument types specified"
but what I want is a type of ellipsis that means:
"the number of arguments is known to be *greater*
than the number of argument types specified"
Is there anyway to do this ? I've tried Print(char *, ...) which
makes sense to me but that gave the same result.
2)
In an inline function with variable number of arguments, is there a
way to reference the variable part, ie how can I pass them on to the
function that knows about varargs ? I would like to do something
like this:
// class definitions ...
void Print(char* fmt ... ) { wprintw( w, fmt, ... ); }
^
|
I want to put the rest of
the arguments here, but how
do I name them ?
// end of class
I don't want to mess with varargs in the in-line code as wprintw()
knows how to deal with variable number of arguments. I want:
Print("Foo %d %d %d\n", 1, 2, 3);
to be expanded to this inline code:
wprintw( this->w, "Foo %d %d %d\n", 1, 2 3);
It can't be done with the standard C preprocessor, but I hope C++ is
better.
--
clindh@stride.COM :: The Fittest Shall Survive
Stride Product Group, R&D :: Yet The Unfit May Live.
MicroSage Computer Systems, Inc. ::
Reno, NV, (702) 322-6868 :: We Must Repeat !
english@stromboli.usc.edu (Joe English) (11/28/88)
In the Book, section 8.5.1 (Static Members), p. 275, it states: "No initializer can be specified for a static member, and __it cannot be of a class with a constructor__." I'm not sure if I'm parsing this sentence correctly: does it mean that a class with a constructor can have no static members? Also, if no initializer can be specified, where and how does a static member get initialized? Second question: if automatic temporary objects are created when necessary and are destroyed at the first opportunity (like when a function returns a class object), how does the compiler know when the object is no longer needed? For example, is the following correct: (What I want to happen is for the addresses of the (automatic) frobs created by the calls to frob::frob(int) to be stored in v, and for the frobs themselves to *remain intact* for the life of the function.) class frob { ... public: frob(int) {...} }; class frobpvector { // a vector of pointers to frobs frob *list[MAX]; frob **next; public: frobpvector() { next = list; } void append(frob&); }; void frobpvector::append(frob& f) { *next++ = &f; } void diddle() { frobpvector v; v.append(frob(1)); // frob(1) returns a temporary // ... v.append(frob(2)); // frob(2) should be a *different* frob // ... do stuff with the frobs pointed to in v } In other words, if an automatic temporary object is passed to a function as a reference parameter, _is the space allocated to that temporary object guarranteed to not be overwritten in the current scope_? This is probably in TFM, but I cannot for the life of me find where it is explicitly stated. I need to know for sure, or else I will have to rethink an important class interface. BTW, Zortech C++ seems to be doing (what I feel is) the right thing here. /|/| "How do you convince your dermatologist -----< | | that you're being sexually responsible? O \|\| english%lipari@oberon.usc.edu (Joe English)
shap@polya.Stanford.EDU (Jonathan S. Shapiro) (11/29/88)
In article <13683@oberon.USC.EDU> english@stromboli.usc.edu (Joe English) writes: > In the Book, section 8.5.1 (Static Members), p. 275, it states: >"No initializer can be specified for a static member, and __it cannot >be of a class with a constructor__." I'm not sure if I'm parsing this >sentence correctly: does it mean that a class with a constructor can >have no static members? Also, if no initializer can be specified, >where and how does a static member get initialized? I believe that what this means is that if I say: class Fred { static Member member; ... } where Member is also a class, then Member is not permitted to have a constructor. It is my vague recollection that this may have been relaxed to a lesser restriction: that Member can have a constructor that takes no arguments. Could someone confirm or deny this last bit? Bjarne? > Second question: if automatic temporary objects are created when >necessary and are destroyed at the first opportunity (like when a >function returns a class object), how does the compiler know when the >object is no longer needed? For example, is the following correct: > >[... much of example deleted ...] > >void diddle() > >{ > frobpvector v; > > v.append(frob(1)); // frob(1) returns a temporary > // ... > v.append(frob(2)); // frob(2) should be a *different* frob > > // ... do stuff with the frobs pointed to in v > >} I do not know quite what the scoping rules are for temporaries. I believe that GNU G++, for example, attempts to reuse the space and to peephole them out of existence when possible. In any case, I read "destroyed at the first opportunity" to mean "destroyed as soon as possible." My general reaction to this is that functions that accept an object reference are not entitled to assume that the object continues to exist after the function call. This means that the scope of the temporary above is free to be the temporal scope of the function call itself, and the compiler is free to destroy the object after the function call returns. If the fropvector wants to ensure that it retains a copy of the object, than in C++ it is necessary to make a copy. It also seems to me that if my understanding is correct, then applying foo(X&) to a temporary is for exactly this reason an unreasonable thing to do, and ought to be a compiler error. The compiler ought to apply foo(X), which presumably is smart enough to understand that the behavior must be copying. Until someone indicates what the correct behavior is, the safest bet is to modify your function to read: void diddle() { frobpvector v; Frob f1(1), f2(2); v.append(f1); v.append(f2); ... } As a side observation, this sort of detail is the kind of thing that compilers routinely get wrong or abuse in their optimization phases. It is therefore not a real good thing to rely on. Further, instantiating the temporaries explicitly makes it easier on the reader - it is now obvious that they are temporaries. Jonathan S. Shapiro AT&T Bell Laboratories
mwg@inxsvcs.UUCP (Phil Blecker) (11/29/88)
In article <13683@oberon.USC.EDU>, english@stromboli.usc.edu (Joe English) writes: > > In the Book, section 8.5.1 (Static Members), p. 275, it states: > "No initializer can be specified for a static member, and __it cannot > be of a class with a constructor__." That means that a static member of a class can be an instance of a class. But the static member's class cannot define a constructor. The class containing the static member CAN have a constructor. For instance: class has_constructor { int i; public: has_constructor( int j ) : i( j ) {} }; class no_constructor { int i; public: }; class static_member { static has_constructor h; // won't compile static no_constructor n; // will compile: n.i is uninitialized static int i; // will compile: i is uninitialized public: static_member(); }; Static members are initialized the same way any other static uninitialized data is during startup (usually [hopefully?] to all 0's). > > Second question: if automatic temporary objects are created when > necessary and are destroyed at the first opportunity (like when a > function returns a class object), how does the compiler know when the > object is no longer needed? > This is implementation dependent. When space is allocated automatically, it will not be overwritten during the scope of the object: that is guaranteed. Many compilers allocate space on the stack for ALL automatic objects declared when the function is entered, regardless of their scope (the function's stack space is not enlarged dynamically during the execution of a function). It is up to the compiler to create code that makes sure the object is only referred to within its proper scope. There may be other cases I can't think of right now, but automatic objects must at least be destroyed at the end of the block in which they are declared, or before a return statement causes the function to be exited (or before exiting if the function returns implicitly). If a class has a destructor, then this means that each instance's destructor will be called after calculating the value the return statement will return, but before the function is exited. It also means that the destructor is called before a block ends. So, a function which allocates many automatic objects and has many return statements will have many lines of code code to destroy the automatics (once before each return) if you're using a compiler that translates to C code (actual compilers might use another method that produces far less code). > > (What I want to happen is for the addresses of the (automatic) frobs > created by the calls to frob::frob(int) to be stored in v, and for the > frobs themselves to *remain intact* for the life of the function.) > There's no problem with this, although the method you used only works by accident in my opinion: > > v.append(frob(1)); // frob(1) returns a temporary > Basically what you have done is created a frob instance within the scope of a block that is a function call. Technicaly, the object should only exist for as long as it takes to call v.append(), and then be destroyed. Since there is no destructor associated with a frob, destroying the object does nothing; and since most compilers allocate space for the object when the function is entered, and deallocate the space when the function is exited, you are left with a side-effect that the object still exists even though it should have been destroyed. You can dramatically see the effects of this unless you put your code inside a loop. The compiler I have interprets the code you supplied as indicating that you have allocated actual stack space for only 2 frob objects, which are constantly reinitialized (by calling the frob constructor) because of the fact that a function call can be considered as block. In this case your vector would end up getting many, many pointers to the same 2 frob objects which are hanging around much longer than they should due to the implementation of the compiler. Thus the value of the objects pointed to in your array will vary depending on the current number of iterations through the loop; probably not what you expected. It would probably be better to use new to allocate the frobs, and design an interface that expects ~frobpvector to delete the frobs stored in the vector. Either that or at least include a warning to users of frobpvector. Just to make all these words easier to understand, I wrote an example using the code you originally supplied: class frob { int i; public: friend ostream &operator<<( ostream &o, frob &f ) { return o << f.i; } frob(int j) : i(j) {} ~frob() { cout << "~" << i << " "; } }; const int MAX=8; class frobpvector { // a vector of pointers to frobs frob **next; public: frob *list[MAX]; frobpvector() { next = list; } void append(frob&f) { *next++ = &f; } }; int main() { frobpvector v; for ( int i = 0; i < 6; ) { v.append(frob(i++)); v.append(frob(i++)); frob **next = v.list; for ( int j = i; j--; ) { cout << *(*next++) << " "; } cout << "\n"; } return 0; } and got the following (which is what I expected to happen): ~1 ~0 0 1 ~3 ~2 2 3 2 3 ~5 ~4 4 5 4 5 4 5 Note that the destructor is called twice each iteration of the loop (right after v.append()), and that the same memory space is used over and over again. I put the destructor in to graphically show that the space is only guaranteed to exist during the scope of the v.append() function. But the compiler, rather than dynamically allocating space on the stack, allocates space only once (which makes a lot of sense to me -- dynamically allocating stack space would be an extremely inefficient use of a CPU's time in this or any other case -- that's what new and delete are for -- to give the programmer control over dynamically allocated memory). If frob contained pointers to something, this could cause incredible havoc (especially is the pointers were to memory from new). If you really want to do things the way you showed, allocate the frob objects before the call to v.append(): frob f1( 1 ); v.append( f1 ); That will extend the scope to the block in which f1 is declared, which must be the entire block in which v is to be used. Hope that clears things up a bit -- it wasn't easy to describe briefly. If you have any further questions, feel free to contact me via e-mail or by phone (+1 818-243-3053). -- Phil Blecker none of my ideas belong to me and uunet!inxsvcs!mwg i can't see anything wrong with that