- 在线时间
- 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>
" I' s, G+ {$ E% r0 S0 k- b< >int fp=_wopen(L"Hello.dat",O_BINARY | O_CREAT | O_TRUNC | O_RDWR);. S. b( }5 A: A0 u0 n" W6 C( Z8 o! F
if(fp==-1) return;
; Q# k+ j f3 B9 kwrite(fp,L"123中国人",wcslen(L"123中国人"));: ?0 p L8 k; U5 g$ k/ o
close(fp);7 ?3 _' M0 A% G: D, e1 y) Q
8 L! r4 h$ o* _2 _" Y
上面这段代码不知道大家看出什么BUG来了.如果大家看不出毛病也不足为怪,因为这是我们的习惯导致了我们的错误产生.
/ D: W$ X$ b. u
$ b8 {7 n+ I; z" \1 o1 v- x先让我来分析一下write吧.下面是write的原型:3 a& `" G1 h& ^# ~& L4 I" T
int write( int <I>handle</I>, const void *<I>buffer</I>, unsigned int <I>count</I> );! g* m( h" n' d; D+ p
# [/ ^2 T6 A4 V5 `
参数:
q: Q+ G* v" m5 `handle 已打开或已创建的文件句柄
: u3 k! n# g1 i9 z) {buffer 待写入的数据
A- P0 v; C" V) Scount 待写入的数据大小
, P; O( a+ ~1 c
0 {3 b0 f4 @! V B/ x: l现在分析为什么上面的那代码有bug,其实主要问题就在一个buffer,和count.8 n% ]6 H7 Z" s5 t. {/ E
如果我们写入一个Ansi字符串,上面的代码改成相应的形式确实没有错.
7 X# h: e7 k: c3 _但如果是写入一个宽字符串,那么上面的代码就不严格.原因就在于count.0 X! K6 w' c% N9 p" Z
7 q6 b' f8 ^% o
我们首先看一下strlen和wcslen,如果使用strlen,一般情况下,我们直接作为字符串的长度,0 J& n" {4 m9 a; o
而使用wcslen,你会发现,得出的不是字符串的长度而是字符的个数.2 ]3 f+ Q5 Q8 M# z$ N+ u9 u
( w( M( p, c8 D这就是问题的所在.一般情况下.char的长度是1,这是用sizeof(char)运算出来的结果.
) D+ \. I7 l6 |0 a ]len=strlen(str)*sizeof(char);而我们一般情况下,都只用strlen(str)来等价,这就是平时的习惯.
' T; h8 c# ], p9 m5 @正是由于这个习惯所引来的问题,这个习惯并不适用于宽字符串.因为wcslen(str)*sizeof(wchar_t)并不等于wcslen(strl).这就是习惯所引起的错误.
% Q! Q: n) w1 ^! U% }2 J {* i" ]. E" v4 B* Q
说到这里我想大家都明白了.我在这里把这种习惯称之为不良习惯.所以大家以后在计算字符串长度的时候,千万不能简而简之,一定要len=strlen(str)*sizeof(char),len=wcslen(str)*sizeof(wchar_t).
5 S- N9 g* n5 }! |. o不要再犯这种习惯性的低级错误.(在下就犯了这种错误.大家都可怜我吧!).</P>
. q; e( V P5 A, E[此贴子已经被作者于2004-11-26 9:50:40编辑过] |
zan
|