連接配接字元串
sprintf 的格式控制串中既然可以插入各種東西,并最終把它們“連成一串”,自然也就能夠連接配接字元串,進而在許多場合可以替代strcat,但sprintf 能夠一次連接配接多個字元串(自然也可以同時在它們中間插入别的内容,總之非常靈活)。比如:
char* who = "I";
char* whom = "CSDN";
sprintf(s, "%s love %s.", who, whom); //産生:"I love CSDN. "
strcat 隻能連接配接字元串(一段以'\0'結尾的字元數組或叫做字元緩沖,null-terminated-string),但有時我們有兩段字元緩沖區,他們并不是以'\0'結尾。比如許多從第三方庫函數中傳回的字元數組,從硬體或者網絡傳輸中讀進來的字元流,它們未必每一段字元序列後面都有個相應的'\0'來結尾。如果直接連接配接,不管是sprintf 還是strcat 肯定會導緻非法記憶體操作,而strncat 也至少要求第一個參數是個null-terminated-string,那該怎麼辦呢?我們自然會想起前面介紹列印整數和浮點數時可以指定寬度,字元串也一樣的。比如:
char a1[] = {'A', 'B', 'C', 'D', 'E', 'F', 'G'};
char a2[] = {'H', 'I', 'J', 'K', 'L', 'M', 'N'};
如果:
sprintf(s, "%s%s", a1, a2); //Don't do that!
十有八九要出問題了。是否可以改成:
sprintf(s, "%7s%7s", a1, a2);
也沒好到哪兒去,正确的應該是:
sprintf(s, "%.7s%.7s", a1, a2);//産生:"ABCDEFGHIJKLMN"
這可以類比列印浮點數的”%m.nf”,在”%m.ns”中,m 表示占用寬度(字元串長度不足時補空格,超出了則按照實際寬度列印),n 才表示從相應的字元串中最多取用的字元數。通常在列印字元串時m 沒什麼大用,還是點号後面的n 用的多。自然,也可以前後都隻取部分字元:
sprintf(s, "%.6s%.5s", a1, a2);//産生:"ABCDEFHIJKL"
在許多時候,我們或許還希望這些格式控制符中用以指定長度資訊的數字是動态的,而不是靜态指定的,因為許多時候,程式要到運作時才會清楚到底需要取字元數組中的幾個字元,這種動态的寬度/精度設定功能在sprintf 的實作中也被考慮到了,sprintf 采用“*”來占用一個本來需要一個指定寬度或精度的常數數字的位置,同樣,而實際的寬度或精度就可以和其它被列印的變量一樣被提供出來,于是,上面的例子可以變成:
sprintf(s, "%.*s%.*s", 7, a1, 7, a2);
或者:
sprintf(s, "%.*s%.*s", sizeof(a1), a1, sizeof(a2), a2);
實際上,前面介紹的列印字元、整數、浮點數等都可以動态指定那些常量值,比如:
sprintf(s, "%-*d", 4, 'A'); //産生"65 "
sprintf(s, "%#0*X", 8, 128); //産生"0X000080","#"産生0X
sprintf(s, "%*.*f", 10, 2, 3.1415926); //産生" 3.14"
原文:
http://baike.baidu.com/view/1295144.htm