首页 > 其他分享 >部分库函数及其模拟

部分库函数及其模拟

时间:2024-09-03 22:52:29浏览次数:19  
标签:const int 及其 destination char source 字符串 模拟 库函数

 前言:当我们学习c/c++库函数的时候,我们可以用网站

cplusplus.com - The C++ Resources Network

来进行查阅,学习。 

库函数:

1.字符串函数

字符串函数一般要用到头文件:<string.h> 

1.1求字符串长度

strlen

size_t strlen ( const char * str );

1.首先,字符串是以\0来结尾的,strlen返回的是字符串中" \0 "之前的元素个数,不包括 " \0 "

2.参数指向的字符必须以" \0 "来结尾

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

使用

函数模拟:

1.计数器:

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

2.不能使用计数器

int my_strlen(char* str)
{
	if (*str == '\0')
		return 0;
	else
	{
		return 1 + my_strlen(str + 1);
	}
}

3.指针

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

}

1.2长度不受限制的字符串函数

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

1.源字符串必须以“\0”结尾

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

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

4.目标空间必须可变

int main()
{
	char arr1[10] = "1234567";
	char arr2[20] = "0";
	printf("%s\n", arr2);
	strcpy(arr2, arr1);
	printf("%s\n",arr2);
	return 0;
}

模拟:

char* my_strcpy(char* destination, const char* source)
{
	assert(destination && source);
	char* ret = destination;
	while (*source)
	{
		*destination = *source;
		destination++;
		source++;
	}
	return ret;
}

 因为我们的源地址,只是用来被拷贝,其指针指向的内容不能被改变,所以我们加上一个const修饰,而且,destination和source都不能是空指针,所以我们用assert断言一下。

 

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

一个字符串从一个字符串的终止符号加上去

#include<stdio.h>
#include<string.h>

int main()
{
	char arr[80] = { 0 };
	strcat(arr, " you ");
	strcat(arr, " are ");
	strcat(arr, " the ");
	strcat(arr, " best ");
	char arr1[20] = "ef";
	char arr2[] = "abcd";
	strcat(arr1,arr2);
	printf("%s\n" ,arr2);
	printf("%s\n", arr);
	return 0;
}

 我们来验证strcat是否是从目标字符串的终止符号(就是“\0”)来进行操作的

我们用上面的代码在vs2022上按快捷键f10,进行调试。

我们打开监视窗口,在窗口上输入arr1和arr2我们来进行观察

 

然后按下f10到14行位置,此时,arr1和arr2已经完成初始化。

 

我们的目标字符串是arr1,arr2是源字符串,此时我们按下f10

观察到arr1数组的第3个元素从" \0 "变成了字符‘ a ’,由此我们得以验证,strcat是从目标字符的终止字符“ \0 ”来进行操作的。

下面我们来模拟这个函数:

1.这个函数是用来把一个字符串放到另一个字符串的末尾

2.是从目标字符串的末尾来进行操作的

3.返回类型是目标字符串的起始地址

char* my_strcat(char* destination, const char* source)
{
	char ret = destination;
	while (*destination)
	{
		destination++;
	}
	while ((*destination++ = *source++))
	{
		;
	}
	return ret;
}
1.2.3strcmp

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

两个字符串进行比较,返回一个int类型的值

比较到出现另个字符不一样或者一个字符串结束或者n个字符全部比较完

如果出现第一个不一样的字符,第一个字符串的这个字符比第二个字符串的这个字符小就返回一个<0的数

如果出现第一个不一样的字符,第一个字符串的这个字符比第二个字符串的这个字符大就返回一个>0的数

如果比较结束都不一样就返回0;

int main()
{
	char arr1[] = "abcde";
	char arr2[] = "abcdf";
	int ret = strcmp(arr1, arr2);
	printf("%d ", ret);
	return 0;
}


int main()
{
	char arr1[] = "abcde";
	char arr2[] = "abcde";
	int ret = strcmp(arr1, arr2);
	printf("%d ", ret);
	return 0;
}

