C++ Forever

my feeling and C++'s
posts - 5, comments - 12, trackbacks - 0, articles - 0

2008年7月8日

     摘要: 前不久看了 敏捷软件开发 原则、模式与实践,有点想法,所以就把我以前写得一个RedoUndo的框架改了一下,成了下面的情况。我将操作RedoUndo的模块的接口独立了出来,这样在将来的扩展上就不会有太大的困难,并且将他独立与显示模块很大程度上避免了耦合。在使用的时候可以有选择的使用RedoUndo的功能,如果某个工具需要RedoUndo的功能则直接继承与TransactionCommand即可。这...  阅读全文

posted @ 2008-07-08 15:00 HYin 阅读(2510) | 评论 (5)编辑 收藏

2008年6月26日

一、strpbrk:

调用格式:result = strpbrk(source-string, searching-string)
用途:在源字符串(source-string)中找出最先含有搜索字符串(searching-string)中的任一字符的位置并返回,若找不到则返回空指针。
注意:返回值是指向源字符串的指针,所以在使用返回结果的时候,请确保源字符串的有效性。
举MSDN中的例子:
数据:
char string[100] = "The 3 men and 2 boys ate 5 pigs\n";
char *result;
调用:
result = strpbrk( string, "0123456789" );//在string中找出最先出现0~9中某一个字符的位置
printf( "1: %s\n", result++ );
result = strpbrk( result, "0123456789" );
printf( "2: %s\n", result++ );
result = strpbrk( result, "0123456789" );
printf( "3: %s\n", result );
输出的结果是:
1. 3 men and 2 boys ate 5 pigs//最先发现3

2: 2 boys ate 5 pigs//最先发现2,注意这时开始源串是上一次的结果的下一位置,即跳过了数字'3',下同

3: 5 pigs//最先发现5

=========================
二、strtok:

用法:token = strtok(source-str,seperator-str)
用途:在source-str中找出以seperator-str中的字符为分隔的字符串,即是源串中除去了含有分隔串中的所有字符后余下的一段段的字符串,每调用一次找到一串,找不到则返回空串。
注意:返回值是指向源串的指针,所以同样要保证源串的有效性,此外,每次调用返回找到的子串的时候都会把源串中该子串的尾部字符(原来是搜索串中的某一字符)修改成'\0'字符。还有一个要注意的是,这个函数使用全局的静态缓冲区(函数自己的静态buffer),所以在某一个线程里不要同时调用这个函数来处理两个字符串,否则极易出现不正常的结果,正确的处理方法是先找出一个字符串中的所有子串,然后才转到处理另一个字符串。多线程之间则不会出现这种干扰。
同样举MSDN中的例子:
数据准备:
char string[] = "A string\tof ,,tokens\nand some more tokens";
char seps[] = " ,\t\n";
char *token;
调用示例:
token = strtok( string, seps );
while( token != NULL )
{
/* While there are tokens in "string" */
printf( " %s\n", token );
/* Get next token: */
token = strtok( NULL, seps );
/*注意到上面这个NULL,它表明的是从上次调用结果中strtok自有的缓冲区中继续取出余下的子串*/
}
显示结果:
A //此时string[1]='\0',原先为空格
string
of
tokens
and
some
more
tokens

posted @ 2008-06-26 22:16 HYin 阅读(5483) | 评论 (4)编辑 收藏

我在QT中,为了实现一个能够实现字符串字段分析的,花了很长的时间,后来找到了,并且它提供了很多好的功能。
今天上网偶尔发现了这样一个C函数 sscanf(const char *format [,argument]...);他可以进行一些常用的字段分析,并且可以进行过滤,这是我原来没有发现的,用得更多的是scanf(const char *format [,argument]...)。
下面贴出来大家分享下!

 sscanf() - 从一个字符串中读进与指定格式相符的数据.
