C语言进阶——字符串函数和内存函数

news2024/11/27 13:37:18

目录

一.   strlen

二.   strcpy 

三.   strcat

四.   strcmp 

五.   strncpy

六.   strncat

 七.   strncmp

八.   strstr

九.   strtok


一.   strlen

字符串以 '\0' 作为结束标志,strlen函数返回的是在字符串中 '\0' 前面出现的字符个数(不包
含 '\0' )。

参数指向的字符串必须要以 '\0' 结束。

注意函数的返回值为size_t,是无符号的
 

模拟实现

//计数
int my_strlen1(const char* s)
{
	int count = 0;
	while (*s++ != '\0')
	{
		count++;
	}
	return count;
}

//递归
int my_strlen2(const char* s)
{
	if (*s != '\0')
		return 1 + my_strlen2(s + 1);
	return 0;
}

//指针相减
int my_strlen3(const char* s)
{
	char* tail = s-1;
	while (*++tail!= '\0')
	{
		;
	}
	return tail-s;
}

而在我们的模拟实现中,返回值使用的是int,而库函数的返回值是无符号型,这两种其实各有利弊,使用int若字符串长度较大会出现负数的情况,而使用无符号型无法实现strlen相减得到字符串长度的差值。



二.   strcpy 

源字符串必须以 '\0' 结束。

会将源字符串中的 '\0' 拷贝到目标空间。

目标空间必须足够大,以确保能存放源字符串。

目标空间必须可变。

模拟实现

char* my_strcpy(char* s1,const char* s2)
{
	assert(s1 && s2);
	while (*s2 != '\0')
	{
		*s1++ = *s2++;
	}
    return s1;
}


三.   strcat


源字符串必须以 '\0' 结束。

目标空间必须有足够的大,能容纳下源字符串的内容。

目标空间必须可修改。

模拟实现

char* my_strcat(char* s1,const char* s2)
{
	assert(s1 && s2);
	while (*s1++ != '\0')
	{
		;
	}
	s1--;
	while (*s2 != '\0')
	{
		*s1++ = *s2++;
	}
    return s1;
}

我们再考虑一个问题,s1和s2能否相同呢

int main()
{
	char s[20] = "abc";
	my_strcat(s, s);
	printf("%s\n", s);
	return 0;
}



可以看到,程序发生了崩溃


 可以看到,若是s1与s2相同,首先s1要加至‘\0’处,在进行赋值时,‘\0’被替换为a,这也就使得s2在向后移动时找不到‘\0’,进而导致死循环。



四.   strcmp 


 

模拟实现

int  my_strcmp(const char* s1,const char* s2)
{
	int ret = 0;
	assert(s1 && s2);
	while (*s1 == *s2)
	{
		s1++;
		s2++;
	}
	if (*s1 > *s2)
		ret = 1;
	if (*s1 < *s2)
		ret = -1;
	return ret;
}


五.   strncpy

 与strcpy的功能类似,只是多了一个参数num,决定拷贝的字符串长度

若拷贝的字符串小于num,则补‘\0’

模拟实现

char* my_strncpy(char* s1, const char* s2, int num)
{
	assert(s1 && s2);
	while (*s2 != '\0'&&num>0)
	{
		*s1++ = *s2++;
		num--;
	}
	return s1;
}


六.   strncat

与strcat的功能类似,只是多了一个参数num,决定拼接的字符串长度

模拟实现

char* my_strncat(char* s1, const char* s2, int num)
{
	assert(s1 && s2);
	while (*s1 != '\0')
	{
		s1++;
	}
	while (*s2 != '\0'&&num>0)
	{
		*s1++ = *s2++;
		num--;
	}
	*s1 = '\0';
    return s1;
}


 七.   strncmp

 num同上

模拟实现

int  my_strncmp(const char* s1, const char* s2, int num)
{
	assert(s1 && s2);
	int ret = 0;
	while (*s1 == *s2&&num>0)
	{
		num--;
		if (num != 0)
		{
			s1++;
			s2++;
		}
		if (*s1 == '\0' || *s2 == '\0')
			break;
	}
	if (*s1 > *s2)
		ret = 1;
	if (*s1 < *s2)
		ret = -1;
	return ret;
}


八.   strstr

模拟实现

char*  my_strstr(const char* s1, const char* s2)
{
	assert(s1 && s2);
	char* head1 = s1;
	char* head2 = s2;
	while(*s1!='\0')
	{
		while (*s1 != '\0' && *s1 != *s2)
			s1++;
		if (s1 == '\0')
			return NULL;
		head1 = s1;
		while (*s1 == *s2)
		{
			s1++;
			s2++;
			if (*s2 == '\0')
				return head1;
		}
		s1 = head1 + 1;
		s2 = head2;
	}
	return NULL;
}


