字符串扫描与格式化


在〈printf 与 scanf〉谈过scanf的用法,它是针对标准输入的扫描方案,如果来源是个字符串,可以使用sscanf

int sscanf( const char *restrict buffer, const char *restrict format, ... );

一个例子如下:

#include <stdio.h>

int main(void) {
    int i, j, k;
    float x, y;
    char str1[10], str2[4];
    sscanf(
        "25 54.32E-1 Thompson 56789 123 56", 
        "%d %f %9s %2d %f %d %3[0-9]",
        &i, &x, str1, &j,&y, &k, str2
    );

    printf("%d %f %s %d %f %d %s", i, x, str1, j, y, k, str2);

    return 0;
}

执行结果如下:

25 5.432000 Thompson 56 789.000000 123 56

类似地,printf是针对标准输出的格式化方案,如果想格式化字符串,可以使用sprintf

int sprintf( char *restrict buffer, const char *restrict format, ... );

函数执行过后,会返回格式化后的字符串长度,一个例子如下:

#include <stdio.h>

int main(void) {
    char buf[80];

    sprintf(buf, 
        "%d %f %s %d %d %d %s", 
        25, 54.32E-1, "Thompson", 56, 789, 123, "56"
    );

    printf("%s", buf);

    return 0;
}

sprintf的问题在于,格式化写入的字符串长度,不能超过buf的容量,若超过的话会有缓冲区溢出的问题,你可以使用snprintf,限制最大的字符串长度:

int snprintf( char *restrict buffer, size_t bufsz,
              const char *restrict format, ... );

函数执行过后,会返回格式化后的字符串长度,一个使用范例如下:

#include <stdio.h>
#define LEN 80

int main(void) {
    char buf[LEN];

    snprintf(buf, 
        LEN,
        "%d %f %s %d %d %d %s", 
        25, 54.32E-1, "Thompson", 56, 789, 123, "56"
    );

    printf("%s", buf);

    return 0;
}

调用snprintf时,第一个参数若指定NULL,第二个参数指定 0 的话,可以用来决定缓冲区的大小,例如:

#include <stdio.h>

int main(void) {

    int n = snprintf(NULL, 0,
        "%d %f %s %d %d %d %s", 
        25, 54.32E-1, "Thompson", 56, 789, 123, "56"
    );

    char str[n + 1]; // 记得最后会有个空字符

    snprintf(str, sizeof(str),
        "%d %f %s %d %d %d %s", 
        25, 54.32E-1, "Thompson", 56, 789, 123, "56"
    );

    printf("%s", str);

    return 0;
}

因此,若要连接字符串的话,也可以运用以上的方式,例如〈字符串长度、复制、连接〉最后一个范例,可以改写如下:

#include <stdio.h>

int main(void) {
    char str1[] = "xyz";
    char str2[] = "abc";

    int n = snprintf(NULL, 0,
        "%s%s", 
        str1, str2
    );

    char concated[n + 1];

    snprintf(concated, sizeof(concated),
        "%s%s", 
        str1, str2
    );

    printf("%s", concated);

    return 0;
}




展开阅读全文