一、字符串
字符:
在计算机中字符都是以整数形式存储的,当需要显示字符时,就会工具ASCII码表中的对应关系显示出相应的符号或图案
'\0' 0; 空字符
'0' 48;
'A' 65;
'a' 97;
串:
是一种数据结构,由一组连续的若干个类型相同的数据组成,末尾一定有一个结束标志
对于这种数据结构的处理都是批量性的,从开始的地方一直处理到结束标志停止
字符串:
由字符组成的串形结构,结束标志是'\0'
二、字符串的存在形式
字符数组:
char str[6]={'a','b','c','d','e'};
由char类型组成的数组,要为’\0‘预留位置
使用的是栈内存,数据是可以中途修改的
字符串字面值:
由双引号包含的若干个字符
const char* p="123456789";
末尾会自动补'\0'
sizeof(p)=4
sizeof("abcd")=5 //计算字符串字面值在代码段中所占的内存字节数
字符串字面值以地址形式存在,字符数据存在代码段,如果修改一定会产生段错误
注意:完全相同的字符串字面值,在代码段中永远只存在一份
常用方式:
字符数组[]=“字符串字面值”;
char str[]="78434";
会自动给'\0'
预留位置、可以修改内容、初始化简单
赋值完成后字符串存在两份,一份在代码段、一份在栈内存(可被修改)
三、字符串的输入、输出
输入:
char str[256]={};
scanf("%s",str); //不能输入带空格的字符串
char *gets(char *s); //可以输入带空格字符串
//返回值:返回s,为了链式调用,一个函数的返回值直接作为另一个函数的参数
//缺点:不限制输入长度,有安全隐患,编译器会有警告
char *fgets(char *s, int size, FILE *stream);
//从stream中最多输入size-1个字符到s中
//s:字符数组 size:最多接受的字符个数+1 stream:stdin 相当于键盘文件,固定写
//返回值:返回s,为了链式调用,一个函数的返回值直接作为另一个函数的参数
//注意:如果输入超过size-1个字符,多出来的不接收 如果输入不足,'\n'也会接收
输出:
printf("%s",字符数组/字面值);
int puts(const char *s);//输出字符串 返回值:成功输出字符的个数
//注意:会自动打印换行符
练习1:实现一个判断字符串是否是回文串的函数
#include <stdio.h>
#include <string.h>
int is_pal(const char* str,int len);
int main(int argc,const char* argv[])
{
char str[100]={};
scanf("%s",str);
if(is_pal(str,strlen(str)))
printf("Yes");
else
printf("No");
}
int is_pal(const char* str,int len)
{
int l=len-1;
for(int i=0;i<len/2;i++,l--)
{
if(str[i]==str[l])
continue;
else
return 0;
}
return 1;
}
练习2:实现一个函数,把一个由数字字符组成的字符串转换成对应的整数
#include <stdio.h>
#include <string.h>
int str_tint(const char* str);
int main(int argc,const char* argv[])
{
char str[100]={};
scanf("%s",str);
printf("%d",str_tint(str));
}
int str_tint(const char* str)
{
int sum=0;
for(int i=0;str[i]!='\0';i++)
{
if(*(str+i)>=0 && *(str+i)<='9')
{
sum*=10;
sum+=*(str+i)-48;
}
}
return sum;
}
练习3:实现一个函数,把一个字符串逆序
四、输出缓冲区
程序中要输出显示内容并不会立即显示到屏幕上,二十先存储在输出缓冲区中,当满足一定条件时才会从缓冲区中显示内容到屏幕上
- 遇到
'\n'
后 - 遇到输入语句时
- 当输出缓冲区满了 4k
- 程序结束时
- 手动刷新
fflush(stdout);
五、输入缓冲区
程序并不是立即从键盘获取输入的内容,而是当按下回车后终端输入的内容会先存储到输入缓冲区中,然后输入函数再从输入缓冲区中读取数据到内容中
-
当想要读取整型或浮点型数据,但是缓冲区中的的数据是符号或字母时,读取会失败,数据会残留在缓冲区中,影响接下来所有数据的读取
解决:判断
scanf
的返回值是否全部数据输入正确,如果不是,先清理输入缓冲区再重新输入,直到全部正确 -
fgets
可以接收指定size-1个字符,如果有多余的字符会残留在缓冲区中,可能会影响接下来的输入解决:
法1:
1、先判断\n在不在字符串内,如果不在,则说明在缓冲区内,需要清理缓冲区 2、如果在缓冲区内,则 scanf("%*[^\n]"); //从缓冲区中读取任意数据并丢弃,如果不是\n,继续读取,遇到\n停止(正则表达式) scanf("%*c"); //从缓冲区中读取一个字符并丢弃 //可以封装新的fgets函数解决输入过多问题
法2:
stdin->_IO_read_ptr = stdin->_IO_read_end; //仅Linux、UNIX可用 //把输入缓冲区中当前位置指针,移动到缓冲区末尾,相当于清理了输入缓冲区
-
当先输入整型、浮点型数据,再输入字符、字符串时,前一次的输入会残留
\n
,影响了后面字符、字符串的输入解决:
scanf(" %c",&ch);
六、字符串相关操作函数
#include <string.h>
size_t strlen(const char *s); //计算字符串长度,不包括\0
char *strcpy(char *dest, const char *src); //把src拷贝给dest,相当于给dest赋值
//返回值:链式调用
char *strcat(char *dest, const char *src); //把src追加到dest末尾,相当于合并两个字符串
//返回值:链式调用
int strcmp(const char *s1, const char *s2);
/*比较两个字符串大小.从头开始,一对一比较,按照字典序,谁出现在前谁小,一旦比较出结果立即返回,后面不比
返回值:
0 s1==s2
正数 s1>s2
负数 s1<s2
*/
char *strncpy(char *dest, const char *src, size_t n);
char *strncat(char *dest, const char *src, size_t n);
int strncmp(const char *s1, const char *s2, size_t n);
char *strstr(const char *haystack, const char *needle);
//在haystack中查找是否存在子串needle中
//返回值:needle在haystack中首次出现的位置,找不到则返回NULL
void *memcpy(void *dest, const void *src, size_t n);
//从src位置拷贝n个字节到dest的位置
//链式调用
int memcmp(const void *s1, const void *s2, size_t n);
/*比较两块内存的值,按照字节比较,一旦比较出结果后不再比较
返回值:
0 s1==s2
正数 s1>s2
负数 s1<s2
*/
#include <stdlib.h>
int atoi(const char *nptr); //字符串转int
long atol(const char *nptr); //字符串转long
long long atoll(const char *nptr); //字符串转long long
double atof(const char *nptr); //字符串转double
#include <stdio.h>
int sprintf(char *str, const char *format, ...);
//把各种类型的数据输出给字符串str
//返回值:转换后的字符个数
sprintf(str,"%d %lf %hd",num,f,s);
int sscanf(const char *str, const char *format, ...);
//从str中解析读取数据到变量中
//返回值:成功解析出的变量个数
const char* str="10086 3.14 88";
int num=0; double d=0; short s=0;
sscanf(str,"%d %lf %hd",&num,&d,&s);