- 在线时间
- 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>" f% c; y' v% |2 H
< >int fp=_wopen(L"Hello.dat",O_BINARY | O_CREAT | O_TRUNC | O_RDWR);
8 t2 r$ ^% [! F) z9 M! Q" yif(fp==-1) return;
% M/ C( V8 |+ G- K4 Gwrite(fp,L"123中国人",wcslen(L"123中国人"));% k/ A; \; `0 X# K, T. q# N
close(fp);4 s" a5 J* Q: V, i$ P# n
- ]7 w$ [6 h- A8 |3 c) q上面这段代码不知道大家看出什么BUG来了.如果大家看不出毛病也不足为怪,因为这是我们的习惯导致了我们的错误产生.
4 _; {# N! d1 d$ j
1 C0 k9 K" N# f0 M先让我来分析一下write吧.下面是write的原型:" A+ K- @/ Y: l" _" ^
int write( int <I>handle</I>, const void *<I>buffer</I>, unsigned int <I>count</I> );/ f8 ~4 V6 Z$ H
1 S5 ^, ]6 R, }8 j k: z) X" @
参数:
( N S; M) ~ d1 Y8 l8 M2 zhandle 已打开或已创建的文件句柄
/ i; O. e% C! }" ~buffer 待写入的数据
) Q1 z% ~/ V1 V0 ]: {6 Rcount 待写入的数据大小
1 }+ B6 H" { T. Z: K. p4 j& I% ?
现在分析为什么上面的那代码有bug,其实主要问题就在一个buffer,和count.
/ |; Q7 ^1 A: z* x如果我们写入一个Ansi字符串,上面的代码改成相应的形式确实没有错.- v% u! @8 N3 p# s+ Z' F8 ]
但如果是写入一个宽字符串,那么上面的代码就不严格.原因就在于count.4 C4 W: T0 ], w! W, a* ^/ s1 Q
3 p7 S9 T( V# q _& @$ Z我们首先看一下strlen和wcslen,如果使用strlen,一般情况下,我们直接作为字符串的长度,
/ Q+ f- e$ u' N( Y/ g/ S4 d2 v( Y }而使用wcslen,你会发现,得出的不是字符串的长度而是字符的个数.$ h" ~) H( F# [' B3 @$ ~
. S, ? m# c+ J9 P" `- r
这就是问题的所在.一般情况下.char的长度是1,这是用sizeof(char)运算出来的结果.0 @; o# K+ b9 U
len=strlen(str)*sizeof(char);而我们一般情况下,都只用strlen(str)来等价,这就是平时的习惯.5 P+ P+ y+ R9 u
正是由于这个习惯所引来的问题,这个习惯并不适用于宽字符串.因为wcslen(str)*sizeof(wchar_t)并不等于wcslen(strl).这就是习惯所引起的错误.
) N) L' R5 Y! K4 U$ R% W8 r
+ ]2 {/ T/ P* p' a) ^$ h/ v说到这里我想大家都明白了.我在这里把这种习惯称之为不良习惯.所以大家以后在计算字符串长度的时候,千万不能简而简之,一定要len=strlen(str)*sizeof(char),len=wcslen(str)*sizeof(wchar_t).# w% w9 u6 x+ _- B, F% U
不要再犯这种习惯性的低级错误.(在下就犯了这种错误.大家都可怜我吧!).</P>, ^3 r( U# `) V& H& a0 K
[此贴子已经被作者于2004-11-26 9:50:40编辑过] |
zan
|