数学建模社区-数学中国

标题: C++中几个比较不常用的关键字 [打印本页]

作者: xShandow    时间: 2005-1-6 18:20
标题: C++中几个比较不常用的关键字
<><FONT face=宋体>mutable关键字</FONT> </P>
* o& T$ R# [! g<>) w5 e3 @, f% S( L5 E" S
<FONT face=宋体 size=2>关键字mutable是C++中一个不常用的关键字,他只能用于类的非静态和非常量数据成员6 K2 H" w6 X8 ^4 ~0 ^( z: f" X
我们知道一个对象的状态由该对象的非静态数据成员决定,所以随着数据成员的改变,, C! T4 G$ K0 ?5 ~# i  A
对像的状态也会随之发生变化!</FONT></P>
" e% i6 M9 E: B! f  \- p6 `9 U; C3 S<><FONT face=宋体 size=2>如果一个类的成员函数被声明为const类型,表示该函数不会改变对象的状态,也就是
. b4 b. ]) ]/ }该函数不会修改类的非静态数据成员.但是有些时候需要在该类函数中对类的数据成员
. Y$ ], f& F! q3 c8 c进行赋值.这个时候就需要用到mutable关键字了</FONT></P>
( L" z7 ?7 e+ I# F1 A/ l<><FONT face=宋体 size=2>例如:
& G8 h) K: |& l: ]class Demo
0 i, h+ v# X, r0 m4 I1 \  m{( M- C: `6 w6 v4 f: R$ q
public:
8 i- k5 p: Q& e5 K9 ?    Demo(){}
+ ?1 ?$ z: f4 @2 f6 X. Z    ~Demo(){}; Y* R% A3 z% d8 g& B3 D
public:% O  b% K* @' v* s# a& I. d
    bool getFlag() const
4 O/ A4 W- V: P    {  ]7 v/ D: z. E, _, R5 |4 ~: d2 N
        m_nAccess++;; y  p* \( j  [/ s
        return m_bFlag;
0 X2 S, d6 X9 i8 u) i7 j    }4 [) I5 u* o9 w4 b8 I1 y
private:2 y  u$ j" }* I! f7 L+ s9 G( O, U
    int  m_nAccess;9 P" m' z2 e/ o  q& r
    bool m_bFlag;
% j8 b8 }3 l6 Y, F};</FONT></P>) }% F: P8 \+ |- c! X( ?. y
<><FONT face=宋体 size=2>int main()% r* p2 z6 ~8 m# R8 D! H1 R
{
4 S4 o4 c9 q6 A$ V/ v    return 0;6 m1 G+ O6 S+ P
}</FONT></P>
# L, r1 T/ T! ?" N( z5 x3 C<><FONT face=宋体 size=2>编译上面的代码会出现 error C2166: l-value specifies const object的错误
' q- a5 N1 T5 q- r" y说明在const类型的函数中改变了类的非静态数据成员.</FONT></P>8 |+ k0 E% c/ n3 I, d
<><FONT face=宋体 size=2>这个时候需要使用mutable来修饰一下要在const成员函数中改变的非静态数据成员
6 |- B/ A0 n1 S4 J2 \9 |m_nAccess,代码如下:</FONT></P>* s; s0 C  q! h( E, E/ Y, n  o' y
<><FONT face=宋体 size=2>class Demo
/ E& M' |, x+ i9 k/ Y{$ h$ F3 @2 e9 d) h5 ?
public:+ g8 n% ^; W1 U) y( ]
    Demo(){}2 b. ]& X1 D1 l+ [) ]) k6 O
    ~Demo(){}; r/ A9 ~3 m2 }6 x% i
public:
4 Q0 l% u$ J1 @( B' ]' n& O    bool getFlag() const
* j$ v) Z9 P5 x0 i/ \9 J5 A    {, y0 H+ E1 P9 z* A' ]) t: F
        m_nAccess++;
6 n: I8 e, p  z        return m_bFlag;
& N5 X- ?* a, S- u3 P. G    }& [: l# U4 @6 l7 W4 I  |6 `
private:
5 ^. o" Q! I! C3 J6 K    mutable int  m_nAccess;
6 w$ u6 S  j: ]! ?2 V    bool m_bFlag;
, g8 O5 r# {3 k) J' _( _3 S};</FONT></P>* z. e! z. ]# X+ Y
<><FONT face=宋体 size=2>int main()! B+ |5 |3 |  \6 _. X; k( t% M
{1 J* J! M2 W. ~' o) K/ }" k
    return 0;+ O7 t1 O/ Q6 \) v! x5 b
}</FONT></P>, g  N3 V) o# S9 I: g
<><FONT face=宋体 size=2>这样再重新编译的时候就不会出现错误了!</FONT></P>
  i; i! Y" x, r3 C% m7 s- M  r<><FONT face=宋体 size=2> </P>+ M2 {: y9 F1 F' o9 a1 A
