首页 > 代码库 > 学习笔记(C++Primer)--易错点总结(Chapter2)
学习笔记(C++Primer)--易错点总结(Chapter2)
2.1.2Type Conversions(1/10/2017)
1.If we assign an out-of-range value to an object of unsigned type, the result is
the remainder of the value modulo
the number of values the target type can
hold. For example, an 8-bit unsigned char can hold values from 0 through
255, inclusive. If we assign a
value outside this range, the compiler assigns the
remainder of that value modulo 256.
Therefore, assigning –1 to an 8-bit
unsigned char gives that object the value 255.
2. If we assign an out-of-range value to an object of signed type, the result is
undefined. The program might appear to work, it might crash, or it might
produce garbage values.
3.Advice: Avoid Undefined and Implementation-Defined Behavior
Undefined behavior results from errors that the compiler is not required (and
sometimes is not able) to detect. Even if the code compiles, a program that
executes an undefined expression is in error.
Unfortunately, programs that contain undefined behavior can appear to
execute correctly in some circumstances and/or on some compilers. There is
no guarantee that the same program, compiled under a different compiler or
even a subsequent release of the same compiler, will continue to run
correctly. Nor is there any guarantee that what works with one set of inputs
will work with another.
Similarly, programs usually should avoid implementation-defined behavior,
such as assuming that the size of an int is a fixed and known value. Such
programs are said to be nonportable. When the program is moved to another
machine, code that relied on implementation-defined behavior may fail.
Tracking down these sorts of problems in previously working programs is,
mildly put, unpleasant.
4. The compiler applies these same type conversions when we use a value of one
arithmetic type where a value of another arithmetic type is expected. For example,
when we use a nonbool value as a condition (§ 1.4.1, p. 12), the arithmetic value is
converted to bool in the same way that it would be converted if we had assigned
that arithmetic value to a bool variable:
int i = 42;
if (i) // condition will evaluate as true
i = 0;
If the value is 0, then the condition is false; all other (nonzero) values yield true.
By the same token, when we use a bool in an arithmetic expression, its value
always converts to either 0 or 1. As a result, using a bool in an arithmetic expression
is almost surely incorrect.
5. Caution: Don’t Mix Signed and Unsigned Types
Expressions that mix signed and unsigned values can yield surprising results
when the signed value is negative. It is essential to remember that signed
values are automatically converted to unsigned. For example, in an
expression like a * b, if a is -1 and b is 1, then if both a and b are ints,
the value is, as expected -1. However, if a is int and b is an unsigned,
then the value of this expression depends on how many bits an int has on
the particular machine. On our machine, this expression yields 4294967295
6.2的32次方是4294967296
7.
unsigned u = 10, u2 = 42;
std::cout << u2 - u << std::endl;//32
std::cout << u - u2 << std::endl;//4294967264
int i = 10, i2 = 42;
std::cout << i2 - i << std::endl;//32
std::cout << i - i2 << std::endl;//-32
std::cout << i - u << std::endl;//0
std::cout << u - i << std::endl;//0
2.1.3literals
1.
Although integer literals may be stored in signed types, technically speaking, the
value of a decimal literal is never a negative number. If we write what appears to be
a negative decimal literal, for example, -42, the minus sign is not part of the literal.
The minus sign is an operator that negates the value of its (literal) operand.
2.
3. The type of a string literal is array of constant chars, a type we’ll discuss in § 3.5.4
(p. 122). The compiler appends a null character (’\0’) to every string literal. Thus, the
actual size of a string literal is one more than its apparent size. For example, the
literal ‘A‘ represents the single character A, whereas the string literal "A" represents
an array of two characters, the letter A and the null character.
4. Two string literals that appear adjacent to one another and that are separated only
by spaces, tabs, or newlines are concatenated into a single literal. We use this form of
literal when we need to write a literal that would otherwise be too large to fit
comfortably on a single line:
// multiline string literal
std::cout << "a really, really long string literal "
"that spans two lines" << std::endl;
学习笔记(C++Primer)--易错点总结(Chapter2)