函数原型:
Int  sscanf( string str, string fmt, mixed var1, mixed var2 ... );
int  scanf( const char *format [,argument]... );
说明:
sscanf与scanf类似,都是用于输入的,只是后者以屏幕(stdin)为输入源,前者以固定字符串为输入源。
其中的format可以是一个或多个 {%[*] [width] [{h | l | I64 | L}]type | ' ' | '\t' | '\n' | 非%符号}
注:
1、 * 亦可用于格式中, (即 %*d 和 %*s) 加了星号 (*) 表示跳过此数据不读入. (也就是不把此数据读入参数中)
2、{a|b|c}表示a,b,c中选一,[d],表示可以有d也可以没有d。
3、width表示读取宽度。
4、{h | l | I64 | L}:参数的size,通常h表示单字节size,I表示2字节 size,L表示4字节size(double例外),l64表示8字节size。
5、type :这就很多了,就是%s,%d之类。
6、特别的:%*[width] [{h | l | I64 | L}]type 表示满足该条件的被过滤掉,不会向目标参数中写入值
  
支持集合操作:

     %[a-z] 表示匹配a到z中任意字符,贪婪性(尽可能多的匹配)

     %[aB'] 匹配a、B、'中一员,贪婪性

     %[^a] 匹配非a的任意字符,贪婪性
例子:
1. 常见用法。
    char buf[512] = ;
    sscanf("123456 ", "%s", buf);
printf("%s\n", buf);

结果为:123456

2. 取指定长度的字符串。如在下例中,取最大长度为4字节的字符串。
    sscanf("123456 ", "%4s", buf);
printf("%s\n", buf);

结果为:1234

3. 取到指定字符为止的字符串。如在下例中,取遇到空格为止字符串。
    sscanf("123456 abcdedf", "%[^ ]", buf);
printf("%s\n", buf);

结果为:123456

4.  取仅包含指定字符集的字符串。如在下例中,取仅包含1到9和小写字母的字符串。
    sscanf("123456abcdedfBCDEF", "%[1-9a-z]", buf);
printf("%s\n", buf);

结果为:123456abcdedf

5.  取到指定字符集为止的字符串。如在下例中,取遇到大写字母为止的字符串。
    sscanf("123456abcdedfBCDEF", "%[^A-Z]", buf);
printf("%s\n", buf);

结果为:123456abcdedf
  
6、给定一个字符串iios/12DDWDFF@122,获取 / 和 @ 之间的字符串,先将 "iios/"过滤掉,再将非'@'的一串内容送到buf中
sscanf("iios/12DDWDFF@122", "%*[^/]/%[^@]", buf);
printf("%s\n", buf);

结果为:12DDWDFF

7、给定一个字符串““hello, world”,仅保留world。(注意:“,”之后有一空格)
sscanf(“hello, world”,  "%*s%s",  buf);   
printf("%s\n", buf);

结果为:world

%*s表示第一个匹配到的%s被过滤掉,即hello被过滤了

如果没有空格则结果为NULL。

sscanf的功能很类似于正则表达式, 但却没有正则表达式强大,所以如果对于比较复杂的字符串处理,建议使用正则表达式.
//-------------------------------------------------------
sscanf,表示从字符串中格式化输入
上面表示从str中,输入数字给x,就是32700
久以前,我以为c没有自己的split string函数,后来我发现了sscanf;一直以来,我以为sscanf只能以空格来界定字符串,现在我发现我错了。
sscanf是一个运行时函数,原形很简单:
int sscanf(
const char *buffer,
const char *format [,
argument ] ...
);
它强大的功能体现在对format的支持上。
我以前用它来分隔类似这样的字符串2006:03:18:
int a, b, c;
sscanf("2006:03:18", "%d:%d:%d", a, b, c);
以及2006:03:18 - 2006:04:18:
char sztime1[16] = "", sztime2[16] = "";
sscanf("2006:03:18 - 2006:04:18", "%s - %s", sztime1, sztime2);
但是后来,我需要处理2006:03:18-2006:04:18
仅仅是取消了‘-’两边的空格,却打破了%s对字符串的界定。
我需要重新设计一个函数来处理这样的情况?这并不复杂,但是,为了使所有的代码都有统一的风格,我需要改动很多地方,把已有的sscanf替换成我自己的分割函数。我以为我肯定需要这样做,并伴随着对sscanf的强烈不满而入睡;一觉醒来,发现其实不必。
format-type中有%[]这样的type field。如果读取的字符串,不是以空格来分隔的话,就可以使用%[]。
%[]类似于一个正则表达式。[a-z]表示读取a-z的所有字符,[^a-z]表示读取除a-z以外的所有字符。
所以那个问题也就迎刃而解了:
sscanf("2006:03:18 - 2006:04:18", "%[0-9,:] - %[0-9,:]", sztime1, sztime2);
在softmse (Jake) 的问题贴http://community.csdn.net/Expert/topic/4843/4843294.xml?temp=. 4321558中 ,周星星给出了一个很cool的sscanf用例,而后通过学习,发现sscanf真棒,现做一总结。
原问题:
iios/12DDWDFF@122
获取/和@之间的字符串怎么做
C程序里面有什么函数吗?
周星星的代码:
#include <stdio.h>
int main()
{
const char* s = "iios/12DDWDFF@122";
char buf[20];
sscanf( s, "%*[^/]/%[^@]", buf );
printf( "%s\n", buf );
return 0;
}
结果为:12DDWDFF
sscanf与scanf类似,都是用于输入的,只是后者以屏幕(stdin)为输入源,前者以固定字符串为输入源。
函数原型:
int scanf( const char *format [,argument]... );
其中的format可以是一个或多个 {%[*] [width] [{h | l | I64 | L}]type | ' ' | '\t' | '\n' | 非%符号},
注:{a|b|c}表示a,b,c中选一,[d],表示可以有d也可以没有d。

width:宽度,一般可以忽略,用法如:
const char sourceStr[] = "hello, world";
char buf[10] = ;
sscanf(sourceStr, "%5s", buf); //%5s,只取5个字符
cout << buf<< endl;
结果为:hello
{h | l | I64 | L}:参数的size,通常h表示单字节size,I表示2字节 size,L表示4字节size(double例外),l64表示8字节size。
type :这就很多了,就是%s,%d之类。

特别的:
%*[width] [{h | l | I64 | L}]type 表示满足该条件的被过滤掉,不会向目标参数中写入值。如:
const char sourceStr[] = "hello, world";
char buf[10] = ;
sscanf(sourceStr, "%*s%s", buf); //%*s表示第一个匹配到的%s被过滤掉,即hello被过滤了
cout << buf<< endl;
结果为:world
支持集合操作:
%[a-z] 表示匹配a到z中任意字符,贪婪性(尽可能多的匹配)
%[aB'] 匹配a、B、'中一员,贪婪性
%[^a] 匹配非a的任意字符,贪婪性
是不是感觉眼熟了啊,不错,这和正则表达式很相似,而且仍然支持过滤,即可以有%*[a-z].如:
星星大哥例子回顾:
const char* s = "iios/12DDWDFF@122";
char buf[20];
sscanf( s, "%*[^/]/%[^@]", buf );
printf( "%s\n", buf );
先将 "iios/"过滤掉,再将非'@'的一串内容送到buf中,cool.得到结果。

posted @ 2008-06-26 22:00 HYin 阅读(635) | 评论 (1)编辑 收藏

2008年5月22日

最近在Qt下的使用了QRubberBand,我要实现在QGLWidget上面画一个选择框。我们在测试的时候发现在WindowsXP的主题下,画出来的选择框是实心的,并且不是透明的。可是在Windows经典主题上,却没有出现这样的问题。
我们尝试想设置透明,可是没有成功,又想到要重新实现自己的一个RubberBand,可是它实际上还是一个QWidget,这样在设置透明的时候就会出现同样的问题,然后我们想就单纯的画四条直线,可是实现起来很复杂。
看有没有高手可不可以帮下忙?

posted @ 2008-05-22 17:32 HYin 阅读(1610) | 评论 (0)编辑 收藏

2008年4月26日

     摘要: 我们网络实验使用socket实现一个ftp客户端,前几天我一个很困惑一些技术上的问题,因为在网上基本上没有这样的样例,所以我将代码贴了上来,不是完整的代码,只有一个核心部分,我想要实现的就是像QFtp一样的功能,不过还有很多地方要完善。 /**//** HLftp.h* \date 2008-04-22* \author HanYin*/#ifn...  阅读全文

posted @ 2008-04-26 09:24 HYin 阅读(5980) | 评论 (2)编辑 收藏