九.   strtok

sep参数是个字符串,定义了用作分隔符的字符集合

第一个参数指定一个字符串,它包含了0个或者多个由sep字符串中一个或者多个分隔符分割的标记。

strtok函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注:strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容并且可修改。)

strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置。

strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标
记。

如果字符串中不存在更多的标记,则返回 NULL 指针。

int main()
{
    char* p = "sizhitong@abc.com";
    const char* sep = ".@";
    char arr[30];
    strcpy(arr, p);
    char* str;
    str = strtok(arr, sep);
    printf("%s\n", str);
    str = strtok(NULL, sep);
    printf("%s\n", str);
    str = strtok(NULL, sep);
    printf("%s\n", str);
    return 0;
}


 

而若是想要直接打印出去除间隔符后的结果,我们可以利用循环

int main()
{
    char *p = "sizhitong@abc.com";
    const char* sep = ".@";
    char arr[30];
    char *str = NULL;
    strcpy(arr, p);
    for(str=strtok(arr, sep); str != NULL; str=strtok(NULL, sep))
    {
        printf("%s\n", str);
    }
    return 0;
}


十.   memcpy

函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。

这个函数在遇到 '\0' 的时候并不会停下来。

如果source和destination有任何的重叠,复制的结果都是未定义的。
 

模拟实现

void* my_memcpy(void* source,void* destination,int num)
{
	for(int i=0;i<num;i++)
	{
		*((char*)source + i) = *((char*)destination + i);
	}
	return source;
}

int main()
{
	int nums1[10] = { 1,2,3,4,5,6,7 };
	int nums2[10] = { 3,4,5,6,7,8,9 };
	my_memcpy(nums1, nums2, 5 * sizeof(int));
	for (int i = 0; i < 7; i++)
	{
		printf("%d ", nums1[i]);
	}
	printf("\n");
	return 0;
}

实现起来还是很容易的,但是,我们还是要思考一下source和destination重叠的问题

例如我们给出这样的用例

void* my_memcpy(void* source,void* destination,int num)
{
	for(int i=0;i<num;i++)
	{
		*((char*)source + i) = *((char*)destination + i);
	}
	returns source;
}

int main()
{
	int nums[10] = { 1,2,3,4,5,6,7 };
	my_memcpy(nums+2, nums, 5 * sizeof(int));
	for (int i = 0; i < 7; i++)
	{
		printf("%d ", nums[i]);
	}
	printf("\n");
	return 0;
}

我们想要的结果应该是这样的

1 2 1 2 3 4 5 

而实际上却是这样的

我们来分析一下

 我们可以得知,source和dest都要遍历后面的五个数据,而source的后三个数据与dest部分重叠,这也就导致在source移动到一开始dest的位置时,3已经被改变为1,因此会出现上面的情况,这个问题我们会在下一个函数memmove中解决,而在我们使用vs中的memcpy,并没有出现上述的问题,这个问题我们也在介绍memmove中说明



十一.   memmove

 

和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。

如果源空间和目标空间出现重叠,就得使用memmove函数处理。

在一些编译器中,将库中的memcpy修改得与memmove类似,使得在这些编译器中memcpy也可以处理重叠的数组。

模拟实现

首先memmove的模拟实现与memcpy类似,只需要再考虑重叠的问题即可

首先便是memcpy中出问题的数据

void* my_memmove(void* source, void* destination, int num)
{
	for (int i = 0; i < num; i++)
	{
		*((char*)source + i) = *((char*)destination + i);
	}
	return source;
}

int main()
{
	int nums[10] = { 1,2,3,4,5,6,7 };
	my_memmove(nums+2, nums, 5 * sizeof(int));
	for (int i = 0; i < 7; i++)
	{
		printf("%d ", nums[i]);
	}
	printf("\n");
	return 0;
}

可以看到,这组数据中,source与destination之间有重叠部分并且source地址高于destination的地址。

那么我们有什么解决方法呢?

从前向后赋值,source的值会被dest改变,那么我们可以选择从后向前赋值

void* my_memmove(void* source, void* destination, int num)
{
	while(num--)
	{
		*((char*)source + num) = *((char*)destination + num);
	}
	return source;
}

int main()
{
	int nums[10] = { 1,2,3,4,5,6,7 };
	my_memmove(nums+2, nums, 5 * sizeof(int));
	for (int i = 0; i < 7; i++)
	{
		printf("%d ", nums[i]);
	}
	printf("\n");
	return 0;
}

的确,这样会解决以上的问题,但若是重叠且source的地址低于dest

int nums[10] = { 1,2,3,4,5,6,7 };
my_memmove(nums, nums+2, 5 * sizeof(int));


 

