C语言中字符串函数和部分内存操作函数

文章详细介绍了C语言中处理字符串的一些基本函数,如strlen计算字符串长度,strcpy和strcat进行字符串拷贝和追加,strcmp进行字符串比较,以及针对有限长度字符串的strncpy等。此外,还讨论了内存操作函数memcpy和memmove。每个函数都提供了模拟实现和使用示例,帮助理解其工作原理和应用场景。
摘要由CSDN通过智能技术生成

字符函数和字符串函数

求字符串长度——strlen

size_t strlen(const char*str);

注意

  • 字符串以’\0’作为结束标志,strlen函数返回的是字符串中’\0’之前出现的字符个数(不包括’\0’)
  • 参数指向的字符串必须要以’\0’结束
  • 函数的返回值为size_t,是无符号整型
    举个栗子(功能展示)
#include<stdio.h>
#include<string.h>
int main(){
char arr[]="avsdjfo";
int num=strlen(arr);
printf("%d\n",num);
return 0;
}

模拟实现
方法一:计数实现

#include<stdio.h>
size_t my_strlen(const char*str){
int num=0;
while(*str!='\0'){
num++;//用来计数的变量
str++;//指针后移,读取下一个字符
}
return num;
}
int main(){
char a[]="abcdefg";
int b=my_strlen(a);
printf("%d\n",b);
return 0;
}

方法二:递归实现(不创建临时变量)

int my_strlen(const char*str){
if(*str!='\0')
return 1+my_strlen(str+1);//实现递归
else return 0;
}

长度不受限制的字符串函数——strcpy、strcat、strcmp

1.strcpy
将一个字符串拷贝到另一个字符串中

char* strcpy(char*destination,const char* source);

注意

  • 源字符串必须以’\0’结束
  • 会将源字符串中的’\0’拷贝到目标空间
  • 目标空间必须足够大,确保能够存放源字符串
  • 目标空间必须可变
  • 返回的是目标空间的起始地址
    功能展示:
#include<stdio.h>
int main(){
char arr1[]="abcdefg";
char arr2[20]={0};
strcpy(arr2,arr1);
printf("%s\n",arr2);
return 0;
}

模拟实现

char* my_strcpy(char* dest,const char* src){
char*ret=dest;//保存目标空间的起始地址
while(*dest++=*src++){
;
}//实现拷贝功能
return ret;//返回目标空间的起始地址
}

2.strcat
字符串追加

char * strcat ( char * destination, const char * source );

注意

  • 源字符串必须以’\0’结束
  • 目标空间必须足够大,能容纳下拷贝后的字符串中的所有内容
  • 目标空间必须可修改
  • strcat是从’\0’的位置开始追加的
  • 两个字符串都必须有’\0’(以便明确从何处开始追加,到何处停止追加)
  • 该函数不能自己给自己追加(会造成死循环)
    功能展示
#include<stdio.h>
int main(){
char arr1[20]="hello ";
char arr2[]="world";
strcat(arr1,arr2);
printf("%s\n",arr1);
return 0;
}

模拟实现

