常用的字符函数和字符串函数

2023-05-16

 

一、分类

  • 求字符串长度:strlen
  • 长度不受限制的字符串函数:strcpy、strcat、strcmp
  • 长度受限制的字符串函数:strncpy、strncat、strncmp
  • 字符串查找:strstr strtok
  • 错误信息报告:strerror
  • 字符操作
  • 内存操作函数:memcpy、memmove、memset、memcmp                                                                                                   

二、详细介绍

1.strlen

①作用:获取一个字符串长度
②用法:size_t strlen( const char *string )
③头文件:<string.h>
④注意:
  a.字符串已经 '\0' 作为结束标志,strlen函数返回的是在字符串中 '\0' 前面出现的字符个数(不包含 '\0' )
  b. 
参数指向的字符串必须要以 '\0' 结束。
  c.
注意函数的返回值为size_t,是无符号的。
⑤模拟实现
a.计数器实现(
每次往后移动一位,计数器count++。直到遇到\0结束。

int my_strlen(char* str)
{
	int count = 0;
	while (*str++)
	{
		count++;
	}
	return count;
}

int main()
{
	char arr[] = "abcdef";
	int ret = my_strlen(arr);
	printf("%d\n", ret);
	return 0;
}

b.递归实现(不创建临时变量)
这样说吧,my_strlen("abc")=1+my_strlen("bc")=1+1+my_strlen("c")=1+1+1+my_strlen(" "),将求整个字符串的长度转换为求1+剩下字符串的长度,层层剥开,这就是递归思想。

  • 递归的限制条件是*str == '\0'时返回
  • 每次调用函数是让str+1,越来越接近这个限制条件
int my_strlen(char* str)
{
	if (*str == '\0')
	{
		return 0;
	}
	else
	{
		return  my_strlen(str + 1) + 1;
	}
}

int main()
{
	char arr[] = "abcdef";
	int ret = my_strlen(arr);
	printf("%d\n", ret);
	return 0;
}

c.指针-指针
指针和指针相减是元素个数的绝对值。

int my_strlen(char* str)
{
	char *p = str;
	while (*p != '\0')
	{
		p++;
	}
	return p - str;
}

int main()
{
	char arr[] = "abcdef";
	int ret = my_strlen(arr);
	printf("%d\n", ret);
	return 0;
}

2.strcpy和strncpy

①作用:拷贝字符串和拷贝字符串有长度限制
②用法:char *strcpy( char *strDestination, const char *strSource )
              char
*strncpy( char *strDest, const char *strSource, size_t count )
③头文件:<string.h>
④注意:
   a.源字符串必须以 '\0' 结束。
   b.会将源字符串中的 '\0' 拷贝到目标空间。
   c.目标空间必须足够大,以确保能存放源字符串。
   d.目标空间必须可变
   A.拷贝count
个字符从源字符串到目标空间。
   B.如果源字符串的长度小于num
,则拷贝完原字符串后,在目标后面追加0,直到count个。
⑤模拟实现

#include <stdio.h>
#include <assert.h>
char* my_strcpy( char* des,const char* src)
{
	assert(des != NULL);
	assert(src != NULL);
	char* ret = des;
	while ((*des++ = *src++))
	{
		;
	}
	return ret;
}

int main()
{
	char arr[20] = { 0 };
	char str[] = "hello world";
	char* ret = my_strcpy(arr, str);
	printf("%s\n",(char*) ret);
	return 0;
}

 

#include <stdio.h>
#include <assert.h>
char* my_strncpy(char* des, const char* src, size_t count)
{
	char* ret = des;
	assert(des && src);
	while (count && (*des++ = *src++))  //拷贝字符串 (拷贝\0 )
	{
		count--;
	}
	if (count)                //目标后面追加0           
	while (--count)   //遇到\0就要结束所以为--count
	{
		*des++ = '\0';
	}
	return ret;
}

int main()
{
	char arr[20] = { 0 };
	char str[] = "hello world";
	char* ret = my_strncpy(arr, str, 5);
	printf("%s\n", ret);
	return 0;
}

3.strcat和strncat
①作用:追加字符串和追加字符串长度受限制
②用法:char *strcat( char *strDestination, const char *strSource )
               char *strncat( char *strDest, const char *strSource, size_t count )
③头文件:<string.h>
④注意:
   a.源字符串必须以 '\0' 结束。
   b. 目标空间必须有足够的大,能容纳下源字符串的内容。
   c. 目标空间必须可修改。
   d. 字符串自己给自己追加,造成\0被覆盖,死循环导致数组越界

⑤模拟实现

#include <stdio.h>
#include <assert.h>
char* my_strcat(char* des, const char* src)
{
	char* ret = des;
	assert(des != NULL);
	assert(src != NULL);
	//判断\0
	while (*des++)
	{
		des++; //不要把\0传进去
	}
	//进行拷贝
	while ((*des++ = *src++))
	{
		; //需要把\0传进去
	}
	return ret;
}

int main()
{
	char arr[20] = "hello ";
	char str[] = "world";
	char* ret = my_strcat(arr, str);
	printf("%s\n", ret);
	return 0;
}

 


#include <stdio.h>
#include <assert.h>
char* my_strncat(char* des, char* src, size_t count)
{
	char* ret = des;
	assert(des && src);
	while (*(++des))
	{
		;
	}
	while (count && (*des++ = *src++))
	{
		count--;
	}
	*des = '\0';
	return ret;
}

int main()
{
	char str1[20] = "hello ";
	char str2[5] = "world";
	char* ret = my_strncat(str1,str2,3);
	printf("%s\n", ret);
	return 0;
}

4.strcmp和strncmp
①作用:比较两个字符串的大小和比较两个字符串大小长度受限制
②用法:int strcmp( const char *string1, const char *string2 )
             int strncmp( const char *string1, const char *string2, size_t count )
③头文件:<string.h>
④注意:
   第一个字符串大于第二个字符串,则返回大于0的数字
   第一个字符串等于第二个字符串,则返回0
   第一个字符串小于第二个字符串,则返回小于0
的数字
⑤模拟实现

#include <stdio.h>
#include <assert.h>
int my_strcmp(const char* arr,const char* str)
{
	int ret = 0;
	assert(arr && str);
	while (*arr == *str)
	{
	    arr++;
	    str++;
	} 
	return ret = *arr - *str;
}

int main()
{
	char arr[] = "abcd";
	char str[] = "abce";
	int ret = my_strcmp(arr, str);
	if (ret > 0)
		printf("大于\n");
	else if (ret < 0)
		printf("小于\n");
	else
		printf("等于\n");
	return 0;
}

 

#include <stdio.h>
#include <assert.h>
int my_strncmp(char* s1, char* s2, size_t count)
{
	int ret = 0;
	assert(s1 && s2);
	while ((*s1 == *s2) && count)
	{
	    s1++;
		s2++;
		count--;
	} 
	return ret = *s1 - *s2;
}

int main()
{
	char str1[] = "abcef";
	char str2[] = "abcdg";
	int ret = my_strncmp(str1, str2,3);
	if (ret > 0)
		printf("大于\n");
	else if (ret < 0)
		printf("小于\n");
	else
		printf("等于\n");
	return 0;
}

*4.strstr
①作用:在字符串中查找字符串
②用法:char *strstr( const char *string, const char *strCharSet )
③头文件:<string.h>
④注意:
    返回的是被查找字符第一次出现查找字符的首元素地址
⑤模拟实现
 

#include <stdio.h>
#include <assert.h>
char* my_strstr(const char* s1, const char* s2)
{
	assert(s1 && s2);
	const char* p1 = s1;
	const char* p2 = s2;
	const char* start = p1;
	if (*s2 == '\0')
		return s1;
	while (*start)
	{
		p2 = s2;
		p1 = start;
		while (*p1 == *p2 && *p1 && *p2)
		{
			p1++;
			p2++;
		}
		if (*p2 == '\0')
		{
			 return (char*)start;
		}
		start++;
	}
	return NULL;
}

int main()
{
	char str1[] = "abbbcdef";
	char str2 [] = "bbc";
	char* ret = my_strstr(str1, str2);
	if (ret == NULL)
	{
		printf("找不到\n");
	}
	else
	{
		printf("找到了,下标是%s\n", ret);
	}
	return 0;
}

5.strtok

①作用:切割字符串
②用法:char *strtok( char *strToken, const char *strDelimit )
③头文件:<string.h>
④注意:
a.strDelimit参数是个字符串,定义了用作分隔符的字符集合
b.第一个参数指定一个字符串,它包含了0个或者多个由strDelimit字符串中一个或者多个分隔符分割的标记
c.strtok
函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注:strtok函数会改
变被操作的字符串,所以在使用
strtok函数切分的字符串一般都是临时拷贝的内容并且可修改。)
d.strtok
函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置。
strtok
函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标记。
e.如果字符串中不存在更多的标记,则返回 NULL 指针