<><FONT face=宋体 size=2> </P>0 ~3 c0 R- l; ~- \5 i/ f' {+ M
<><FONT face=宋体 size=2>volatile关键字</FONT></P>
8 q  R* A/ Q8 t; F<><FONT face=宋体 size=2>volatile是c/c++中一个鲜为人知的关键字,该关键字告诉编译器不要持有变量的临时拷贝,它可以适用于基础类型' p& ?5 |2 i) a! z0 \* S1 X
如:int,char,long......也适用于C的结构和C++的类。当对结构或者类对象使用volatile修饰的时候,结构或者0 k' y3 e& Y6 u% x; Q% Q2 k  g- M
类的所有成员都会被视为volatile.</FONT></P>
& _. S( i! {0 H<><FONT face=宋体 size=2>使用volatile并不会否定对CRITICAL_SECTION,Mutex,Event等同步对象的需要* s8 m: B) I1 K* h
例如:
, h! X4 y7 B/ W; R' a$ t- h0 B3 Tint i;
* n) n5 i5 V0 M, M2 |( f- ni = i + 3;9 }- J9 z' G) M7 K
无论如何,总是会有一小段时间,i会被放在一个寄存器中,因为算术运算只能在寄存器中进行。一般来说,volatitle+ m& V. |( [1 M
关键字适用于行与行之间,而不是放在行内。</FONT></P>3 P, y) v, T: c+ T8 W9 }( @1 j
<><FONT face=宋体 size=2>我们先来实现一个简单的函数,来观察一下由编译器产生出来的汇编代码中的不足之处,并观察volatile关键字如何修正
: p' T" J  x/ N这个不足之处。在这个函数体内存在一个busy loop(所谓busy loop也叫做busy waits,是一种高度浪费CPU时间的循环方法)</FONT></P>
5 s. _+ o; c: R2 b  G/ ^7 Y2 e( Z<><FONT face=宋体 size=2>void getKey(char* pch)
: U2 e! C1 L* \& N{
6 X. b0 @! ?) o' f* f* n9 g while (*pch == 0)
* S5 Y; u0 R1 ?  ;
. X4 D& H7 [: e: ]1 c0 R}</FONT></P>& Y1 C- f7 o8 t7 |
<><FONT face=宋体 size=2>当你在VC开发环境中将最优化选项都关闭之后,编译这个程序,将获得以下结果(汇编代码)/ m3 N! e  y0 V
;       while (*pch == 0)
% w- |0 W+ m1 V( L0 f# p* l8 g$L27
  |& `2 S/ n" c ; Load the address stored in pch
# z$ ~* J9 z' e6 e. c mov eax, DWORD PTR _pch$[ebp]
/ |* x6 m5 O+ p% ?3 K ; Load the character into the EAX register
; m3 z$ L( {. `7 O- t$ D* x movsx eax, BYTE PTR [eax]+ w# L! m5 [  A: S/ W
; Compare the value to zero
$ F9 s1 |1 G) _0 p  J test eax, eax
( a0 A1 ~$ T. M' q. D# o! d$ { ; If not zero, exit loop
( P  t: t9 u/ B4 G3 S9 l& x jne $L28) E7 C- c. C8 X4 t, F9 i
;
; ?! U3 C% A& A' b$ R jmp $L27' o& j+ y% Q7 F/ V7 O
$L286 w, l1 {8 Z4 j8 ?5 ~4 Q
;}</FONT></P>  H' Z6 x$ d5 G9 `  `
<><FONT face=宋体 size=2>这段没有优化的代码不断的载入适当的地址,载入地址中的内容,测试结果。效率相当的低,但是结果非常准确</FONT></P>
# w4 R# }* E. R& H<><FONT face=宋体 size=2>现在我们再来看看将编译器的所有最优化选项开关都打开以后,重新编译程序,生成的汇编代码,和上面的代码' T' w7 R: c: B0 ^6 f* R
比较一下有什么不同! p8 U# g8 w6 `: d
;{
/ ]6 S) G* v& F3 I, G, i ; Load the address stored in pch
. v8 N8 K* g$ S, a1 ? mov eax, DWORD PTR _pch$[esp-4]
# K4 W) U  k7 A ; Load the character into the AL register  U/ V0 |8 t8 p; S1 V6 R2 z
movsx al, BYTE PTR [eax]
5 k8 O1 ?/ k9 p' f) H; while (*pch == 0)
. a+ h9 n7 u4 [; |( y5 q2 N- X ; Compare the value in the AL register to zero
+ |8 v7 U% P. ~, Z1 p; o test al, al2 B# n! [6 {1 R1 k' ]( Y8 [9 \6 \( o2 S
; If still zero, try again
3 L5 z0 F+ j7 m$ W! `+ B  y je SHORT $L84
3 u$ \5 p4 m# r0 r6 Y$ H& { ;
% F3 n8 d! F. B* W5 K;}</FONT></P>
, N8 Z7 R8 }5 c' n% F<><FONT face=宋体 size=2>从代码的长度就可以看出来,比没有优化的情况要短的多。需要注意的是编译器把MOV指令放到了循环之外。这在
5 \' Y4 O, h+ W. M$ c+ d单线程中是一个非常好的优化,但是,在多线程应用程序中,如果另一个线程改变了变量的值,则循环永远不会3 b1 ]; j& |4 G7 W! A4 Q6 o; r
结束。被测试的值永远被放在寄存器中,所以该段代码在多线程的情况下,存在一个巨大的BUG。解决方法是重新7 v# S  q( b' {1 A3 f2 j2 }0 X) v7 M
写一次getKey函数,并把参数pch声明为volatile,代码如下:</FONT></P>. |7 F9 {0 ~- n5 ^) W
<><FONT face=宋体 size=2>void getKey(volatile char* pch)
- E9 Q; W2 L  y' W/ i& I{, o. T  O: [5 f0 ?' g
while (*pch == 0)  K) V" U; F3 G0 j  l! T$ W& S. M
  ;# E& M: ]4 C% l" W( q; m
}</FONT></P>( {* K: t  o0 n3 J
<><FONT face=宋体 size=2>这次的修改对于非最优化的版本没有任何影响,下面请看最优化后的结果:</FONT></P>7 {! ?6 H. l& o
<><FONT face=宋体 size=2>;{
% [# S' H( @) y  j- k ; Load the address stored in pch  K5 c0 \8 N9 o" ]  m) {# l! _2 o2 t9 A7 P
mov eax, DWORD PTR _pch$[esp-4]
! }& }0 g" [' [. @! J;       while (*pch == 0)4 f4 W2 u. {& E2 L0 `
$L84:
' }- y6 {0 Q. e0 G( q9 ^ ; Directly compare the value to zero& \9 W  c2 x  y
cmp BYTE PTR [eax], 0
" h& Y, m6 Z) l2 T5 V( i; z+ w ; If still zero, try again( L% b. [  [/ h/ @3 x. n
je SHORT $L84
- w5 c" ?2 L& w- r% U9 s; v9 d/ W ;' w) r5 L2 D7 N7 x
;}</FONT></P>) ^6 ^" ]/ x* F. l3 {# ^7 n
<><FONT face=宋体 size=2>这次的修改结果比较完美,地址不会改变,所以地址声明被移动到循环之外。地址内容是volatile,所以每次循环8 g3 x3 W# f& r, m9 Y9 h
之中它不断的被重新检查。</FONT></P>
8 S0 Z8 W3 W$ x<><FONT face=宋体 size=2>把一个const volatile变量作为参数传递给函数是合法的。如此的声明意味着函数不能改变变量的值,但是变量的
3 u+ u% d7 M+ k  X! N! s值却可以被另一个线程在任何时间改变掉。</FONT></P>
7 T' ~5 Z+ _/ D3 ]7 g5 E" _9 P+ d$ _, w<>
( A! Q2 g, l& B$ o8 @2 X5 @<FONT face=宋体 size=2>explicit关键字</FONT></P>
3 R) m6 f+ v( [" A! o<>
4 b. _. X$ X9 D8 \4 [<FONT face=宋体 size=2>我们在编写应用程序的时候explicit关键字基本上是很少使用,它的作用是"禁止单参数构造函数"被用于自动型别转换,$ D4 `# K- \9 X9 V
其中比较典型的例子就是容器类型,在这种类型的构造函数中你可以将初始长度作为参数传递给构造函数.
0 H, U5 X8 O1 M3 ?% G例如:1 t9 `* _* r/ b3 b! Y. b
你可以声明这样一个构造函数
- X; r$ \6 ^7 cclass Array) |7 j5 I9 r# v4 f9 n
{2 U' {( ~" u0 x7 {1 l* S
public:. L( @- c' G+ q( J2 V6 r7 l8 `1 G. z
explicit Array(int size);
7 N8 i6 H& I, G5 q ......
$ x: c% M9 S1 }5 X. T};
: A+ }; f9 C+ B2 Z在这里explicit关键字起着至关重要的作用,如果没有这个关键字的话,这个构造函数有能力将int转换成Array.一旦这种
- o$ Y7 o3 z% S5 H% d) m. i情况发生,你可以给Array支派一个整数值而不会引起任何的问题,比如:1 E3 a! m; G" k" K
Array arr;
3 H9 ?4 S* N# v# h, ~& n8 E- S* T...
: o5 v  U6 E1 w3 Sarr = 40;- j+ E5 h% O( Q; Y+ W
此时,C++的自动型别转换会把40转换成拥有40个元素的Array,并且指派给arr变量,这个结果根本就不是我们想要的结果.如果
$ M: K2 G: X- ]. n: v. W我们将构造函数声明为explicit,上面的赋值操作就会导致编译器报错,使我们可以及时发现错误.
2 ?7 G) r: ^# z2 H需要注意的是:explicit同样也能阻止"以赋值语法进行带有转型操作的初始化";
. h1 m! e3 s4 e$ ~4 M( q& V3 S例如:
. k' ?, e" Q- C# d( ?Array arr(40);//正确
0 C' o6 `7 k9 `  ?6 DArray arr = 40;//错误</FONT></P>
) N5 @+ b/ N7 s2 @<><FONT face=宋体 size=2>看一下以下两种操作:% q; u, y$ z3 _/ [$ N) \
X x;
# m1 {2 n$ C0 C; HY y(x);//显式类型转换
" _7 ]" W/ I# h. p1 s' M* ^, I$ ?另一种3 |2 k1 ~( a5 F# I4 c: M
X x;- I" ]$ m: i- [9 Q- J! m
Y y = x;//隐式类型转换</FONT></P>7 ^, A7 z0 A; d: L4 c
<><FONT face=宋体 size=2>这两种操作存在一个小小的差别,第一种方式式通过显式类型转换,根据型别x产生了型别Y的新对象;第二种方式通过隐式转换
+ b) L9 O+ _0 U产生了一个型别Y的新对象.  C1 m1 q, R0 i4 X+ a
explicit关键字的应用主要就是上面所说的构造函数定义种,参考该关键字的应用可以看看STL源代码,其中大量使用了该关键字</FONT></P>' ?4 p7 H4 b# x" R9 Q9 h
<P><FONT face=宋体 size=2> </P>' Y2 q) b' C, H4 y( [! H2 K9 ?! p
<P><FONT face=宋体 size=2>__based关键字</FONT></P>& }5 N: t# \" j3 }
<P>" i: x5 [, @; e$ g
<FONT face=宋体 size=2>该关键字主要用来解决一些和共享内存有关的问题,它允许指针被定义为从某一点开始算的32位偏移值,而不是内存种的绝对位置
- N7 N$ O4 j" d* X$ M举个例子:</FONT></P>* V3 s4 K  h* w$ L; f) e
<P><FONT face=宋体 size=2>typedef struct tagDEMOSTRUCT {9 z: T3 t# I. j& v
int a;. s2 \6 r+ K$ u% C
char sz[10];
8 n/ H7 r2 N8 H" C2 |0 U( o8 M8 \} DEMOSTRUCT, * PDEMOSTRUCT;</FONT></P>. U  n% k$ C/ V
<P><FONT face=宋体 size=2>HANDLE hFileMapping = CreateFileMapping(...);
7 b3 y) D$ x8 V# D9 L! PLPVOID lpShare = (LPDWORD)MapViewOfFile(...);</FONT></P>5 }7 `5 S' L, Q( B4 ^
<P><FONT face=宋体 size=2>DEMOSTRUCT __based(lpShare)* lpDemo;</FONT></P>0 F; u, k( \$ r: r2 U) q: P6 I
<P><FONT face=宋体 size=2>上面的例子声明了一个指针lpDemo,内部储存的是从lpShare开始的偏移值,也就是lpHead是以lpShare为基准的偏移值.
9 \# P0 l6 h3 M! U: h7 i" H; {# ~上面的例子种的DEMOSTRUCT只是随便定义的一个结构,用来代表任意的结构.</FONT></P>% q; g  C2 I6 i* p" ?+ V( x. ~
<P><FONT face=宋体 size=2>虽然__based指针使用起来非常容易,但是,你必须在效率上付出一定的代价.每当你用__based指针处理数据,CPU都必须! @! U4 W- c, o
为它加上基地址,才能指向真正的位置.</FONT></P>; \' ?, J! S& c) \1 \- Y/ r
<P><FONT face=宋体 size=2> </P>
5 [1 n: r. y9 {" R<P><FONT face=宋体 size=2>在这里我只是介绍了几个并不时很常见的关键字的意义即用法,其他那些常见的关键字介绍他们的文章已经不少了在这里
7 K, X+ g- W* x& w" ^* T8 m! z; B就不再一一介绍了.希望这些内容能对大家有一定的帮助!</FONT></P></FONT></FONT></FONT></FONT>
作者: ilikenba    时间: 2005-1-7 01:47
<>好!都是以前不知道的!顶!</P>
作者: pallas1204    时间: 2005-3-13 18:38
ok




欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5