思考
在C语言这种静态语言处理可变长的字符串往往是个令人头大的事。是不是就没办法处理这种令人头的字符串呢?答案应该是否定的!这种令人头痛的东西,那些研发人员早就注意到,应该不会这么长时间没有一个解决方案。
查找
俗话说的好,“世上无难事,只怕有心人”。在查阅《21st Century C》时候,找到了处理这种令人头痛的字符串的方法。
书中在例 9-3 中运用了C语言的可变参数宏的特性和asprintf函数实现了一个可输入变长字符串的字符串拓展宏。
代码
#include
#include //free
//Safer asprintf macro
typedef char* string;
#define str_extend( wt , ... ) { \
string tmp = ( wt ); \ // 记录字符串原地址
if(asprintf(&(wt), __VA_ARGS__) < 0) abort(); \ // 更新变量 wt 地址
if(tmp) { free(tmp);tmp=NULL; } \ // 释放原地址数据
}
int main(){
int i = 3;
string q =NULL; // 指针初始化置NULL值
str_extend(q,"select * from tab\n");
str_extend(q, "%swhere col %d is not null\n", q ,i);
printf("%s", q);
free(q);
q=NULL;
}
结果
解释
首先这个宏有两个比较特别的地方:
- 使用了可变参数宏。
- 使用了asprintf函数。
可变参数宏
这个相关解释可以看另一篇博客
关于C语言如何实现默认参数的解决方法
asprintf()函数
asprintf 函数分配必要的字符串空间并填充字符串,如果内存已满返回-1,属于动态内存,需要手动释放。
注意:传给 asprintf 函数的字符串 应该 预先进行基本的字符串安全性检查。
提示
我们在使用C语言字符串时候,大部分情况字符串长度都不是固定,这个可变参数宏方法可以很好的解决变长字符串问题。
有时候我们需要接受一个字符串的情况,一般如下:
char dtm[100];
strcpy( dtm, "Saturday March 25 1989" );
这样做不仅直接限制了dtm长度,还容易使得字符串操作越界。如果直接定义一个 char * 的变量,那么strcpy就无法使用,运行直接奔溃。
常量字符串又不是我们需要处理的,而且我们大部分时候也不是想要一个常量,也不想去管这个字符串具体的长度(但是 我们心里要对操作的字符串长度有点数,别太过于大,当然这种情况一般比较少见)。
这个时候 asprintf 函数的优势就体现出来了,它一不需要传入字符串长度参数,二只需要一个普普通通的 char * 变量作为接收者即可。
这时上面的情况我们就可以如直接调用示例代码中的str_extend() 宏来完成这个任务,如下:
char *dtm =NULL;
str_extend(dtm,"Saturday March 25 1989");
非常简洁明了且舒服~!
当然,我们也可以使用这种方法实现类似 strcpy 函数的功能。
如果只是初始化并赋值一个字符串,又不想其被编译器解释为一个常量(定die了的那种),那么可以使用以下这利用asprintf函数制作出如下strdup函数来避免这种情况:
代码
#include
#include
typedef char* string;
string strdup(const string in){
if(!in) return NULL;
string out;
if(asprintf(&out,"%s", in) < 0) {
printf("Memory is full!\n return NULL!");
return NULL;
}
return out;
}
int main(){
string q=strdup("爱你呀!\n"); // 初始化赋值一步走,得到字符串变量
printf("%s",q);
free(q); // 因为strdup函数创建动态分配内存
q=NULL;
}
结果
通过 strdup 函数创建出来的字符串就是遵循正常字符串变量使用逻辑,而非常量。
总结
在字符串处理任务中,变量字符串的处理往往是一件令人头的事情。现在,我们有了 asprintf 函数,就可以轻松愉快的处理这些麻烦的问题。因为这种方法会涉及指针的使用,所以不要忘记给自己“擦屁股”。
注意:
文章为本人在平台原创,盗用必究!