6.strerror
①作用:返回错误码所对应的错误信息
②用法:char *strerror( int errnum )
③头文件:<string.h>

7.字符分类函数

iscntrl    任何控制字符

isspace    空白字符:空格‘ ’,换页‘\f’,换行'\n',回车‘\r’,制表符'\t'或者垂直制表符'\v'

isdigit    十进制数字 0~9

isxdigit    十六进制数字,包括所有十进制数字,小写字母a~f,大写字母A~F

islower    小写字母a~z

isupper    大写字母A~Z

isalpha    字母a~zA~Z

isalnum   字母或者数字,a~z,A~Z,0~9

ispunct    标点符号,任何不属于数字或者字母的图形字符(可打印)

isgraph    任何图形字符

isprint    任何可打印字符,包括图形字符和空白字符

8.memcpy

①作用:内存拷贝
②用法:void *memcpy( void *dest, const void *src, size_t count )
③头文件:<string.h>
④注意:
a.函数memcpysrc的位置开始向后复制num个字节的数据到dest的内存位置。
b.这个函数在遇到 '\0'
的时候并不会停下来。
c.如果source
destination有任何的重叠,复制的结果就是未定义的。
⑤模拟实现
 

#include <stdio.h>
#include <assert.h>
void* my_memcpy(void* dest, const void* src, size_t count)
{
	void* ret = dest;
	assert(dest && src);
	while(count--)	
	{
		*(char*)dest = *(char*)src;
		++(char*)dest;
		++(char*)src;
		//dest = (char*)dest+1;
		//src = (char*)src+1;
	}
	return ret;
}
int main()
{
	int arr1[10] = {0};
	int arr2[] = {1,2,3,4,5,6 };
    my_memcpy(arr1, arr2, 20);
	return 0;
}

