深入理解并模拟实现函数

目录

前言:

一、使用并模拟实现qsort

二、使用模拟实现strlen 

 三、使⽤和模拟实现strcpy 函数

四、使用并模拟实现strcat 函数

五、使用并模拟实现strcmp函数 

六、使用并模拟实现strstr函数 

七、使用strtok函数

八、使用strerror函数 

九、使用strncmp函数 

 十、使用strncat函数

十一、使用并模拟实现memcpy函数 

十二、使用并模拟实现memmove函数

十三、使用memset函数

十四、使用 memcmp函数

 十五、使用并模拟实现atoi函数


前言:

        我们前面已经学习了函数、指针等一系列知识,接下来,咱们一起来模拟实现strlen、qsort、strcpy、strcat、strcmp、strstr、memmove、memcpy、atoi,以及strncat、strncmp、strtok、strerror、memset、memcmp的使用。

一、使用并模拟实现qsort

        qsort可以对函数进行排序,使函数从无序变成有序。以下是官网解释:

           qsort函数也称之为快排。在确定好是降序还是升序后,经行快速排序,在所有排序中时间复杂度最低,效率更高。 

        下面是用代码演示一下:

#include <stdio.h>      /* printf */
#include <stdlib.h>     /* qsort */

int values[] = { 40, 10, 100, 90, 20, 25 };

int compare(const void* a, const void* b)
{
    return (*(int*)a - *(int*)b);
}

int main()
{
    int n;
    qsort(values, 6, sizeof(int), compare);
    for (n = 0; n < 6; n++)
        printf("%d ", values[n]);
    return 0;
}

        运行结果如下: 

         相信大家对此函数有了初步了解,那下面进行模拟实现(冒泡实现)。

#include<stdio.h>
#include<assert.h>
int cmp(const void* p1,const void* p2)
{
	return ((*(int*)p1) - (*(int*)p2));
}
void sweap(void* p1, void* p2, int sz)
{
	for (int i = 0; i < sz; i++)
	{
		int cmp = *((char*)p1+i);
		*((char*)p1 + i) = *((char*)p2 + i);
		*((char*)p2 + i) = cmp;
	}
}
void bubble(int* arr, int n, int sz, int(*cmp)(void*, void*))//使用void*为了适应各种指针类型
{
    assert(arr);
	for (int i = 0; i < n-1; i++)
	{
		for (int j = 0; j < n - 1 - i; j++)
		{
			if (cmp((char*)arr + j * sz, (char*)arr + (j + 1) * sz) > 0)
			{
				sweap((char*)arr + j * sz, (char*)arr + (j + 1) * sz,sz);//使用char*能够访问到每个比特位
			}
		}
	}
}
int main()
{
	int arr[10] = { 1,2,3,8,9,4,5,1,3,10 };
	bubble(arr, sizeof(arr)/sizeof(arr[0]),sizeof(int), cmp);
	for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
	{
		printf("%d", arr[i]);
	}
	return 0;
}

二、使用模拟实现strlen 

        strlen可以实现统计数组中元素个数(包括空格)。官网解释如下:

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

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

        注意函数的返回值为size_t,是⽆符号的( 易错 )

         strlen的使⽤需要包含头⽂件 

        使用代码如下:

/* strlen example */
#include <stdio.h>
#include <string.h>

int main ()
{
  char szInput[256];
  printf ("Enter a sentence: ");
  gets (szInput);
  printf ("The sentence entered is %u characters long.\n",(unsigned)strlen(szInput));
  return 0;
}

        对其模拟如下:

#include<stdio.h>
//法一:使用计数器方式
my_strlen(char* arr)
{
	int count = 0;
	while (*arr != 0)
	{
		count++;
		arr++;
	}
	return count;
}
//法二:使用递归方式
int my_strlen(char* arr)
{
	if (*arr == '\0')
	{
		return 0;
	}
	else
		return 1 + my_strlen(arr + 1);
}
//法三:使用指针-指针方式
int my_strlen(char* arr)
{
	char* s = arr;
	while (*arr != '\0')
	{
		arr++;
	}
	return arr - s;
}
int main()
{
	char arr[] = "12345";
	int ret = my_strlen(arr);
	printf("%d", ret);
	return 0;
}

        以上三种方式供大家学习。 

 三、使⽤和模拟实现strcpy 函数

        strcpy可将一个字符串拷贝到一个数组中。官网解释如下: 

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

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

         ⽬标空间必须⾜够⼤,以确保能存放源字符串。

        ⽬标空间必须可修改。

        代码实现如下:

