I\'m reading some lecture notes of my C++ lecturer and he wrote the following:
- Use Indentation // OK
- Never rely on operator preced
I have my doubts as to the competence of the lecturer. Considering his points:
(b*b) - ((4*a)*c)
?
Some precedences are obvious (or should be), and the extra parentheses
just add to confusion. (On the other hand, you _should_ use the
parentheses in less obvious cases, even if you know that they're not
needed.)
if ( cond ) { code; }and:
if ( cond ) { code; }In the first, I'd agree with him. The opening
{
is not that visible,
so it's best to assume it's always there. In the second, however, I
(and most of the people I've worked with) have no problem with omitting
the braces for a single statement. (Provided, of course, that the
indentation is systematic and that you use this style consistently.
(And a lot of very good programmers, writing very readable code, omit
the braces even when formatting the first way.)
if ( NULL == ptr )
are ugly enough to hinder
readability. Write the comparisons intuitively. (Which in many cases
results in the constant on the right.) His 4 is bad advice; anything
which makes the code unnatural makes it less readable.
int
is reserved for special cases. To
experienced C and C++ programmers, the use of unsigned
signals bit
operators. C++ doesn't have a real cardinal type (or any other
effective subrange type); unsigned
doesn't work for numeric values,
because of the promotion rules. Numerical values on which no
arithmetic operations would make sense, like serial numbers, could
presumably be unsigned
. I'd argue against it, however, because it
sends the wrong message: bitwise operations don't make sense either.
The basic rule is that integral types are int
, _unless_ there is a
significant reason for using another type.
delete this;
is often
the most frequent case (and you can't set this
to NULL
), and
otherwise, most delete
are in destructors, so you can't access the
pointer later anyway. And setting it to NULL
doesn't do anything
about any other pointers floating around. Setting the pointer
systematically to NULL
gives a false sense of security, and doesn't
really buy you anything.
Look at the code in any of the typical references. Stroustrup violates every rule you've given except for the first, for example.
I'd suggest that you find another lecturer. One who actually knows what he's talking about.
It makes your code more readable by clearly defining the scope of your loops and conditional blocks. It also saves you from accidental mistakes.
wrt 6: It's safer because deleteing a null pointer is a no-op. So if you happen to accidentally go through that path twice, you won't cause memory corruption be freeing memory that is either free or has been allocated to something else.
This is most of an issue with static file scope objects and singletons that have not very clear lifetimes and have been known to get recreated after they've been destroyed.
In most cases, you can avoid the need for this by using auto_ptrs
It is best to set the pointer to NULL when you have finished with it.
Here is an example why:
Class A does the following:
Class B does the following
At this point both Class A and Class B have pointers pointing to the same memory block, as far as Class A is concerned this block of memory does not exists because it is finished with it.
Consider the following problem:
What if there was a logic error in Class A which resulted in it writing to memory that now belongs to Class B?
In this particular instance, you will not get an bad access exception error because the memory address is legal, all the while class A is now effectively corrupting class B data.
Class B may eventually crash if it encounters unexpected values and when it does crash, chances are, you will spend quite a long time hunting this bug in class B when the problem is in class A.
If you had set the deleted memory pointer to NULL, you would have gotten an exception error as soon as any logic errors in Class A tried to write to NULL pointer.
If you are worried about the logic error with double delete when pointers are NULL for the second time, then add assert for this.
Also: If you are going to down vote, please explain.
Looking through the answers no one's explicitly stated the sort of practice I make a habit of, telling the story of your code:
int j = 0;
for (int i = 0 ; i < 100 ; ++i)
{
if (i % 2 == 0)
{
j++;
}
}
Becomes:
int j = 0;
for (int i = 0 ; i < 100 ; ++i)
{
if (i % 2 == 0) j++;
}
Putting the j++
on the same line as the if should signal to anyone else, "I only want this block to ever increment j
". Of coursethis is only worthwhile if the line is as simplistic as possible, because putting a breakpoint here, as peri mentions, is not going to be very useful.
In fact I've just run across part of the Twitter Storm API that has this 'sort' of code in java, here is the relvant snippet form the execute code, on page 43 of this slideshow:
...
Integer Count = counts.get(word);
if (Count=null) count=0;
count++
...
The for loop block has two things in it, so I wouldn't inline that code. I.e never:
int j = 0;
for (int i = 0 ; i < 100 ; ++i) if (i % 2 == 0) j++;
It's awful and I don't even know if it works (as intended); don't do this. New lines and braces help distinguish separate but related pieces of code, in the same way a comma or a semi-colon do in prose. The above block is as bad a really long sentence with a few clauses and some other statements that never break or pause to distinguish separate parts.
If you really want to telegraph to someone else it's a one-line only job use a ternary operator or ?: form:
for (int i = 0 ; i < 100 ; ++i) (i%2 ? 0 : >0) j++;
But this is verging on code-golf, and I think not great practice (It's not clear to me if I should put the j++ on one side of the :
or not). NB I've not run a ternary operator in C++ before, I don't know if this works, but it does exist.
Imagine how your reader (i.e. the person maintaing the code) interprets your story (code). Make it as clear for them as possible. If you know the novice coder/student is maintaining this, perhaps even leave in as many {}
as possible, just so they don't get confused.