描述:C语言中对于字符串类型并未定义,但存在许多有关字符串的库函数。
例如:strstr(在原串中匹配子串)strcpy(将原串内容拷贝到目标串中),strncpy(原串内容拷贝到目标串---可控拷贝数), strcmp(比较两个字符串的内容),strcat(字符串的追加)等等,我们可以自己编写程序实现以上功能的函数,而不调用库函数。可以对照库函数进行学习,提高自身的编程能力。
1、模拟strcpy函数(复制字符串)
#include<stdio.h>
#include<assert.h>
#include<string.h>
#include<stdlib.h>
char * my_strcpy(char *dest, const char *src)
{
assert(dest);//判断指针有效性
assert(src);
char *ret = dest;
while (*dest++ = *src++)
;
return ret;
}
int main()
{
char arr[] = "hello world!";
char src[20];
printf("%s\n",my_strcpy(src, arr));
system("pause");
return 0;
}
2、模拟实现strncpy函数(字符串复制-----可控复制字节数量)
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<string.h>
char *my_strncpy(char *dest, const char *src, size_t count)
{
assert(dest);
assert(src);
char *ret = dest;
while (count--)//count为要复制的字节数
{
*dest++ = *src++;
}
*dest = '\0';//字符串最后要加上'\0'
return ret;
}
int main()
{
char arr[] = "abcdef";
char str[10];
my_strncpy(str, arr, 4);
printf("%s\n", str);
system("pause");
return 0;
}
3、模拟strcmp函数(比较两个字符/字符串大小)
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
int my_strcmp(const char *dest, const char *src)
{
assert(dest);
assert(dest);
int ret = 0;
while( (*(unsigned char*)dest == *(unsigned char*)src) && *dest)//注意此处需要强制类型转换,转换成无符号的char型,否则会出错
{
dest++;
src++;
}
if (*(unsigned char*)dest == '\0')
ret = 0;//*dest为0时,表示比较完了,两个字符串相等,返回0
else
ret = *(unsigned char*)dest - *(unsigned char*)src;//两个字符串不等返回两个字符串的差(强制转换类型)
return ret;
}
int main()
{
char arr[] = "abcde";
char src[] = "adcde";
int var;
var = my_strcmp(arr, src);
printf("%d\n", var);
system("pause");
return 0;
}
4、模拟实现strncmp函数(字符串比较------可控比较字节数量)
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<string.h>
int my_strncmp(const char *str1, const char *str2,size_t count)
{
assert(str1);
assert(str2);
int ret;
while (count--)
{
if (*(unsigned char*)str1 == *(unsigned char*)str2)//注意此处需要强制类型转换,转换成无符号的char型
str1++, str2++;
}
if (count == 0)//count=0表示比较完成,两个字符串相等
ret = 0;
else
ret = *(unsigned char*)str1 - *(unsigned char*)str2;//注意此处需要强制类型转换
return ret;
}
int main()
{
char arr1[] = "cdabctr";
char arr2[] = "cdabfe";
int len = sizeof(arr1) / sizeof(arr1[0]);
printf("%d\n", my_strncmp(arr1, arr2, len));
system("pause");
return 0;
}
5、模拟strcat函数(追加/合并字符串)
#include<stdio.h>
#include<assert.h>
#include<string.h>
#include<stdlib.h>
char * my_strcat(char *dest, const char *src)
{
assert(dest);
assert(src);
char *ret = dest;
while (*ret)//数组ret自加,直到'\0'停止
ret++;
while (*ret++ = *src++)//src第一个元素替换'\0',其余元素追加到ret中
;
return (dest);
}
int main()
{
char arr[30] = "hello";
char src[] = " to world!";
printf("%s\n",my_strcat(arr, src));
system("pause");
return 0;
}
6、模拟实现strncat函数(字符串追加-------可控追加字节数量)
#define _CRT_SECURE_NO_WARNINGS//使用scanf时需要加在头文件(VS2013中)
#include<stdio.h>
#include<assert.h>
#include<string.h>
#include<stdlib.h>
char *my_strncat(char *dest,char *src,int count)
{
assert(dest);
assert(src);
char *ret = dest;
while (*dest) //数组dest自加,直到'\0'停止
dest++;
while (count--) //src第一个元素替换'\0',其余元素追加到dest中
*dest++ = *src++;
*dest = '\0'; //字符串后加上'\0'
return ret;
}
int main()
{
char arr[30] = "hello"; //注意此处定义数组大小要充分大,可以存放两个数组
char str[] = " to world!";
int len;
printf("请输入追加长度>");
scanf("%d", &len);
my_strncat(arr,str,len);
printf("%s\n", arr);
system("pause");
return 0;
}
7、模拟实现strstr函数(判断一个字符串是不是另一个字符串的子字符串)
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<assert.h>
char *my_strstr(const char *str1,const char *str2)
{
assert(str1);
assert(str2);
char *p = (char *)str1;
if (*str2=='\0')//如果str2为空,返回str1
return((char *)str1);
while (*p)
{
char *p1 = p;
char *p2 = (char *)str2;
while (*p2 && *p1 && (*p1 == *p2))//注意p1,p2为'\0'时也要停止循环
{
p1++;
p2++;
}
if (*p2 == '\0')//成功匹配p2中每一个字符,返回p(子字符串开始处)
return p;
p++;//p1和p2匹配没成功,p1指向下一个字符,重新与p2进行匹配
}
return NULL;
}
int main()
{
char arr[20] = "abccdefgh";
char str[20] = "cdefg";
char *ret=my_strstr(arr, str);
printf("%s\n", ret);
system("pause");
return 0;
}
上述程序运行结果为:cdefgh
8、模拟实现memset函数(内存赋值或数据重置-----可控赋值字节数量)
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
//memset作用是在一段内存块中填充某个给定的值,它对较大的结构体或数组进行清零操作的一种最快方法
void *my_memset(void *dest, int c, size_t count)
{
assert(dest);
char *p = (char *)dest;
while (count--)
{
*p++ = c;
}
return dest;
}
int main()
{
char arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };//memset是按字节赋值
int len = sizeof(arr) / sizeof(arr[0]);
int i;
my_memset(arr, 0, len);
for (i = 0; i < len; i++)
{
printf("%d ",arr[i]);
}
system("pause");
return 0;
}
9、模拟实现memcmp函数(内存中的内容比较大小-----可控比较字节数量)
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
int my_memcmp(const void *p1, const void *p2, size_t count)
{
assert(p1);
assert(p2);
int ret = 0;
unsigned char *dest = (unsigned char *)p1;
unsigned char *src = (unsigned char *)p2;
while (count--)
{
if (*(unsigned char *)dest == *(unsigned char *)src)
{
dest++;
src++;
}
}
if (count == 0)
ret = 0;
else
ret = *(unsigned char *)dest - *(unsigned char *)src;;
return ret;
}
int my_memcmp(const void *s1,const void *s2,size_t count)
{
int res = 0;
const unsigned char *p1 =(const unsigned char *)s1;//注意是unsigned char *
const unsigned char *p2 =(const unsigned char *)s2;
for(p1 ,p2;count > 0;p1++,p2++,count--)
if((res =*p1 - *p2) != 0) //不相当则结束比较
break;
return res;
}
int main()
{
char arr[10] = "abcdefg";//memset是按字节进行比较的
char str[10] = "abcdfd";
int len = sizeof(arr) / sizeof(arr[0]);
printf("%d\n", my_memcmp(arr, str, len));
system("pause");
return 0;
}
转载于:https://blog.51cto.com/luoyafei/1716895