/* strcpy example */
#include <stdio.h>
#include <string.h>

int main()
{
	char str1[] = "Sample string";
	char str2[40];
	char str3[40];
	strcpy(str2, str1);
	strcpy(str3, "copy successful");
	printf("str1: %s\nstr2: %s\nstr3: %s\n", str1, str2, str3);
	return 0;
}

        模拟实现如下: 

#include<stdio.h>
char* my_strcpy(char* dest, const char* src)
{
	char* str = dest;
	while (*dest++ = *src++)
	{
		;
	}
	return str;
}
int main()
{
	char arr1[] = "123456";
	char arr2[] = "abcdef";
	my_strcpy(arr1, arr2);
	printf("%s", arr1);
	return 0;
}

四、使用并模拟实现strcat 函数

          strcat函数是字符串追加函数,也就是在字符串后面追加另一个字符串。官网解释如下:

        使用 该函数代码如下:

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

int main()
{
	char str[80];
	strcpy(str, "these ");
	strcat(str, "strings ");
	strcat(str, "are ");
	strcat(str, "concatenated.");
	puts(str);
	return 0;
}

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

        ⽬标字符串中也得有 \0 ,否则没办法知道追加从哪⾥开始。

        ⽬标空间必须有⾜够的⼤,能容纳下源字符串的内容。

        ⽬标空间必须可修改。 

        模拟实现如下:

#include<stdio.h>
char* my_strcat(char* destination, const char* source)
{
	char* ret = destination;
	while (*destination)
	{
		destination++;
	}
	while (*destination++ = *source++)
	{
		;
	}
	return ret;
}
int main()
{
	char arr1[] = "hello";
	char arr2[] = " world";
	my_strcat(arr1,arr2);
	printf("%s", arr1);
	return 0;
}

五、使用并模拟实现strcmp函数 

        strcmp函数是C语言中字符串比较函数,用于比较两个字符串的大小。官网解释如下:

        如果相等则,返回零,若str>str2,则大于零,反之小于零。

        使用该函数如下:

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

int main()
{
    char key[] = "apple";
    char buffer[80];
    do 
    {
        printf("Guess my favorite fruit? ");
        fflush(stdout);
        scanf("%79s", buffer);
    } while (strcmp(key, buffer) != 0);
    puts("Correct answer!");
    return 0;
}

         模拟实现该函数如下:

#include<stdio.h>
#include<assert.h>
int my_strcmp(const char* str1, const char* str2)
{
	assert(str1 && str2);
	while (*str1++ == *str2++)
	{
		if (*str1 == *str2)
		{
			return 0;
		}
	}
	return *str1 - *str2;
}
int main()
{
	char arr1[] = "abcd";
	char arr2[] = "ace";
	int ret = my_strcmp(arr1, arr2);
	if (ret > 0)
	{
		printf("arr1>arr2\n");
	}
	else if (ret < 0)
	{
		printf("arr1<arr2\n");
	}
	else
	{
		printf("arr1=arr2\n");
	}
	return 0;
}

六、使用并模拟实现strstr函数 

        strstr函数是在字符串str1中查找是否含有字符串str2,如果存在,返回str2在str1中第一次出现的地址;否则返回NULL。官网解释如下:

        对其使用代码如下:

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

int main()
{
    char str[] = "This is a simple string";
    char* pch;
    pch = strstr(str, "simple");
    if (pch != NULL)
        strncpy(pch, "sample", 6);
    puts(str);
    return 0;
}

        Returns a pointer to the first occurrence of str2 in str1, or a null pointer if str2 is not part of str1. (函数返回字符串str2在字符串str1中第⼀次出现的位置)。

         The matching process does not include the terminating null-characters, but it stops there.(字符 串的⽐较匹配不包含 \0 字符,以 \0 作为结束标志)。 

         模拟实现该函数如下:

#include<stdio.h>
char* my_strstr(const char* str1, const char* str2)
{
	char* p = (char*)str1;
	if (!*str2)
	{
		return str1;
	}
	while (*p)
	{
		char* s1 = p;
		char* s2 = (char*)str2;
		while (*s1 && *s2 && !(*s1 - *s2))
		{
			s1++;
			s2++;
		}
		if (*s2)
		{
			return p;
		}
		p++;
	}
	return NULL;
}
int main()
{
	char arr1[] = "This is a simple string";
	char arr2[] = "simple";
	my_strstr(arr1, arr2);
	printf(arr2);
	return 0;
}

七、使用strtok函数

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

         sep参数指向⼀个字符串,定义了⽤作分隔符的字符集合

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

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

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

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

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

        官网解释如下:

        使用该函数:

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

int main()
{
	char str[] = "- This, a sample string.";
	char* pch;
	printf("Splitting string \"%s\" into tokens:\n", str);
	pch = strtok(str, " ,.-");
	while (pch != NULL)
	{
		printf("%s\n", pch);
		pch = strtok(NULL, " ,.-");
	}
	return 0;
}

八、使用strerror函数 

        strerror函数可以把参数部分错误码对应的错误信息的字符串地址返回来。 在不同的系统和C语⾔标准库的实现中都规定了⼀些错误码,⼀般是放在 errno.h 这个头⽂件中说明 的,C语⾔程序启动的时候就会使⽤⼀个全⾯的变量errno来记录程序的当前错误码,只不过程序启动 的时候errno是0,表⽰没有错误,当我们在使⽤标准库中的函数的时候发⽣了某种错误,就会讲对应 的错误码,存放在errno中,⽽⼀个错误码的数字是整数很难理解是什么意思,所以每⼀个错误码都是 有对应的错误信息的。strerror函数就可以将错误对应的错误信息字符串的地址返回。

         官网解释如下:

        代码使用函数:

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

int main()
{
	FILE* pFile;
	pFile = fopen("unexist.ent", "r");
	if (pFile == NULL)
		printf("Error opening file unexist.ent: %s\n", strerror(errno));
	return 0;
}

九、使用strncmp函数 

        strcmp函数与strncmp函数从函数名看来基本一样,确实,它们确实大体上相同,区别为:n确定了比较多少个元素,官网解释如下:

        其返回值也是相同。         对其使用如下:

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

int main()
{
	char str[][5] = { "R2D2" , "C3PO" , "R2A6" };
	int n;
	puts("Looking for R2 astromech droids...");
	for (n = 0; n < 3; n++)
		if (strncmp(str[n], "R2xx", 2) == 0)
		{
			printf("found %s\n", str[n]);
		}
	return 0;
}

 十、使用strncat函数

        strncat函数与strcat函数的区别也类似与上述函数,只不过区别是拼接字节的个数仅此而已。官网解释如下:

        使用该函数如下:

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

int main()
{
	char str1[20];
	char str2[20];
	strcpy(str1, "To be ");
	strcpy(str2, "or not to be");
	strncat(str1, str2, 6);
	puts(str1);
	return 0;
}

十一、使用并模拟实现memcpy函数 

         memcpy函数是C语言中内存函数,其功能和strcpy函数类似,以下为官网解释:

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

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

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

        对其使用结果如下:

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

struct {
	char name[40];
	int age;
} person, person_copy;

int main()
{
	char myname[] = "Pierre de Fermat";

	/* using memcpy to copy string: */
	memcpy(person.name, myname, strlen(myname) + 1);
	person.age = 46;

	/* using memcpy to copy structure: */
	memcpy(&person_copy, &person, sizeof(person));

	printf("person_copy: %s, %d \n", person_copy.name, person_copy.age);

	return 0;
}

         对其模拟代码如下:

#include<stdio.h>
#include<assert.h>
void* my_memcpy(void* destination, const void* source, size_t num)
{
	assert(destination && source);
	void* p = destination;
	while (num--)
	{
		*(char*)destination = *(char*)source;
		destination = (char*)destination + 1;
		source = (char*)source + 1;
	}
	return (p);
}
int main()
{
	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[20] = { 0 };
	my_memcpy(arr2, arr1,40);
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr2[i]);
	}
	return 0;
}

十二、使用并模拟实现memmove函数

        memmove函数也是内存函数,它的作用是可以实现对目标字符串的拷贝并放入指定位置。官网解释如下:

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

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

         试用该函数如下:

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