int main()
{
	char arr1[] = "abcdf";
	char arr2[] = "abcde";
	int ret = strcmp(arr1, arr2);
	printf("%d ", ret);
	return 0;
}

 

 我们下面来模拟这个函数

1.当两个字符串不同相同时,终止循环并进行比较

2.当2个字符串比完之后,终止循环并返回0

int my_strcmp(const char* str1, const char* str2)
{
	while ((*str1 == *str2) || *str1 != '\0')
	{
		str1++;
		str2++;
	}
	if (*str1 < *str2)
		return -1;
	else if(*str1 > *str2)
		return 1;
	else 
		return 0;
}

1.3长度受限制的字符串函数介绍

1.3.1strncpy
char * strncpy ( char * destination, const char * source, size_t num );
拷贝 num 个字符从源字符串到目标空间。 如果源字符串的长度小于 num ,则拷贝完源字符串之后,在目标的后边追加 0 ,直到 num 个。

 

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

 

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

 

1.4字符串查找

1.4.1strstr
const char * strstr ( const char * str1, const char * str2 );
      char * strstr (       char * str1, const char * str2 );

 

1.4.2strtok

char * strtok ( char * str, const char * delimiters );
sep 参数是个字符串,定义了用作分隔符的字符集合 第一个参数指定一个字符串,它包含了0 个或者多个由 sep 字符串中一个或者多个分隔符分割的标 记。 strtok 函数找到 str 中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注: strtok函数会改变被操作的字符串,所以在使用 strtok 函数切分的字符串一般都是临时拷贝的内容并且可修改。) strtok 函数的第一个参数不为 NULL ,函数将找到 str 中第一个标记, strtok 函数将保存它在字符串 中的位置。 strtok 函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标 记。 如果字符串中不存在更多的标记,则返回 NULL 指针。

1.5.错误信息报告

strerror

char * strerror ( int errnum );

返回错误码,所对应的错误信息。

 必须包含头文件#include<errno.h>

2.字符函数

1.字符分类函数

 函数如果他的参数符合以下条件就返回真
iscntrl 任何控制字符
isspace 空白字符:空格 ‘ ’ ,换页 ‘\f’ ,换行 '\n' ,回车 ‘\r’ ,制表符 '\t' 或者垂直制表符 '\v'
isdigit 十进制数字 0~9
isxdigit 十六进制数字,包括所有十进制数字,小写字母 a~f ,大写字母 A~F
islower 小写字母 a~z
isupper 大写字母 A~Z
isalpha 字母 a~z 或 A~Z
isalnum 字母或者数字, a~z,A~Z,0~9
ispunct 标点符号,任何不属于数字或者字母的图形字符(可打印)
isgraph 任何图形字符
isprint 任何可打印字符,包括图形字符和空白字符

 

2.字符转换函数

 int tolower ( int c );

 int toupper ( int c );

 

#include<stdio.h>

#include <ctype.h>
int main()
{
	int i = 0;
	char str[] = "Test String.\n";
	char c;
	while (str[i])
	{
		c = str[i];
		if (isupper(c))
			c = tolower(c);
		putchar(c);
		i++;
	}
	return 0;
}

3.内存函数

1.memcpy

void * memcpy ( void * destination, const void * source, size_t num );

这里我们发现,返回类型还有参数类型都是void*,这是为了方便接收任意指针,

函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。这个函数在遇到 '\0' 的时候并不会停下来。如果source和destination有任何的重叠,复制的结果都是未定义的。

2.memmove

void * memmove ( void * destination, const void * source, size_t num );

3.memcmp

int memcmp ( const void * ptr1, const void * ptr2, size_t num );

 

标签:const,int,及其,destination,char,source,字符串,模拟,库函数
From: https://blog.csdn.net/2302_77268606/article/details/141273204

