字符函數和字符串函數
C語言中對字符和字符串的處理非常頻繁,可是C語言自己是沒有字符串類型的,字符串一般放在常量字符串中或者字符數組中。字符串常量適用於那些對它不作修改的字符串函數.
1、字符串操做函數
1.求字符串長度
1.1 strlen
size_t strlen(const char *str);
- 字符串以 ‘\0’ 做爲結束標誌,strlen函數返回的是在字符串中 ‘\0’ 前面出現的字符個數。
注意:這裏不包含’\0’,例如"abc",使用strlen返回的值就是3。 - 參數指向的字符串必需要以 ‘\0’ 結束。
注意:函數的返回值爲size_t,是無符號整數( 易錯 ),舉個例子:
int main() { const char*str1 = "abcdef"; const char*str2 = "bbb"; if(strlen(str2)-strlen(str1)>0) { printf("str2>str1\n"); } else { printf("srt1>str2\n"); } return 0; }
結果
緣由:strlen返回值爲無符號整形,所以不會出現負數狀況,也就不會出現小於0的狀況。
git
2. 長度不受限制的字符串
2.1 strcpy
char * strcpy(char * destination, const char * source);
- 將source指向的C字符串複製到目標指向destination的數組中,包括’\0’。
注意: - 源字符串必須以’\0’結束。
- 會將字符串中的’\0’拷貝到目標空間。
- 目標空間必須足夠大,以確保能存放源字符串。
- 目標空間必須可變。
- source不會被修改,只能被讀取,因此使用const。
2.2 strcat
char * strcat(char * destination, const char * source);
- 將源的第一個數字字符追加到目標,加上一個終止的空字符
- 若是源中的C字符串長度小於num,則僅複製直到終止空字符的內容。
- 源字符串必須以 ‘\0’ 結束。
- 目標空間必須有足夠的大,能容納下源字符串的內容。
- 目標空間必須可修改。
2.3 strcmp
int strcmp(const char * str1, const char * str2 );
如何比較兩個字符串大小?
該函數比較每一個字符串的第一個字符。若是它們相等,繼續比較第二個字符,直到字符不一樣或’\0’字符爲止。字符串的大小與長度無關。
字符比較的原則爲ascll表的碼值,以下圖所示。例如:「abc」,「abd」,前兩個字符a,b的ascll碼值相同,比較c和d,能夠得出c=99<d=100,所以"abd">「abc」。
這裏須要注意的是strcmp返回值:
windows
- 標準規定:
- 第一個字符串大於第二個字符串,則返回大於0的數字
- 第一個字符串等於第二個字符串,則返回0
- 第一個字符串小於第二個字符串,則返回小於0的數字
例子:
const char * str1 = "abc"; const char * str2 = "abc"; if (strcmp(str1, str2)) { printf("相等\n"); } else { printf("不相等\n"); }
結果:
緣由:若是兩個字符串相等,strcmp的返回值爲0。正確寫法以下:
數組
const char * str1 = "abc"; const char * str2 = "abc"; if (strcmp(str1, str2)==0) { printf("相等\n"); } else { printf("不相等\n"); }
結果:less
3.長度受限制的字符串函數介紹
3.1 strncpy
char * strncpy(char * destination, const char * source, size_t num);
將源的前幾個字符複製到目標。若是源C字符串的末尾(由空字符表示)是在複製數字字符以前找到的,目標將被填充零,直到總共有數字字符被寫入其中。
拷貝num個字符從源字符串到目標空間。
若是源字符串的長度小於num,則拷貝完源字符串以後,在目標的後邊追加0,直到num個。
函數
3.2 strncat
char * strncat( char * destination, const char * source, size_t num);
將源的前幾個字符追加到目標,加上一個終止的空字符。若是源中的C字符串長度小於num,則僅複製直到終止空字符的內容。spa
注意: strncpy不會自動添加’\0’,而strncat會自動添加’\0’。
例子:
3d
char str1[20]; char str2[20]; strcpy(str1, "To be "); strcpy(str2, "or not to be"); strncat(str1, str2, 6); puts(str1);
結果:
這裏須要瞭解puts函數
指針
int puts ( const char * str );
C語言默認有三個輸入輸出函數code
- stdin:標準輸入->鍵盤
- stdout:標準輸出->顯示器
- stderror:標準錯誤->顯示器
puts與printf區別:blog
puts("hello world %d\n"); printf("hello world %d\n", 10);
puts默認帶一個"\n"。
3.3 strncmp
int strncmp(const char * str1, const char * str2, size_t num);
比較到出現另個字符不同或者一個字符串結束或者num個字符所有比較完。
例子:
char str[][5] = { "R2D2","B7UD","R2O9" }; int n; puts("Looking for R2 astromech droids..."); for (n = 0; n < 3; n++) { if (strncmp(str[n], "R2xx", 2) == 0) { printf("founds %s\n",str[n]); } }
結果:
4.字符串查找
4.1 strstr
const * strstr( const char *, const char *);
返回指向str1中第一個出現的str2的指針,若是str2不是str1的一部分,則返回空指針
例子:
char str[] = "This is a simple string."; char *pch; pch = strstr(str, "simple"); strncpy(pch, "sample", 6); puts(str);
結果:
4.2 strtok
char * strtok( char * str, const char * sep)
- sep參數是個字符串,定義了用做分隔符的字符集合。
- 第一個參數指定一個字符串,它包含了0個或者多個由sep字符串中一個或者多個分隔符分割的標記。
- strtok函數找到str中的下一個標記,並將其用 \0 結尾,返回一個指向這個標記的指針。(注:strtok函數會改變被操做的字符串,因此在使用strtok函數切分的字符串通常都是臨時拷貝的內容而且可修改。)
- strtok函數的第一個參數不爲 NULL ,函數將找到str中第一個標記,strtok函數將保存它在字符串中的位置。
- strtok函數的第一個參數爲 NULL ,函數將在同一個字符串中被保存的位置開始,查找下一個標記。
- 若是字符串中不存在更多的標記,則返回 NULL 指針。
注意:strtok是用來進行字符串分割的,首次調用須要傳入字符串起始地址,後續調用設爲NULL便可,調一次拿一個字符串。
例子:以@和.分隔字符串
char *p = "kyrieleon@gmail.com"; const char* sep = ".@"; char arr[30]; char *str = NULL; strcpy(arr, p);//將數據拷貝一份,處理arr數組的內容 for (str = strtok(arr, sep); str != NULL; str = strtok(NULL, sep)) { printf("%s\n", str); }
結果:
5.錯誤信息報告
5.1 strerror
char * strerror(int errnum);
調用該函數必須包含頭文件
#include<errno.h>
返回錯誤碼,所對應的錯誤信息。
例子:查看windows下的錯誤碼對應的信息
for (int i = 1; i < 50; i++) { printf("%d->%s\n", i, strerror(i)); }
結果:
2、字符操做
1.字符分類函數
字符操做判斷的爲單個字符的結果,須要與字符串區分,不要混淆。
函數 | 若是他的參數符合下列條件就返回真 |
---|---|
iscntrl | 任何控制字符 |
isspace | 空白字符:空格‘ ’,換頁‘\f’,換行’\n’,回車‘\r’,製表符’\t’或者垂直製表符’\v’ |
isdigit | 十進制數字 0~9 |
isxdigit | 十六進制數字,包括全部十進制數字,小寫字母af,大寫字母AF |
islower | 小寫字母a~z |
isupper | 大寫字母A~Z |
isalpha | 字母az或AZ |
isalnum | 字母或者數字,az,AZ,0~9 |
ispunct | 標點符號,任何不屬於數字或者字母的圖形字符(可打印) |
isgraph | 任何圖形字符 |
isprint | 任何可打印字符,包括圖形字符和空白字符 |
例子:判斷字符c是否爲空格
char c = ' '; if (isspace(c)) { printf("c爲空格\n"); }
2.字符轉換
int tolower ( int c ); int toupper ( int c );
例子:將字符串中大寫字母轉化爲小寫字母
int i = 0; char str[] = "Test String.\n"; char c; while (str[i]) { c = str[i]; if (isupper(c)) c = tolower(c); putchar(c); i++; }
3、內存操做函數
注意:內存拷貝操做的基本單元是字節,與類型無關。 實例見memset操做函數的例子
1.memcpy
void * memcpy (void * destination, const void * source, size_t num)
- 函數memcpy從source的位置開始向後複製num個字節的數據到destination的內存位置。
- 不一樣於字符串拷貝函數strcpy,這個函數在遇到 ‘\0’ 的時候並不會停下來。
- 若是source和destination有任何的重疊,複製的結果都是未定義的
memcpy能夠拷貝結構體,例如:
struct { char name[16]; int age; }person,person_copy; char myname[] = "kyrie"; memcpy(person.name, myname, strlen(myname) + 1); person.age = 46; /* using memcpy to copy structure: */ memcpy(&person_copy, &person, sizeof(person)); printf("person_copy: %s, %d \n", person_copy.name, person_copy.age);
這裏須要注意的是memcpy傳入的是地址
2.memmove
void * memmove ( void * destination, const void * source, size_t num );
3.memset
void * memset ( void * ptr, int value, size_t num );
ptr:爲指針或是數組,
value:是賦給ptr的值,
num:是ptr的長度.
這裏須要記住:內存拷貝操做的基本單元是字節,與類型無關
例子:
int a[5] = { 0 }; memset(a, 1, sizeof(a)); for (int i = 0; i < 5; i++) printf("a[%d]=%d\n", i, a[i]);
誤區:將整形數組a的每一個值賦1 ,這種理解錯誤
正確理解:對整形數組a的每一個字節賦1
4.memcmp
int memcmp ( const void * ptr1, const void * ptr2, size_t num );
比較從ptr1和ptr2指針開始的num個字節
例子:比較兩個字符串大小
char buffer1[] = "DWgaOtP12df0"; char buffer2[] = "DWGAOTP12DF0"; int n; n = memcmp(buffer1, buffer2, sizeof(buffer1)); if (n>0) printf("'%s' is greater than '%s'.\n", buffer1, buffer2); else if (n<0) printf("'%s' is less than '%s'.\n", buffer1, buffer2); else printf("'%s' is the same as '%s'.\n", buffer1, buffer2);