*9.memmove

①作用:内存移动
②用法:void *memmove( void *dest, const void *src, size_t count )
③头文件:<string.h>
④注意:
a.和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
b.如果源空间和目标空间出现重叠,就得使用memmove
函数处理。
⑤模拟实现
 

#include <stdio.h>
#include <assert.h>
void* my_memmove(void* des, void* src, size_t count)
{
	assert(des && src);
	if (src < des) //从后向前
	{
		while (count--)
		{
			*((char*)des+count) = *((char*)src+count);
		}
	}
	else //从前向后
	{
		while (count--)
		{
			*((char*)des) = *((char*)(src));
			((char*)des)++;
			((char*)src)++;
		}
	}
}

int main()
{
	int arr[] = {1,2,3,4,5,6,7,8,9,10};
	my_memmove(arr+2, arr, 16);
	return 0;
}

10.memcpy
①作用:内存比较
②用法:void *memcpy( void *dest, const void *src, size_t count )
③头文件:<string.h>
④注意:
比较从dest和src指针开始的count个字节
⑤模拟实现

10.memset
①作用:内存初始化
②用法:void *memset( void *dest, int c, size_t count )
③头文件:<string.h>
④注意:
dest为初始化字符串首元素地址,c为要初始化成的字符,count为初始化字符串大小,单位是字节
 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

常用的字符函数和字符串函数 的相关文章

