字串掃描與格式化


在〈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;
}