- 討論如何使用字符數(shù)組
- C語言沒有提供字符串的數(shù)據(jù)類型,而是使用char類型的數(shù)組元素存儲(chǔ)字符串
- 標(biāo)準(zhǔn)庫函數(shù)如何簡化字符串的處理
問題:
1、如何創(chuàng)建字符串變量??
2、哪些庫函數(shù)能處理字符串,如何應(yīng)用他們呢?
注意:
字符數(shù)組用來存儲(chǔ)字符串
字符串?dāng)?shù)組的大小必須要大于字符串的長度,當(dāng)不知道字符串的長度時(shí),為了避免越界,則應(yīng)定義一個(gè)足夠達(dá)的字符數(shù)組用來存儲(chǔ)字符串。這樣就會(huì)造成空間的浪費(fèi)
指針是存儲(chǔ)字符串的首地址的,利用指針可以避免空間浪費(fèi)
把一個(gè)字符串賦值給一個(gè)字符型數(shù)組,則數(shù)組名可表示字符串的首地址,
6.1什么是字符串
(1) 必須把字符串中的雙引號(hào)寫成轉(zhuǎn)義序列"
printf (" \" you right \" ");
" you right"
(2)要在字符串中包含反斜杠,也必須使用轉(zhuǎn)義序列 \
因?yàn)樽址械姆葱备芸偸潜硎巨D(zhuǎn)義序列的開頭。
- \0 即空字符,C語言中的字符串總是由\0字符結(jié)束
字符串的長度永遠(yuǎn)比字符串中的字符數(shù)多1
字符串常量是放在一對雙引號(hào)內(nèi)一串字符或符號(hào)
必須把字符串中的雙引號(hào)寫成轉(zhuǎn)義序列
- 空字符(\0)是字符串的終止符,而NULL是一個(gè)符號(hào),表示不引用任何內(nèi)容的內(nèi)存地址
6.2存儲(chǔ)字符串的變量
C根本就沒有字符串變量
但標(biāo)準(zhǔn)庫提供了許多函數(shù)來處理字符串
問題:如何創(chuàng)建字符串變量
使用char類型的數(shù)組保存字符串
char 數(shù)組變量的聲明如下
char saying[20];
這個(gè)變量可以存儲(chǔ)一個(gè)至多包含19個(gè)字符的字符串,因?yàn)楸仨毥o終止符提供一個(gè)數(shù)組元素。
初始化字符串變量
char saying[] = "This is a string.";
用一個(gè)字符串初始化char類型數(shù)組的部分元素
char str[40] = " To be";
初始化一個(gè)char數(shù)組,將它聲明為常量,是處理標(biāo)準(zhǔn)信息的好辦法
const char message[] = "The end of the world is nigh.";
字符串?dāng)?shù)組
可以使用char類型的二維數(shù)組存儲(chǔ)字符串,
char sayings[3][32] = {
"Manners maketh man,",
"Many hands make light work.",
"Too many cooks spoil the broth."
};
- 第一維指定數(shù)組可以包含的字符串個(gè)數(shù)
- 在字符串?dāng)?shù)組中必須指定第二維的大小,還必須確保第二維的空間足以容納最長的字符串,包含終止符。
char sayings[][32] = {
"Manner maketh man,";
"Many hands make lighe work.",
"Too many cooks spoil the broth."
};
for(unsigned int i = 0; i < sizeof(sayings)/sizeof(saying[0]); ++i)
printf("%s\n", sayings[i]);
- 使用sizeof運(yùn)算符可以確定數(shù)組中的字符串個(gè)數(shù)
6.3 字符串的操作
6.3.1檢查對C11的支持
- 所有可選函數(shù)名都以_s結(jié)尾
#include <stdio.h>
int main(void)
{
#if defined __STDC_LIB_EXT1__
printf("Optional functions are defined.\n");
#else
printf("Optional functions are not defined.\n");
#elseif
return 0;
}
6.3.2 確定字符串的長度
strnlen_s函數(shù)
概念: strnlen_s函數(shù)返回字符串的長度,
- strnlen_s函數(shù)需要兩個(gè)參數(shù):
第一個(gè)參數(shù):字符串的地址(這是一維char數(shù)組的數(shù)組名)
第二個(gè)參數(shù):數(shù)組的大小
strnlen_s()函數(shù)把字符串的長度返回一個(gè)size_t類型整數(shù)值
for(unsigned int i=0; i < strCount ; ++i)
{
printf("The string:\n \"%s"\n contains %zu characters.\n",str[i], strnlen_s(str[i], sizeof(str[i])));
}
6.3.3復(fù)制字符串
strcat()函數(shù) 、 strncat()函數(shù)
概念: strcpy_s()函數(shù)可以把一個(gè)字符串變量的內(nèi)容賦予另一個(gè)字符串
strcat()函數(shù)需要三個(gè)參數(shù):
第一個(gè)參數(shù):指定復(fù)制目標(biāo)
第二個(gè)參數(shù):是一個(gè)整數(shù),指定第一個(gè)參數(shù)的大小
第三個(gè)參數(shù):源字符串
char source[] = "Only the mediocre are always at their best.";
char destination[50];
if(strcpy_s(destination, sizeof(destination), source))
printf("An error occurred copying the string,\n");
6.3.4連接字符串
strcat()函數(shù) strncat()函數(shù)
概念: strcat_s()函數(shù)連接是把一個(gè)字符串連接到另一個(gè)字符串的尾部
strcat_s()函數(shù)需要三個(gè)參數(shù)
第一個(gè)參數(shù):要添加新字符串的字符串地址
第二個(gè)參數(shù):第一個(gè)參數(shù)可以存儲(chǔ)的最大字符串長度,
第三個(gè)參數(shù):要添加到第一個(gè)參數(shù)中的字符串地址
strcat()函數(shù)把一個(gè)整數(shù)錯(cuò)誤碼返回為errno_t類型的值,他是一個(gè)取決于編譯器的整數(shù)類型
示例
char str1[50] = " To be, or not to be "
char str2[] = "that is the question.";
int retval = strcat_s( str1, sizeof(str1), str2 );
if(retval)
printf("There was an error joining the strings.Error code = %d",retval);
else
printf("The combined strings:\n%s\n",str1);
工作方式
char str1[50] = "To be,or not to be, ";
char str2[] = "that is the question.";
int retval = strncat_s(str1, sizeof(str1), str2, 4);
if(retval)
printf("There was an error joining the strings. Error code = %d",retval);
else
printf("The combined strings:\n%s\n", str1);
6.3.5比較字符串
什么是字符碼??
兩個(gè)字符串的比較是基于它們的字符碼
函數(shù)strcmp()比較兩個(gè)字符串,返回一個(gè)小于,等于,或大于0 的int值分別對應(yīng)str1小于,等于或大于str2
要確定第一個(gè)字符串是小于還是大于第二個(gè)字符串,應(yīng)比較兩個(gè)字符串中第一對不同的字符,例如:第一個(gè)字符串中某個(gè)字符的字符碼小于第二個(gè)字符串中的對應(yīng)字符,第一個(gè)字符串就小于第二個(gè)字符串,以字母次序安排字符串時(shí),比較適合。
- 注意:比較字符串沒有可選函數(shù)
char str1[] = "The quick brown fox";
char str2[] = "The quick black fox";
if(strcmp(str1, str2) > 0)
printf("str1 is greater than str2.\n");
if(strnamp(str1,str2, 10) <= 0)
printf("\n%s\n%s", str1, str2);
else
printf("\n%s\n%s", str2, str1);
整數(shù)包括正整數(shù)和負(fù)整數(shù)和零。
只有strcmp()函數(shù)返回一個(gè)負(fù)數(shù),才會(huì)執(zhí)行printf()函數(shù),此時(shí),strcmp()函數(shù)會(huì)在這兩個(gè)字符串中找到一對不相同的字符
strncmp()函數(shù)的第三個(gè)參數(shù)是size_t 整數(shù)參數(shù)指定要比較的字符數(shù)。
6.3.6 搜索字符串
頭文件< string.h>聲明了幾個(gè)字符串搜索函數(shù)
1 指針的概念
指針是含有地址的變量
*是取消引用運(yùn)算符,其作用是訪問指針指定的地址中存儲(chǔ)的數(shù)據(jù)。
2 搜索字符串中的一個(gè)字符
概念:strchr()在字符串中搜索給定的字符,
strchr()函數(shù)的參數(shù):
第一個(gè)參數(shù):要搜索的字符串
第二個(gè)參數(shù):要查找的字符
- 函數(shù)返回在字符串中找到的第一個(gè)給定字符的地址。要存儲(chǔ)這個(gè)返回值就必須創(chuàng)建一個(gè)能存儲(chǔ)字符地址的變量,如果沒有找到給定的字符,函數(shù)就會(huì)返回NULLU,表示這個(gè)指針沒有指向任何對象。
函數(shù)strchr()的用法:
char str[] = "The quick brown fox";
char ch = 'q';
char *pGot_char = NULL;
pGot_char = strchr(str , ch);
strchr()函數(shù)希望其第二個(gè)參數(shù)是int類型。
int ch = 'q';
printf(" Character found was %c.", *pGot_char);
if(pGot_char)
printf("Character found was '%c'.", *pGot_char);
這條語句的輸出:
Character found was 'q'.
printf("The substring beginning with '%c' is:\"%s\"\n",ch , pGot_char);
這個(gè)語句的輸出:
The substring beginning with 'q' is: "quick brown fox"
char str[] ="Peter piper picked a peck of pickled pepper.";
char ch = 'p';
char *pGot_char = str;
int count = 0;
while(pGot_char = strchr(pGot_char, ch))
{
++count;
++pGot_char;
}
printf("The character '%c' was found %d times in the following string:\n\"%s\"\n", ch , count ,str);
The character 'p' was found 8 times in the following string:
"Peter piper picked a peck of pickled pepper."
表示文件尾的EOF字符是一個(gè)負(fù)整數(shù)
3.在字符串中查找子字符串
strstr()函數(shù)查找一個(gè)字符串中的子字符串,返回找到的第一個(gè)子字符串的位置指針
strstr()函數(shù)
第一個(gè)參數(shù):要搜索的字符串
第二個(gè)參數(shù):要查找的子字符串
使用strstr()函數(shù)的例子:
char text[] = "Every dog has his day";
char word[] = "dog";
char *pFound = NULL;
pFound = strstr(text, word);
6.3.7單元化字符串
標(biāo)記:是字符串中用某些預(yù)定界定符界定的一個(gè)字符序列
標(biāo)準(zhǔn)庫提供了strtok()函數(shù),來單元化字符串
strtok()strtok()函數(shù)需要兩個(gè)參數(shù):
第一個(gè)參數(shù):要單元化的字符串
四二個(gè)參數(shù):包含所有可能的界定符的字符串可選的單元化函數(shù)strtok_s()函數(shù),允許多次調(diào)用函數(shù),在單個(gè)字符串中連續(xù)查找界定符
strtok_s()函數(shù)需要4個(gè)參數(shù)
str:要單元化的字符串的地址
str_size:包含數(shù)組長度的整數(shù)變量的地址
delimiters:包含所有可能界定符的字符串的地址
pptr:指向char*型變量的指針
gets_s()函數(shù)和scanf_s()函數(shù)的區(qū)別:
使用gets_s()函數(shù)讀取輸入是因?yàn)樗梢宰x取字符串,包括空格,而scanf_s()不能。gets_s()函數(shù)會(huì)把至多buf_len- 1個(gè)字符讀入buf,并追加一個(gè)\0
6.3.8將換行符讀入字符串
fgets()函數(shù)很好的實(shí)現(xiàn)輸入過程,fgets()函數(shù)在輸入的字符串存儲(chǔ)換行符來結(jié)束輸入過程,這是一個(gè)很有用的輸入函數(shù)。可以用于讀取文件和讀取鍵盤的輸入
- fgets()函數(shù)需要三個(gè)參數(shù)
第一個(gè)參數(shù):輸入數(shù)組str的地址
第二個(gè)參數(shù):要讀取的最大字符數(shù)(通常是str的字符串長度)
第三個(gè)參數(shù):輸入源
6.4分析和轉(zhuǎn)換字符串
在頭文件<ctype.h>中聲明的標(biāo)準(zhǔn)庫函數(shù)
分析函數(shù)可以測試有什么樣的字符
- 字符分類函數(shù)
|---------函數(shù)------- |-------測試內(nèi)容 --------- --------| |
|------------------- : |:------------------------------ :|
| islower | 小寫字母 |
| isupper | 大寫字母 |
| isalpha | 大寫或小寫字母 |
| isalnum | 大寫或小寫字母或數(shù)字 |
| iscntrl | 控制字符,包括空格 |
| isgraph | 打印字符,不包括空格 |
| isdigit | 十進(jìn)制數(shù)字'0'·~'9' |
| isxdigit | 十六進(jìn)制數(shù)字 |
| isblank | 標(biāo)準(zhǔn)空白字符 |
| isspace | 空白字符 |
| ispunct |isspace和isalnum返回false的可打印字符 |
6.4.1 轉(zhuǎn)換字符的大小寫形式
標(biāo)準(zhǔn)庫<ctype.h>還包含兩個(gè)轉(zhuǎn)換函數(shù)。函數(shù)toupper()將小寫字母轉(zhuǎn)換為大寫,函數(shù)tolower()將大寫字母轉(zhuǎn)換為小寫。
可以把一個(gè)字符串轉(zhuǎn)換成大寫:
for(int i = 0; (buf[i] = (char)toupper(buf[i])) != '\0' ; ++i);
這個(gè)循環(huán)會(huì)一次一個(gè)字符的遍歷字符串,將buf 數(shù)組的整個(gè)字符串轉(zhuǎn)換成大寫
6.4.2將字符串轉(zhuǎn)換為數(shù)值
在頭文件<stdlib.h>聲明了一些能將字符串轉(zhuǎn)換為數(shù)值的函數(shù)
十六進(jìn)制的值必須以0X或0x開頭
八進(jìn)制的值必須以0開頭
double value = 0;
char str[] = "3.5 2.5 1.26";
char *pstr = str;
char *ptr = str;
while(true)
{
value = strtod(pstr,&ptr);
if(pstr == ptr)
break;
else
{
printf(" %f",value);
pstr = ptr;
}
}
char str[] = " 123 234 0xAB 111011";
char *pstr = str;
char *ptr =NULL;
long a = strtol(pstr, &ptr,0);
pstr = ptr;
unsigned long b = strtoul(pstr, &ptr,0);
pstr = ptr;
long c = strtol(pstr , &ptr, 16);
pstr = ptr;
long d = strtol(pstr,&ptr, 2);
常用的字符串函數(shù)如下:
使用字符串函數(shù)注意以下事項(xiàng):
1、strlen()獲取字符串的長度,在字符串長度中是不包括‘\0’而且漢字和字母的長度是不一樣的。
2、strcmp()在比較的時(shí)候會(huì)把字符串先轉(zhuǎn)換成ASCII碼再進(jìn)行比較,返回的結(jié)果為0表示s1和s2的ASCII碼相等,返回結(jié)果為1表示s1比s2的ASCII碼大,返回結(jié)果為-1表示s1比s2的ASCII碼小,
3、strcpy()拷貝之后會(huì)覆蓋原來字符串且不能對字符串常量進(jìn)行拷貝,
4、strcat在使用時(shí)s1與s2指的內(nèi)存空間不能重疊,且s1要有足夠的空間來容納要復(fù)制的字符串,