相关文章

  • 存储器的分类及其特点
    只读存储器ROM原文来自ROM的分类(https://www.cnblogs.com/softhal/p/5640847.html)FLASH部分的原文(https://blog.csdn.net/xinxinyouyi/article/details/100571824)随机访问存储器RAM(RANDMONACCESSMEMORY)原文(https://blog.csdn.net/qq_42973834/article/details/108676693)......
  • 8.30 上午 becoder 模拟赛总结 & 题解
    T1密码当时想到解法了,却依然认为自己不会做,我真是个人才。结论:对于$\foralli\in[1,n)$,满足密码不是$a_i$的因数,且密码是$a_k$的因数,设满足条件的最小值为$g$则答案为$\frac{n}{g}$。一种最好想的做法:枚举$\gcd(a_k,n)$的因数作为$g$,并枚举$i\in[1,n)$,判断是......
  • 8.31 上午 becoder 模拟赛总结 & 题解
    T1四个质数的和赛场亲测搜索+小剪枝可以得到70pts。考虑$O(p(V)^2)$枚举任意两个质数的和,其中$p(V)$表示$V$以内质数的个数。然后开个数组记录下对于每种和的记录有多少种情况,查询时for循环扫一遍即可,详见代码。复杂度去掉质数筛$O(p(V)^2+tn)$,代码贴在下面(100pts)......
  • 8.31 下午 梦熊联盟 NOIP 模拟赛总结 & 题解
    T1北极星一个比较好想到的点是从后往前枚举数,计算得出它需要的操作次数,然后给所有前面的数都加上这个操作次数,这样就把每个数独立出来了。所以这道题就变成了如何快速通过这些操作得到一个指定的数。观察大样例的输出,发现每一个数都是11?1?1?的形式,其中问号为+或c,我们可......
  • 9.1 上午 becoder 模拟赛总结 & 题解
    T1货车运输Kruskal重构树模板,没什么好说的,不会的把自己重构了算了,跳过。T2Slagalica可以发现拼图1和2、3拼起来还是拼图1,拼图4和2、3拼起来也还是拼图4,这两种拼图还都不能和自己拼,所以我们可以看作只有拼图1和拼图4来做。对于边界拼图分别是5、7的情况,只有......
  • 9.2 上午 becoder 模拟赛总结 & 题解
    T1加法最开始看了好久没想出来,先做T2去了,然后回来想了一会儿发现自己可能智商有点问题。看到求最小值最大,第一反应肯定是二分,那我们怎么应该怎么check呢?考虑顺次枚举序列$A$中的每一个数,然后如果这个数没有达到mid的要求,我们肯定是要添加区间的。那么我们怎么添加区......
  • 9.3 上午 becoder 模拟赛总结 & 题解
    T1能量获取简单的树形DP,设$dp_{i,j}$表示向$i$节点传递了$j$点能量并全部花费完后能激活的封印石的数量。显然有:$ans=\sum\max_{j=0}^{j\leqW_i}{dp_{i,j}}(i\inson_0)$,转移的初始状态为$dp_{i,E_i}=E_i$。设当前枚举到的节点为$x$,子节点为$y$,有经典树上背包转......
  • Python深入理解快速排序算法及其时间复杂度分析
    Python深入理解快速排序算法及其时间复杂度分析快速排序(QuickSort)是一种高效的排序算法,广泛应用于各种实际场景中。它采用分治法(DivideandConquer)策略,通过选择一个基准元素(pivot),将数组分成两部分,使得左侧部分的元素都小于基准元素,右侧部分的元素都大于基准元素。然后递......
  • 调用azure的npm实现outlook_api模拟查看邮件、发送邮件(实现web版接受outlook邮件第一
    文章目录⭐前言⭐注册azure应用......
  • ERA5数据下载及其预处理
    1、ERA5数据下载(1)下载地址:https://cds-beta.climate.copernicus.eu/datasets(2)进入网站进行注册、登录(建议选择一个翻译插件,可以直观的看看)(3)下载步骤----常规下载    [1]搜索框输入ERA5,点击search    [2]从出现的这几个中进行选择,我选择的是月平均数......