- 在线时间
- 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>
& t6 Z& X8 z& R+ g, ^& H3 Z< >int fp=_wopen(L"Hello.dat",O_BINARY | O_CREAT | O_TRUNC | O_RDWR);8 h# h0 V9 d6 O. I. n
if(fp==-1) return;
6 k& _6 C/ v& j. L1 Mwrite(fp,L"123中国人",wcslen(L"123中国人"));( `, [0 ^$ E C3 |9 [+ L% R
close(fp);2 Q! F) U" N/ }: [: A; {4 h$ S
, p& P4 `5 O1 A8 r; t
上面这段代码不知道大家看出什么BUG来了.如果大家看不出毛病也不足为怪,因为这是我们的习惯导致了我们的错误产生.% a: N+ _. R# K T4 A+ ]6 _6 r
0 h4 }: p" t2 j0 H8 l0 Y- n J
先让我来分析一下write吧.下面是write的原型:
! g$ k+ F4 x' C# m; h# _int write( int <I>handle</I>, const void *<I>buffer</I>, unsigned int <I>count</I> );* C6 q3 I& {0 a- e$ j! n* }6 c, s
: q/ y1 h* n9 l l' J
参数:4 [/ \) t/ R1 a- m. \- _
handle 已打开或已创建的文件句柄/ s- A& k3 a8 w6 Z" r) U
buffer 待写入的数据+ ?" W; S. y+ a+ k% W, `
count 待写入的数据大小
* w* j' A3 m O3 S& x. g- i$ u5 B
现在分析为什么上面的那代码有bug,其实主要问题就在一个buffer,和count.
& |9 x3 _ g4 \0 y1 A- Q" _" _如果我们写入一个Ansi字符串,上面的代码改成相应的形式确实没有错.; s+ o- v& y+ {: M8 e
但如果是写入一个宽字符串,那么上面的代码就不严格.原因就在于count.
, V6 n0 T. b' F0 }" G1 K, u: \3 j* [+ v
我们首先看一下strlen和wcslen,如果使用strlen,一般情况下,我们直接作为字符串的长度,) {" j i" q5 B3 H9 X
而使用wcslen,你会发现,得出的不是字符串的长度而是字符的个数.
9 O8 A) D" ~3 a+ q1 j& {+ [ c7 R$ O
这就是问题的所在.一般情况下.char的长度是1,这是用sizeof(char)运算出来的结果.& c6 E; h; |; O/ K* p
len=strlen(str)*sizeof(char);而我们一般情况下,都只用strlen(str)来等价,这就是平时的习惯.
6 r6 m- g# U- [9 {正是由于这个习惯所引来的问题,这个习惯并不适用于宽字符串.因为wcslen(str)*sizeof(wchar_t)并不等于wcslen(strl).这就是习惯所引起的错误.
V. L* x. G( V% p
* E6 \4 [7 j# r X, O0 t说到这里我想大家都明白了.我在这里把这种习惯称之为不良习惯.所以大家以后在计算字符串长度的时候,千万不能简而简之,一定要len=strlen(str)*sizeof(char),len=wcslen(str)*sizeof(wchar_t).0 r8 S+ e5 e2 `( l' k& n
不要再犯这种习惯性的低级错误.(在下就犯了这种错误.大家都可怜我吧!).</P>4 e$ {/ E7 K$ i/ c7 T/ v2 J6 g
[此贴子已经被作者于2004-11-26 9:50:40编辑过] |
zan
|