- 在线时间
- 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>
9 p! y/ ]2 o' o: }< >int fp=_wopen(L"Hello.dat",O_BINARY | O_CREAT | O_TRUNC | O_RDWR);
4 N% i u. D0 Y/ Vif(fp==-1) return;# t# B! ]2 R3 s9 O
write(fp,L"123中国人",wcslen(L"123中国人"));
; S. A* h6 H3 Q. _2 zclose(fp);3 i1 |- R3 f4 ^+ }
( n1 J2 D0 i' f8 i/ d- k2 z r上面这段代码不知道大家看出什么BUG来了.如果大家看不出毛病也不足为怪,因为这是我们的习惯导致了我们的错误产生.
3 `* t- X' F* N6 P) G7 |
& w1 O2 U; L: C! P, h先让我来分析一下write吧.下面是write的原型:- g: s0 |3 N4 j8 p
int write( int <I>handle</I>, const void *<I>buffer</I>, unsigned int <I>count</I> );, L2 b& d1 K8 Q) r% n9 x
% E5 L' y0 I" v4 u& D参数:
7 c1 E& \8 x2 K' |4 Z6 E+ k$ s1 P3 ghandle 已打开或已创建的文件句柄* P8 W- |$ v: B6 H8 W* G
buffer 待写入的数据
3 u+ \3 G3 C/ H! c9 bcount 待写入的数据大小/ L( ^5 A4 G E/ [+ z9 e/ r1 r
: t( |1 w- F6 P. {现在分析为什么上面的那代码有bug,其实主要问题就在一个buffer,和count. n) h! C4 Y# k9 U) T
如果我们写入一个Ansi字符串,上面的代码改成相应的形式确实没有错.
9 t W, \( _3 b6 N但如果是写入一个宽字符串,那么上面的代码就不严格.原因就在于count.3 t; l! H) v" }) n$ p6 d
1 q1 P9 y& W# @* q/ C我们首先看一下strlen和wcslen,如果使用strlen,一般情况下,我们直接作为字符串的长度,; F0 |: D; S0 q) k! m! d8 L
而使用wcslen,你会发现,得出的不是字符串的长度而是字符的个数.9 } Q( e; u! i6 j1 l+ d; I
! m9 \$ O( V& r, e% G# \这就是问题的所在.一般情况下.char的长度是1,这是用sizeof(char)运算出来的结果.
- s/ f* t8 b- W) K, X: ~* ulen=strlen(str)*sizeof(char);而我们一般情况下,都只用strlen(str)来等价,这就是平时的习惯.0 a r1 e$ O. j, L& O! z) V
正是由于这个习惯所引来的问题,这个习惯并不适用于宽字符串.因为wcslen(str)*sizeof(wchar_t)并不等于wcslen(strl).这就是习惯所引起的错误.' m8 R9 e& w4 D& v% ?( P
9 u1 Y& b! L' i& D- W$ `说到这里我想大家都明白了.我在这里把这种习惯称之为不良习惯.所以大家以后在计算字符串长度的时候,千万不能简而简之,一定要len=strlen(str)*sizeof(char),len=wcslen(str)*sizeof(wchar_t).
6 F6 ]3 P& |8 J% k( y: C不要再犯这种习惯性的低级错误.(在下就犯了这种错误.大家都可怜我吧!).</P>
+ R3 O5 D/ N) m4 A3 V( O8 P6 c[此贴子已经被作者于2004-11-26 9:50:40编辑过] |
zan
|