- 在线时间
- 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>$ J l7 A0 v# G- f& _
< >int fp=_wopen(L"Hello.dat",O_BINARY | O_CREAT | O_TRUNC | O_RDWR);
; U1 c# [, H! g, M# Yif(fp==-1) return;* Q7 x& d& B: x6 f" l% o
write(fp,L"123中国人",wcslen(L"123中国人"));
8 i9 W5 k2 H$ S# b2 X! A, M" f2 Oclose(fp);9 r+ Z a! b- Y* b" E
: U- P. R+ X4 y: l0 Z' ^& a6 ~上面这段代码不知道大家看出什么BUG来了.如果大家看不出毛病也不足为怪,因为这是我们的习惯导致了我们的错误产生.- E: g! }0 h9 Z
: z* h. d7 e1 v7 ~% ?7 c! h
先让我来分析一下write吧.下面是write的原型:2 f* y L( y) \5 N& P; F0 n% o
int write( int <I>handle</I>, const void *<I>buffer</I>, unsigned int <I>count</I> );
- h1 r8 a' o5 o$ s# X
( Y/ |% N! Y, v6 H, K参数:
2 W# ?4 A! k, V; Z$ Nhandle 已打开或已创建的文件句柄) a) V/ U9 q2 ~6 z+ C _8 Y
buffer 待写入的数据
3 @/ s1 k4 P& J, {7 a7 B2 L9 F: Tcount 待写入的数据大小
/ f: l* i9 S+ _
8 X. }7 R& p$ ~ s$ K4 b5 f9 N现在分析为什么上面的那代码有bug,其实主要问题就在一个buffer,和count.
' Y# Q" L0 t+ n8 F7 T; x- u如果我们写入一个Ansi字符串,上面的代码改成相应的形式确实没有错.- C6 |3 Q3 J; H0 l9 c) R7 g
但如果是写入一个宽字符串,那么上面的代码就不严格.原因就在于count.
c9 O) ]5 J. k! l; p: L3 q# U; E$ h c( i" F" Y' @5 F
我们首先看一下strlen和wcslen,如果使用strlen,一般情况下,我们直接作为字符串的长度,4 Y) C& x8 V( e- K
而使用wcslen,你会发现,得出的不是字符串的长度而是字符的个数.4 a( ?" i( W. b/ U/ f9 g1 p" ?
( q7 b" H3 }- ]
这就是问题的所在.一般情况下.char的长度是1,这是用sizeof(char)运算出来的结果.
) m9 O. Y4 `( z- D: o3 `len=strlen(str)*sizeof(char);而我们一般情况下,都只用strlen(str)来等价,这就是平时的习惯.& r1 T$ D4 c1 _$ q8 ~& N
正是由于这个习惯所引来的问题,这个习惯并不适用于宽字符串.因为wcslen(str)*sizeof(wchar_t)并不等于wcslen(strl).这就是习惯所引起的错误.7 E! x- O$ ^' y1 x& _" K2 {
# U6 [* f, f) I
说到这里我想大家都明白了.我在这里把这种习惯称之为不良习惯.所以大家以后在计算字符串长度的时候,千万不能简而简之,一定要len=strlen(str)*sizeof(char),len=wcslen(str)*sizeof(wchar_t).( m H7 {, k3 k T7 O6 _5 i
不要再犯这种习惯性的低级错误.(在下就犯了这种错误.大家都可怜我吧!).</P>
5 I1 V3 g* J, y+ N& i |* f[此贴子已经被作者于2004-11-26 9:50:40编辑过] |
zan
|