- 在线时间
- 0 小时
- 最后登录
- 2005-9-21
- 注册时间
- 2004-4-27
- 听众数
- 1
- 收听数
- 0
- 能力
- 0 分
- 体力
- 1027 点
- 威望
- 0 点
- 阅读权限
- 40
- 积分
- 385
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 153
- 主题
- 43
- 精华
- 0
- 分享
- 0
- 好友
- 0
升级   28.33% 该用户从未签到
国际赛参赛者
 |
< >不知道大家有没有我这种体验.大家先看看下面这段代码:</P>: ^- q0 ^) l/ }# l/ [8 [. s. g
< >int fp=_wopen(L"Hello.dat",O_BINARY | O_CREAT | O_TRUNC | O_RDWR);4 k( }/ R/ `0 W: H) K
if(fp==-1) return;5 k# Q6 M4 @& M1 |
write(fp,L"123中国人",wcslen(L"123中国人"));( ]0 V& O$ E+ @, h
close(fp);
) ~" f1 D* Z1 [+ t G4 E; Q/ G9 h/ j7 ]2 Z
上面这段代码不知道大家看出什么BUG来了.如果大家看不出毛病也不足为怪,因为这是我们的习惯导致了我们的错误产生./ |7 q" J5 g7 }9 y2 \. z- h1 ]* ^
, k& h7 x) z& P8 B$ B: @9 Y2 x* U" v
先让我来分析一下write吧.下面是write的原型:
+ ~9 }- l: i5 Q/ O% P2 uint write( int <I>handle</I>, const void *<I>buffer</I>, unsigned int <I>count</I> );
) Z, \- e& L8 d+ z2 z! }
4 S# \6 Y6 l+ N- @5 Y9 [; I参数:4 ]0 i9 H1 y' P [- Q/ r
handle 已打开或已创建的文件句柄' h1 g( d& a/ ` s* z
buffer 待写入的数据
9 `- g9 `' [1 T3 F! ocount 待写入的数据大小9 ]& I) Y6 K2 V& N' O" g. {6 S
& q# V, V1 B& H1 Z) x2 h现在分析为什么上面的那代码有bug,其实主要问题就在一个buffer,和count.9 b+ g6 x# f3 ~/ _' \, k
如果我们写入一个Ansi字符串,上面的代码改成相应的形式确实没有错.# u) c' i' @; o d
但如果是写入一个宽字符串,那么上面的代码就不严格.原因就在于count.
# o3 Q- h! ?8 ^: [5 y# A0 t1 J( O r4 G0 F7 B& K' }+ | `$ J' _
我们首先看一下strlen和wcslen,如果使用strlen,一般情况下,我们直接作为字符串的长度,+ x1 S" x& {/ S6 M
而使用wcslen,你会发现,得出的不是字符串的长度而是字符的个数.6 E' K0 s/ T3 z' r
/ q- K0 x4 ~2 X4 z, X8 [6 R, d0 b8 s这就是问题的所在.一般情况下.char的长度是1,这是用sizeof(char)运算出来的结果.$ O' {. H8 g! I1 l, F. x% _# N8 u
len=strlen(str)*sizeof(char);而我们一般情况下,都只用strlen(str)来等价,这就是平时的习惯.5 u! H: q' [. n9 r9 ~
正是由于这个习惯所引来的问题,这个习惯并不适用于宽字符串.因为wcslen(str)*sizeof(wchar_t)并不等于wcslen(strl).这就是习惯所引起的错误.$ X2 d' l- g* i/ s/ u% X
z8 {& m! z: z) a. `9 ?6 u
说到这里我想大家都明白了.我在这里把这种习惯称之为不良习惯.所以大家以后在计算字符串长度的时候,千万不能简而简之,一定要len=strlen(str)*sizeof(char),len=wcslen(str)*sizeof(wchar_t).' x6 ^0 T9 D4 a# S5 u3 [! v
不要再犯这种习惯性的低级错误.(在下就犯了这种错误.大家都可怜我吧!).</P>
p& v+ {; r+ t4 h& ?$ Y$ J6 H[此贴子已经被作者于2004-11-26 9:50:40编辑过] |
zan
|