逆序会出现问题,而正序反而不会出现问题。

因此,我们需要进行条件判断,当dest的地址大于source的地址选择正序,小于则选择逆序,而且并不需要考虑是否重叠,因为若是不重叠,采用这两种方法都是可以的

void* my_memmove(void* source, void* destination, int num)
{
	while (source > destination && num--)
	{
		*((char*)source + num) = *((char*)destination + num);
	}
	if(source <= destination)
	{
		for (int i = 0; i < num; i++)
		{
			*((char*)source + i) = *((char*)destination + i);
		}
	}
	return source;
}


十二.   memcmp


 比较从ptr1和ptr2指针开始的num个字节
 

 实际上,memcmp与strncpy是非常类似的,只是相较来说,memcmp可以比较任何类型的数据,但若是num大于字符数组的字节数,即使遇到‘\0’,也不会停止,大多数情况下是没什么问题的,而当两个字符数组长度相同时,同时遇到'\0'后依旧会向后进行比较,导致越界。

模拟实现

int  my_memcmp(const void* s1, const void* s2, int num)
{
	assert(s1 && s2);
    for(int i=0;i<num;i++)
    {
        if(*((char*)s1+i)<*((char*)s2+i))
            return -1;
        if(*((char*)s1+i)>*((char*)s2+i))
            return 1;
    }
    return 0;
}

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/153460.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

TCP/IP网络编程(1)——理解网络编程和套接字编程

文章目录一、理解网络编程和套接字编程1.1 socket套接字1.1.1 一个例子来表示TCP的网络连接1.1.2 程序实现1.2 文件操作1.2.1 文件描述符一、理解网络编程和套接字编程 1.1 socket套接字 网络编程又称为套接字编程&#xff0c;为什么要用套接字&#xff1f;我们把插头插到插座…

Linux常用命令——ngrep命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) ngrep 方便的数据包匹配和显示工具 补充说明 ngrep命令是grep命令的网络版&#xff0c;他力求更多的grep特征&#xff0c;用于搜寻指定的数据包。正由于安装ngrep需用到libpcap库&#xff0c; 所以支持大量的操…

【Java】面向对象笔记

开篇 主线 类及类的成员 属性、方法、构造器&#xff1b; 代码块&#xff0c;内部类 面向对象三大特征 继承、封装、多态 其他关键字 this,super,static,final,abstract,interface等 面向对象的两个要素 一、是什么 类&#xff1a;对一类事物的描述&#xff0c;是抽象的…

Java综合练习

Java综合练习一、涉及到的知识点二、卖飞机票三、找质数四、开发验证码五、数组元素的复制六、评委打分七、数字加密八、数字解密九、抢红包方法一&#xff1a;判断是否被抽取方法二&#xff1a;打乱数组十、模拟双色球土、二维数组一、涉及到的知识点 变量、数组运算符程序流…

python采集IP代理数据,防止数据采集IP被封情况

前言 嗨喽&#xff0c;大家好呀~这里是爱看美女的茜茜呐 又到了学Python时刻~ 为什么要IP代理&#xff1a; 当采集数据, 批量采集数据, 请求速度过快, 网站可能会把你IP封掉 <你的网络进不去这个网站> IP代理换一个IP, 再去采集请求数据 一. 抓包分析数据来源 1. 明…

Runtime、ProcessBuilder的区别(Java中,两种方法来启动其他程序)

目录 ■Runtime、ProcessBuilder 区别&#xff1a; ■Java中提供了两种方法来启动其他程序 ■代码 ・Runtime ・ProcessBuilder ■类的方法 ・Process.waitFor()方法 ・Process.getErrorStream()方法 ・Process.redirectErrorStream(true)方法: ■可运行代码 ・java…

解读YOLOv8的改进模块

回顾一下YOLOv5 Backbone&#xff1a;CSPDarkNet结构&#xff0c;主要结构思想的体现在C3模块&#xff0c;这里也是梯度分流的主要思想所在的地方&#xff1b;PAN-FPN&#xff1a;双流的FPN&#xff0c;必须香&#xff0c;也必须快&#xff0c;但是量化还是有些需要图优化才可以…

“数据二十条”发布背后:国企下场探路,技术路径日渐清晰

科技云报道原创。 近日&#xff0c;中共中央、国务院对外发布了《关于构建数据基础制度更好发挥数据要素作用的意见》&#xff08;又称“数据二十条”&#xff09;&#xff0c;为数据要素的流通和使用起到了举旗定向的作用&#xff0c;让从业者都感到振奋。 必须承认的是&…

通信原理与MATLAB(十二):MSK的调制解调