char* my_strcat(char* dest,const char*src){
char*ret=dest;//存储目标空间的起始地址
while(*dest!='\0'){
dest++;
}//找到目标空间的结尾处
while(*dest++=*src++{
;
}//实现字符串追加的功能
return ret;//返回目标空间的起始地址
}

3.strcmp

int strcmp(const char*str1,const char*str2);

注意

  • 第一个字符串大于第二个字符串,则返回大于0的数字
  • 第一个字符串等于第二个字符串,则返回0
  • 第一个字符串小于第二个字符串,则返回小于0的数字
  • 比较的是两个字符串对应位置上的字符的ASCII码值的大小
  • 比较两个字符串必须用该函数,不能直接比较,如下(错误示范):
if("abcdef"=="bbcdef")//这里比较的是两个字符串中首字符的地址,并非两个字符串的内容

功能展示

#include<stdio.h>
int main(){
char arr1[]="abcdef";
char arr2[]="bbq";
int ret=strcmp(arr1,arr2);
printf("%d\n",ret);
return 0;
}

模拟实现

int my_strcmp(const char* str1,const char* str2){
while(*str1==*str2){
if(*str1=='\0')
    return 0;
str1++;
str2++;
}
return *str1-*str2;
}

上述三个函数用于长度不受限制的字符串

长度受限制的字符串——strncpy、strncat、strncmp

由于前面已经分别介绍过三个函数的模拟实现
接下来只粗略介绍该三个函数的调用形式
1.strncpy

char* strncpy(char*destination,const char* source,size_t num);

2.strncat
注意
该函数可以实现自己给自己追加

char * strncat ( char * destination, const char * source, size_t num );

3.strncmp

int strncmp ( const char * str1, const char * str2, size_t num );

字符串查找——strstr、strtok

1.strstr

const char * strstr ( const char * str1, const char * str2 );
//或
char * strstr (       char * str1, const char * str2 );

注意

  • 该函数用于在str1中找str2,(在str1中找str2第一次出现的位置)
  • 如果没有找到,返回空指针
  • 如果找到多个,返回第一次找到的位置
    功能展示
#include<stdio.h>
int main(){
char arr1[]="abcdef";
char arr2[]="bcd";
char *p=strstr(arr1,arr2);
if(p==NULL)printf("找不到\n");
else printf("%s\n",p);
return 0;
}

模拟实现

char* my_strstr(const char*str1,const char* str2){
char* s1=NULL;
char* s2=NULL;
char* cp=(char*)str1;
while(*cp){
s1=cp;
s2=(char*)str2;
while(*s1&&*s2&&*s1==*s2){
s1++;
s2++;
}
if(*s2=='\0')return cp;
cp++;
}
return NULL;
}

2.strtok

char* strtok(char* str,const char* sep);

注意

  • 参数sep是一个字符串,定义了用作分隔符的字符集合
  • 第一个参数指定一个字符串,它包含了0个或者多个由sep字符串中一个或者多个分隔符分割的标记
  • strtok函数找到str中的下一个标记,并将其用\0结尾,返回一个指向这个标记的指针
  • 注:strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容并且可修改
  • strtok函数的第一个参数不为NULL,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置
  • strtok函数的第一个参数为NULL,函数将在同一个字符串中被保存的位置开始,查找下一个标记
  • 如果字符串中不存在更多的标记,则返回NULL指针
int main()
{
	char arr[] = "www@http.net";//三个段,调用三次
	//若为char arr[] = "192.168.3.212";则四个段,调用四次
	//p变为"."
	char buf[30] = { 0 };
	strcpy(buf, arr);//www@http.net
	const char* p = "@.";
	char* str = strtok(buf, p);//zpengwei\0yeah.net   并返回z的地址,保存\0的位置
	printf("%s\n", str);
	str = strtok(NULL, p);//第二次调用,从刚刚保存好的位置开始找,找到“.”后将其改为\0,同时返回y的地址,再记住\0的位置
	printf("%s\n", str);
	str = strtok(NULL, p);//第三次调用,从上次保存的\0的位置开始,向后找,找到\0后停止(结束)返回n的地址
	//再向后找不到标记了,会返回空指针
	printf("%s\n", str);
	//实质就是切割字符串
	return 0;
}

一个函数多次调用感觉麻烦,可看下方改进
改进

int main()
{
	char arr[] = "192.168.3.212";
	char buf[30] = { 0 };
	strcpy(buf, arr);
	//若写为如下形式,则程序崩溃
	//const char* buf = "zpengwei@yeah.net";
	//原因是:常量字符串不能被修改
	const char* p = ".";
	char* str = NULL;
	for (str = strtok(buf, p); str != NULL; str=strtok(NULL,p))
	{
		printf("%s\n", str);
	}
	return 0;
}

错误信息报告——strerror

strerror

char* strerror(int errnum);

注意

  • 返回错误码所对应的错误信息
  • 头文件<errno.h>
    (加上特定情境后的)功能展示(这里的关于文件的打开、读写和关闭读者可以自己先了解一下下)
int main()
{
	//打开文件
	//打开文件的时候,如果文件的打开方式是"r"
	//文件存在则打开成功,文件不存在则打开失败
	//打开文件失败的话,会返回NULL
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		printf("文件打开失败,原因是:%s\n", strerror(errno));
		return 1;
	}
	//读写文件
	//...
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

内存操作函数——memcpy、memmove

1.memcpy
(功能类似于strcpy(拷贝),但范围更广)
功能展示

int main()
{
	//此处以整型数据示范
	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[8] = { 0 };
	//把arr1中的前5个数据拷贝到arr2中
	//不能用strcpy——字符串拷贝,上述数组是整型数据
	memcpy(arr2, arr1, 20);//此处指的是20个字节!!!
	return 0;
}

模拟实现

my_memcpy(void* dest, const void* src, size_t num)
{
	void* ret = dest;
	assert(dest && src);
	while (num--)
	{
		*(char*)dest = *(char*)src;//强制类型转换,因为需保证一次访问一个字节
		dest = (char*)dest + 1;
		src = (char*)src + 1;
	}
	return 0;
}

2.memmove
功能展示

int main()
{
	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
	my_memmove(arr1 + 2, arr1, 20);
	int i = 0;
	for (i = 0; i < 10; i++)
		printf("%d ", arr1[i]);
	return 0;
}

模拟实现

my_memmove(void* dest, const void* src, size_t num)
{
	void* ret = dest;
	assert(dest && src);
	if (dest < src)
	{
		while (num--)
		{
			*(char*)dest = *(char*)src;//强制类型转换,因为需保证一次访问一个字节
			dest = (char*)dest + 1;
			src = (char*)src + 1;
		}
	}
	else
	{
		while (num--)
		{
			*((char*)dest + num) = *((char*)src + num);
		}
	}
}

(附:若有错误,还恳请斧正)
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值