随机推荐

  • 阿里云Linux服务器安装可视化桌面,登录VNC提示Login incorrect*

    登录VNC提示Login incorrect 解决方法只要分为两步 xff1a 1 重置VNC密码 xff1a 先重置VNC密码 xff0c 重置后重启 xff0c 按照要求重置 xff0c 重置成功后会看到右上角出现一个重置成功的字样 2
  • 在计算机视觉论文中benchmark和baseline的区别

    总结 benchmark一般是和同行中比较牛的算法比较 xff0c 比牛算法还好 xff0c 那你可以考虑发好一点的会议 期刊 xff1b baseline一般是自己算法优化和调参过程中自己和自己比较 xff0c 目标是越来越好 xff0c
  • Gazebo仿真建模总结

    Gazebo仿真建模总结 目录 Gazebo仿真建模总结前言介绍总体框架框架介绍1 机器人模型urdf xacro2 模型运行平台rvizdisplayszh launch启动文件代码 xff1a 3 模型仿真平台gazebo1 xff09
  • c++工程中make和cmake的使用(CMakeLists.txt)

    一 make和cmake 1 make make xff0c 常指一条计算机指令 xff0c 可以从一个名为Makefile的文件中获得如何构建程序的依赖关系 通常项目的编译规则就定义在makrfile 里面 xff0c 比如 xff1a
  • 2. 阿里云上搭建ubuntu16.04并使用VNC进行远程连接

    首先 xff0c 用xshell连接阿里云 xff08 便于后续操作方便 xff09 参考文档 xff1a 阿里云服务器 xff08 Ubuntu16 04 64位 xff09 远程连接 再来创建新用户 参考文档 xff1a 为Ubuntu
  • 【哈工大李治军】操作系统课程笔记9:设备驱动与文件管理(显示器、键盘和磁盘)

    xff08 应粉丝催更 xff0c 笔记提前放出来了 xff0c 还剩最后一个视频 xff0c 等有空了再整理 xff09 1 I O与显示器 xff08 终端设备输出 xff09 本次所学的I O设备主要归为两大类 xff1a 键盘和显示
  • MathType7新版本数学公式编辑器上线功能特性

    许多论文或文献中含有大量较复杂的公式或者符号 xff0c 为了使文章中的公式符号更加规范 美观 xff0c 现在很多人选用MathType软件来编辑公式 MathType具有非常强大的公式编辑能力 xff0c 和我们常用的Office软件结
  • LDO:低压差线性稳压芯片

    LDO LDO即low dropout regulator xff0c 是一种 低压差 线性稳压器 这是相对于传统的线性稳压器来说的 传统的线性稳压器 xff0c 如78XX系列的芯片都要求输入电压要比输出电压至少高出2V 3V xff0c
  • git使用gitee 仓库教程详细

    1 先再本地创建 一个git 仓 先创建一个文件夹 在文件夹内运行git 执行git init 命令生成git 仓库 生成git 仓库之后 创建一个想要上传到云库的文件 xff0c 然后通过 git add 添加所有文件命令 添加跟踪 也叫
  • realsenseD435i运行vins-mono

    目录 写在前面准备编译vins mono 修改launch realsense vins mono 运行参考完 写在前面 1 本文内容 realsenseD435i运行vins mono 2 平台 ubuntu1804 ros melodi
  • Hive源码阅读--SQL的语法解析和语义分析--Driver

    前面五个类 xff0c 殊途同归都是CliDriver类 xff0c 他负责接受用户在命令行上输入的信息 xff0c 然后准备执行并将执行的结果返回 而真正底层干事情的是Driver xff0c 他将接受到的命令编译 xff0c 优化为MR
  • C语言字符串结束符“\0”

    C语言字符串结束符 0
  • STM32 PCB设计

    看了自己2018年2月画的PCB和现在2019年7月画的PCB xff0c 不多说了 不多说了 不多说了上图 96 2018年2月 96 2019年7月 大家可以明显看到第一块板子简直就是惨不忍睹 说几点PCB设计注意事项和自己的感想 xf
  • Java 调用第三方接口方法

    Java 调用第三方接口方法 一 通过JDK网络类Java net HttpURLConnection 1 java net包下的原生java api提供的http请求 使用步骤 xff1a 1 通过统一资源定位器 xff08 java n
  • docker基本命令及使用实例

    docker基本命令 特别鸣谢 xff1a B站up主 狂神说java的视频 xff0c 让我能短时间了解docker docker文件系统 docker镜像为分层设计 xff0c 相比于全量的虚拟机镜像 xff0c 少了引导程序bootf
  • ubantu 系统分区介绍

    假设分区有40个G 1 500M引导分区 2 10个G的swap分区 swap是一个内存交换空间 xff0c 当内存溢出 xff0c 或者计算机打算休眠的时候 会将数据存在swap分区中 在windows系统中是三个文件 swap的大小通常
  • H3C交换机常用命令(初学)

    一 显示交换机当前的一些配置信息 1 display current configuration 显示当前的一些配置信息 xff0c 如vlan xff0c 端口详细信息 2 display saved configuration 显示下次
  • Linux下的Tcp通信项目范例【demo】

    一 适合阅读对象 2 4个月的初学者 C语言编程方向 xff09 二 项目内容 xff1a 设计一个可以符合多用户进行线上查阅乐器的商城 xff0c 要求可以多个用户查看 xff0c 管理员可随时修改内容 xff0c 普通用户仅可查看 xf
  • 平衡小车之陀螺仪调参

    一 平衡小车 平衡小车作为控制进阶 xff0c 成为每个学习PID控制算法者的必经之路 xff0c 可难可易 xff0c 找到方法则非常容易 xff0c 找不到方法可能盲调数天也没有收获 要让小车平衡 xff0c 我认为需要掌握的技能有几点
  • 常用的字符函数和字符串函数

    一 分类 求字符串长度 xff1a strlen长度不受限制的字符串函数 xff1a strcpy strcat strcmp长度受限制的字符串函数 xff1a strncpy strncat strncmp字符串查找 xff1a strs