目录1.MSK的调制原理2.MSK的解调原理3.MSK代码4.结果图5.特点1.MSK的调制原理 MSK调制原理如下图所示&#xff0c;基带码元先差分编码&#xff0c;然后经过串并转换分成I、Q两路&#xff0c;再与对应的载波相乘&#xff0c;然后再相加完成QAM的调制。 其中注意:I、Q两路码元分…

【深度学习】LSTM预测股票价格

入行深度学习1年多了&#xff0c;该还的还得还&#xff0c;没接触过LSTM的预测项目&#xff0c;这就来活了。 文章目录前言1. 开工1.1 引入必须的库1.2 数据初探1.3 划分数据集1.4 数据归一化1.5 数据分组1.6 搭建模型1.7 训练1.8 测试集总结前言 LSTM是一个处理时序关联的数据…

数据结构-第七期——并查集的应用(Python)

在学习并查集的应用之前&#xff0c;请大家先学习第六期-并查集的入门 ,这样会比较好理解 真题训练1 合根植物2017年第八届决赛&#xff0c;lanqiao0J题号110 【题目描述】 w 星球的一个种植园&#xff0c;被分成 mn 个小格子&#xff08;东西方向 m 行&#xff0c;南北方向…

dubbo-admin安装

一、dubbo-admin安装 1、环境准备 dubbo-admin 是一个前后端分离的项目。前端使用vue&#xff0c;后端使用springboot&#xff0c;安装 dubbo-admin 其实就是部署该项目。我们将dubbo-admin安装到开发环境上。要保证开发环境有jdk&#xff0c;maven&#xff0c;nodejs 安装n…

基于Python的geopandas库处理矢量几何的教程

前言在许多工作中中&#xff0c;我使用 ArcGIS 平台从事过许多与地理空间相关的项目&#xff0c;我非常喜欢这个平台。 这意味着我可以在具有尖端地理空间技术的项目中进行咨询&#xff0c;例如多维栅格、深度学习和空间物联网自动化。 考虑到这一点&#xff0c;我总是试图跟踪…

openstack增加一个计算节点

1.前言 由于资源有限&#xff0c;所以直接在存储节点&#xff08;block&#xff09;部署 由于存储节点最初只设计了一块网卡&#xff0c;所以需要增加一块网卡&#xff0c;名称为eth1&#xff0c;IP&#xff1a;192.168.200.30编辑ifcfg-eth1&#xff0c;然后重启网络systemct…

【优化】windows双网叠加 多网叠加 网速叠加 教程

【优化】windows双网叠加 多网叠加 网速叠加 教程 1 连接两个以上的网络, 网络不能是同一个 例如 网线-A wifi-B 2 控制面板\所有控制面板项\网络连接 最后 确定保存 同理 修改wifi-B的接口活跃点数为 25 并保存 如果没有生效 可以将两个网络连接禁用 再启用 通过命…

4656. 技能升级

4656. 技能升级 https://www.acwing.com/problem/content/4659/ 第十三届蓝桥杯省赛CC组 算法标签&#xff1a;贪心&#xff1b;多路归并&#xff1b;二分 思路 如果暴力来做的话&#xff0c;会将所有数放到一个集合里面排序&#xff0c;取前 mmm 项之和即可&#xff0c;但时…

Vue3——第六章(侦听器:watch、watchEffect)

一、watch 基本使用 在组合式 API 中&#xff0c;我们可以使用 watch 函数在每次响应式状态发生变化时触发回调函数&#xff1a; 二、侦听数据源类型 watch 的第一个参数可以是不同形式的“数据源”&#xff1a;它可以是一个 ref (包括计算属性)、一个响应式对象、一个 get…

5.2、运输层端口号、复用与分用的概念

1、端口号 运行在计算机上的进程使用进程标识符PID\color{red}进程标识符 PID进程标识符PID来标志。 因特网上的计算机并不是使用统一的操作系统 不同的操作系统(windows&#xff0c;Linux&#xff0c;Mac OS)又使用不同格式的进程标识符\color{red}不同格式的进程标识符不同…

ThreadLocal 实战应用

1 什么是 ThreadLocal&#xff1f;ThreadLocal 是一个关于创建线程局部变量的类。通常情况下&#xff0c;我们创建的变量是可以被任何一个线程访问并修改的。而使用 ThreadLocal 创建的变量只能被当前线程访问&#xff0c;其他线程则无法访问和修改。ThreadLocal 在设计之初就是…

CDN简单介绍

CDN 介绍 CDN (全称 Content Delivery Network)&#xff0c;即内容分发网络&#xff0c;服务器的静态资源存在CDN服务器上&#xff0c;用户在最近的CDN服务器上获取资源。 从功能上看&#xff0c;典型的 CDN 系统由分发服务系统、负载均衡系统和运营管理系统组成。分发服务系…