Re: Variables in for loop (style issue)
Walter Bright wrote:
Andrei Alexandrescu (See Website For Email) wrote:
b) Given 'id', can I change something reachable _directly_ by
dereferencing id (say id.x or id->x, depending on the language)?
We'll
see in a minute why "directly" is important. If I can change id.x,
then
I have C++ reference/lvalue/const pointer/nonconst pointer
semantics. If
I cannot change id.x, then I have C++ const object/pointer to
const/reference to const semantics. So the property is whether or
not I
can change the direct members of whatever 'id' refers to.
Ok, but I wish to point out that:
int x;
const int& end = x;
int* p = &const_cast<int&>(end);
*p = 3;
is legal, conforming C++ code. This makes a hash out of any useful
information that can be gleaned from current C++ reference to const
semantics.
Yes, that is correct, and hopefully we'll be able to move away from this
attractor :o).
I agree, and I also want to add the property of whether a reference
can
escape the current scope or not.
That is useful by itself, and helps const too.
Then whatever universe I can reach by transitively dereferencing
id (and
whatever that yields etc.), it's a read-only universe: "transitive
const" specifies an access gate that forbids any change.
To make sure I understand you, is this the "deep const" you
referred to?
Yes.
So we wind up with:
const
readonly
transitive
noalias
unique
auto
volatile
mutable
final
and various combinations of them. The end result is, I fear, mass
confusion on the part of typical programmers instead of the clarity we
were aiming for.
Here's where the language designer's sense should come into play. For
example, noalias is worthless unless you introduce some checkability.
Then, final as a qualifier is really const or something close to it, and
you could remove that. Auto is implicit. Noalias and unique pertain to
aliasing and that's often tricky to deal with statically, so if you're
in a Occam mood you can just as well throw them. Mutable is of marginal
yield. Auto is implicit. So that leaves us with:
const
readonly
transitive
volatile
transitory
You forgot the "can only pass down" transitory qualifier. This is a much
smaller list, and one that gets most bang from the investment.
Furthermore, these types are useful for specifying interfaces, and as
such regular caller code doesn't mess with them that often. Consider:
// Hypothetical language
sort(begin, end, _1 > _2);
In the hypothetical language above, I assume that an expression can bind
to a function type. The expression will only be evaluated if sort
actually does call the function. _1 and _2 refer to the first and second
parameter of that automatically-generated function. By the way, it would
be great if C++ had that, and it's easy.
The corresponding signature of sort is:
template (T)
void sort(
RandomIterator!(T) from,
RandomIterator!(T) to,
bool(const transitive T& a, const transitive T& b) comparitor
);
In this case the user doesn't have to specify that sort takes its third
argument to be a function with two parameters that doesn't squirrel away
references or modify them. Only the library writer deals with that,
and that's the right place to put the burden. The compiler ensures that
the contract is respected, and that's again the right place to put that
other burden.
The nice thing about it all is that should the user try to do something
wrong, the compiler will signal that. The enforcement was caused by the
type qualifiers planted by sort's implementer.
Homework for anyone interested: figure out which automatic
conversions
make sense among these qualifiers. Hint: it's not trivial.
Right, and that adds to the problem of understandability of the
language. That's where I get stuck. I'd like to find a away to make
these properties implicit and natural, rather than explicit. For
example, in Fortran, noalias for arrays is implicit, optimizers take
full advantage of it, and programmers have no problems with
understanding it. But when noalias was added to C, it was a disaster,
because essentially nobody outside the compiler implementors could use
it properly.
The noalias story is peculiar; we shouldn't generalize from it. We also
shouldn't generalize from Fortran, which has much less language devices
that engender aliasing. (Was there anything else besides COMMON?)
Type qualifier deduction is a worthy goal - actually automated deduction
in general is a worthy goal - but in this case comes in tension with the
very purpose of the qualifiers above. Often, qualifiers are planted
_deliberately_ to describe expressive interfaces. Deducing them is good
sometimes, indeed, but you should leave the human the ability to plant
them by hand.
This very situation is in ML: ML deduces the type of a function - you
"just" have to write it. But sometimes (most often in large
applications) you want to specify the function in separation from its
body, and then ML gives you signatures that specify types in their
entirety.
Similarly, I wouldn't mind if my compiler figured out for me what's
const, and take advantage of it to optimize const. But I use const et
al. as documentation, and consequently I either _want_ to plant the
qualifier myself, or to have the compiler somehow tell _me_ (not the
optimizer) about it so I can change my code, or (more far-fetched)
modify my code automatically.
Even today, ask any C programmer what "restrict" means. You'll get a
glazed stare back. I've never seen anyone use it outside of a test
suite
and the C99 standard.
A good example of how an unenforced, unchecked feature (or law in the
real world) would do. Dennis Ritchie was right to oppose this half-baked
concept.
Andrei
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]