字符串
1. 概念
用雙引號引起來的就是字符串
"a string" |
// C語言編譯器會將兩個並列的字符串自動拼接成一個字符串 "a string""another a string" |
// 折行符'\'是代碼換行連接的標記(一行不夠寫) "a looooooooooong \ string" |
常見ASCII編碼:
'A' == 65 'a' == 97 '0' == 48 '\0' == 0
int a[10]; //表示在棧中分配了40Bytes的內存空間,空間的首地址是a
char a[10]; //表示在棧中分配了10Bytes的內存空間,空間的首地址是a
如何表示和保存字符串:
C語言中沒有字符串類型,用一片內存空間保存一串字符,約定用整數0(或字符'\0')來表示一個字符串的結束。
使用字符串時只需要記錄字符串的開始位置。
C語言中的字符串用字符數組表示:
char a[6] = {'h','e','l','l','o','\0'};//特殊的字符數組(字符串)
char a[6] = "hello"; //簡化版寫法(這種寫法 \0省略)
char *a = "hello"; //定義一個字符串的另一種寫法
2. 初始化
char str[] = "hello"; //常用 char str[] = {'h', 'e', 'l', 'l', 'o', '\0'}; // 部分初始化, 部分初始化中, 沒有被初始化的元素默認是0 (\0對應的ASCII 0) char str[8] = {'h', 'e', 'l', 'l', 'o'}; // hello000 char str[5] = {'h', 'e', 'l', 'l', 'o'}; //該方式不是字符串, 而是字符數組, 因為沒有\0 char str[] = {'h', 'e', 'l', 'l', 'o'}; //同上,是字符數組 |
" "; //包含一個空格 ""; //空字符串,什么東西都沒有 |
字符串的賦值:
- 給 char* 類型的字符串賦值,可以直接使用 "=" 號
- 給 char[] 類型的字符串賦值,需要使用 strcpy 函數
字符串的特點:
- 需要明白的一點就是字符串以\0結尾, 沒有\0就不是字符串
- 只要是用雙引號括起來的都是字符串
- 字符串的本質就是數組(字符數組)
char str[] = "hello";
str[1] = 'y';
printf("%s", str);//hyllo
注意: 字符串變量 和 普通的字符數組 有一定的區別
C語言規定, 字符串必須以\0結尾(作為字符串的結束符號), 所以字符串變量的元素個數比字符數組的元素個數多一個'\0'(0) char a[5] = {'h', 'e', 'l', 'l', 'o'}; //普通的字符數組(不是字符串,因為沒有\0) char a[8] = {'h', 'e', 'l', 'l', 'o'}; //部分初始化(未初始化的部分都為0),以0結尾(是字符串) char a[6] = {'h', 'e', 'l', 'l', 'o', '\0'}; //以\0結尾(是字符串) char a[6] = {'h', 'e', 'l', 'l', 'o', 0}; //以0結尾(是字符串) 注意: '0' = 48;//"abc0d" 雙引號里是字符'0'不是數值0 '\0' = 0; |
3. 輸出
%s的原理, 從傳入的"地址"開始逐個取出, 直到遇到"\0"位置
如何輸出字符串:
- 使用printf的%s占位符來輸出
- 使用puts函數來輸出(自動換行,原樣輸出)
char str[] = "how are you";
printf("%s\n", str); //how are you
puts(str); //how are you
4. 輸入
- 利用scanf函數接收字符串
- 利用gets函數接收字符串
- 利用fgets函數接收字符串(推薦常用!)
/** 給字符數組賦值的三種方式 */ #include <stdio.h> #include <string.h> void mystrcpy(char *str1, const char *str2) { //*str2對*str1逐個字符進行賦值 //*str2直到把'\0'賦值給*str1時,*str1的結果就是0,循環就結束! while ((*str1++ = *str2++)); } int main() { char str[10] = "abc"; //1.使用循環給字符數組賦值 for (int i = 0; i < 10; i++) { str[i] = "ABC"[i]; //等價於 *("ABC"+i),"ABC"返回的是A的地址(即首地址) } printf("str = %s\n", str);//str = ABC //2.使用標准庫函數給字符數組賦值 strcpy(str, "XYZ"); printf("str = %s\n", str);//str = XYZ //3.使用自定義函數給字符數組賦值 mystrcpy(str, "OKOK"); printf("str = %s\n", str);//str = OKOK return 0; } |
5. 庫函數
- 計算字符串的長度(strlen):(計算字符串中有多少個字符,注意不包括\0)
strlen的原理:從傳入的地址開始逐個取出字符串,每取出一個就讓計數器+1.直到遇到\0為止。
- 字符串拼接(strcat)
原理:首先遍歷第一個字符串,直到遇到\0為止,然后取出第二個字符串中的字符,從\0的位置開始添加,添加完畢之后再在最后添加一個\0
- 字符串拷貝(strcpy)
strcpy函數會將源的數據拷貝到目標中,並且會覆蓋掉目標中原有的數據,目標的容積必須能夠存放拷貝的數據, 如果容積不夠會報錯。
- 字符串比較(strcmp)
原理: 取出字符串中的每一個字符進行逐個比較, 如果發現不相等就不會繼續往下比較
#include <stdio.h> #include <string.h> #include <stdbool.h> /** strcpy 字符串賦值函數 */ void test1() { char str[6]={0};//表示在棧中分配了6個字節的內存空間,空間的首地址是str(數組名) strcpy(str, "abc");//給字符數組賦值,str[10] = "abc"; strncpy(str, "AABBCC", sizeof(str)-2);//只賦值前4個字符(AABB);str[6] = "AABB"; } /** strcat 給一個字符串追加內容 */ void test2() { char str[8] = "abc"; strcat(str, "def"); //str[8] = "abcdef"; strncat(str, "kkkkkk", 3);//只追加前3個字符; str[8] = "abckkk"; } /** strcmp 比較字符串內容的大小 */ void test3() { char *str1 = "abcd.c"; char *str2 = "abcf.m"; strcmp(str1, str2); //返回值為: -2 (表示 str1 < str2) strncmp(str1, str2, 3); //比較前3個字符的大小; 返回值為: 0 (表示 str1 == str2) bool r = str1 > str2;//比較地址大小(str1和str2都是地址) } /** memset 內存清理函數(清空) */ void test4() { char str[8] = "abcd"; memset(str, 0, sizeof(str));//清理內存空間(開始位置, 清零, 空間大小/長度); strcpy(str, "ABCD");//清空后重新賦值 printf("str = %s\n", str); } int main() { test4(); return 0; } |
//strcmp函數實際應用(判斷字符串是否相等),驗證登錄密碼! #include <stdio.h> #include <string.h> int main() { char pwd[20] = {0}; do { printf("請輸入密碼:"); scanf("%s", pwd); } while (strcmp(pwd, "abc123")); printf("密碼正確,登錄成功!\n"); return 0; } |
指針變量本身在棧中,指針變量可以指向任何地方。(指針和指針指向的空間是兩個空間)
指針修改值的兩種情況:
1)修改指針本身的值(也就是改變指針的指向) ;
2)修改指針所指向的空間的值
字符串數組
1. 定義
字符串數組:一個數組中的所有的元素都是字符串。
如果想存儲一堆字符串那么可以使用字符串數組,說白了字符串數組就是二維數組
字符串數組兩種表示:
第一種:char str[4][6] = {"aaa", "bbb", "ccc"}; //char類型的二維數組
第二種:char *str[4] = {"aaa", "bbb", "ccc"}; //char*類型的一維數組
2. 應用
命令行參數
在命令行(終端)執行程序時給程序傳入的參數。(如:ls是命令行,-l是參數)
(如:./a.out aaa bbb ccc是命令行參數)