int main()
{
	char str[] = "memmove can be very useful......";
	memmove(str + 20, str + 15, 11);
	puts(str);
	return 0;
}

        模拟实现如下:

#include<stdio.h>
#include<assert.h>
void* my_memmove(void* dest, const void* sour, size_t num)
{
	void* p = dest;
	if (dest < sour)
	{
		while (num--)
		{
			*(char*)dest = *(char*)sour;
			dest = (char *)dest+1;
			sour = (char*)sour + 1;
		}
	}
	else
	{
		dest = (char*)dest + num - 1;
		sour = (char*)sour + num - 1;

		while (num--)
		{
			*(char*)dest = *(char*)sour;
			dest = (char*)dest - 1;
			sour = (char*)sour - 1;
		}
	}
	return p;
}
int main()
{
	char str[] = "memmove can be very useful......";
	my_memmove(str + 20, str + 15, 11);
	puts(str);
	return 0;
}

十三、使用memset函数

        memset是⽤来设置内存的,将内存中的值以字节为单位设置成想要的内容。官网解释如下:

        使用函数如下:

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

int main()
{
	char str[] = "almost every programmer should know memset!";
	memset(str, '-', 6);
	puts(str);
	return 0;
}

十四、使用 memcmp函数

        该函数即为比较函数与上文strncmp函数类似。官网解释如下:

        使用该函数如下: 

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

int main()
{
	char buffer1[] = "DWgaOtP12df0";
	char buffer2[] = "DWGAOTP12DF0";

	int n;

	n = memcmp(buffer1, buffer2, sizeof(buffer1));

	if (n > 0)
	{
		printf("'%s' is greater than '%s'.\n", buffer1, buffer2);
	}
	else if (n < 0)
	{
		printf("'%s' is less than '%s'.\n", buffer1, buffer2);
	}
	else
	{
		printf("'%s' is the same as '%s'.\n", buffer1, buffer2);
	}
	return 0;
}

 十五、使用并模拟实现atoi函数

它的功能:
解析将其内容解释为整数的 C 字符串,该整数作为 类型的值返回。
该函数首先根据需要丢弃尽可能多的空格字符(如 ),直到找到第一个非空格字符。然后,从此字符开始,取一个可选的首字母加号或减号,后跟尽可能多的 10 进制数字,并将它们解释为数值。
字符串可以在构成整数的字符之后包含其他字符,这些字符将被忽略,并且对此函数的行为没有影响。
如果 中的第一个非空格字符序列不是有效的整数,或者由于为空或仅包含空格字符而不存在此类序列,则不执行转换并返回零。

         官网解释:

        使用该函数如下:

#include <stdio.h>
#include <string.h>
int main()
{
	int i;
	char buffer[256];
	printf("Enter a number: ");
	fgets(buffer, 256, stdin);
	i = atoi(buffer);
	printf("The value entered is %d. Its double is %d.\n", i, i * 2);
	return 0;
}

        模拟实现如下:

#include<stdio.h>
#include<assert.h>
#include<ctype.h>
#include<stdlib.h>
enum State
{
	INVAILD,
	VAILD
}a;
int my_atoi(const char* str)
{
	assert(str);
	if (*str == '\0')
	{
		return 0;
	}
	while (isspace(*str))
	{
		str++;
	}
	int flag = 1;
	if (*str == '+')
	{
		flag = 1;
		str++;
	}
	if (*str == '-')
	{
		flag = -1;
		str++;
	}
	long long ret = 0;
	while (*str != '\0')
	{
		if (isdigit(*str))
		{
			ret = ret * 10 + flag * (*str - '0');
			if (ret > INT_MAX)
			{
			return INT_MAX;
			}
			if (ret < INT_MIN)
			{
			return INT_MIN;
			}
		}
		else
		{
			 (int)ret;
		}
		str++;
	}
	if (*str == '\0')
	{
		 a =VAILD;
	}
	return (int)ret;
}
int main()
{
	char arr[20] = "-123456";
	int ret = my_atoi(arr);
	if (a == VAILD)
	{
		printf("合法转化:%d\n", ret);
	}
	else if(a == INVAILD)
	{
		printf("非法转化:%d\n", ret);
	}
	return 0;
}

完! 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值