c语言重要的字符串与内存函数

 更新时间:2021年09月22日 15:01:00   作者:qq_59078816  
这篇文章主要介绍一些c语言中常用字符串函数和内存函数的使用和注意事项,并且为了帮助读者理解和使用,也都模拟实现了他们的代码,需要的朋友可以参考一下

一.字符串函数

1. 求字符串长度的strlen

size_t strlen ( const char * str );

  • 字符串以 ‘\0' 作为结束标志,strlen函数返回的是在字符串中 ‘\0' 前面出现的字符个数(不包含 ‘\0' )。
  • 参数指向的字符串必须要以 ‘\0' 结束。
  • 注意函数的返回值为size_t,是无符号的。

模拟实现strlen

size_t my_strlen(const char*str)
{ 
  size_t count=0;
  while(*str)
  {
   str++;
   count++;
  }

2.比较字符串大小的strcmp

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

  • 从str1和str2指向的位置开始比较,如果遇到两个不相等的字符或者\0函数结束并且返回值。
  1. 第一个字符串的字符大于第二个字符串的字符返回 >0的数字。
  2. 第一个字符串的字符等于第二个字符串的字符返回 =0的数字。
  3. 第一个字符串的字符小于第二个字符串的字符返回 <0的数字。

模拟实现strcmp

int my_strcmp(const char*str1,const char*str2)
{
   while (*str1 == *str2)
   {
    if (*str1 && *str2)//判断'\0'
     return *s2 - *s1;
    *str1++;
    *str2++;
   }//循环结束标志*str1!=*str2
   return *str2 - *str1;
}

3.复制字符串的strcpy

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

拷贝sourcedestination中,返回一个指向dest的char* 的指针。

  • 源字符串必须以 ‘\0' 结束。
  • 会将源字符串中的 ‘\0' 拷贝到目标空间。
  • 目标空间必须足够大,以确保能存放源字符串。
  • 目标空间必须可变。

模拟实现strcpy

char *my_strcpy(char *dest, const char*src)
{ 
 char *ret = dest;
 while((*dest++ = *src++))
 {
  ;
 }
 return ret;
}

4.追加字符串的strcat

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

从dest的末尾 ‘\0'开始添加src直到‘\0'

  • 源字符串必须以 ‘\0' 结束。
  • 目标空间必须有足够的大,能容纳下源字符串的内容。
  • 目标空间必须可修改。

模拟实现strcat

char* my_strcat(const char* str1, const char* str2)
{
 
  while (*str1) str1++;
  while (*(char*)str1++ = *(char*)str2++) {
   ;
  }
}

5.查找字符串函数的strstr

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

在str2中查找str1的字符串,如果找到了返回str2中这个字符串的首地址。如果找不到返回NULL。.0

char* my_strstr(const char* str1, const char* str2)
{
 if (!*str2)//判断字符串是否为空
  return (char*)str1;

  char* ret1= (char*)str1;//将str类型转换
  
  char *cp = ret1;
 while (*ret1)
 {
  cp = ret1;
  char* ret2 =(char*) str2;
  while (*cp== *str2) {//遍历字符串
           
     if (!*ret2)//判断str2是否到达‘0'的位置
      return cp;
     cp++, ret2++;
  }
  ret1++;//循环结束没有返回,从下一个字符开始查找
  if (!*ret1)
   return NULL;
 }
 return NULL;
}

二、内存函数

1.复制 memcpy,memmove

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

dest复制src中num字节的数据。

模拟实现memcpy

void * my_memcpy ( void * dst, const void * src, size_t count)
{
  void * ret = dst;
  while (count--) {
    *(char *)dst = *(char *)src;
    dst = (char *)dst + 1;
    src = (char *)src + 1;
  }
 
  return(ret);
}

·如果dst 和src指向同一个数组会发生什么?

int main()
{
 int arr1[10] = { 1,2,3,4,5,6,7,8,9,0 };
 // 预期结果       1 2 1 2 3 4 7 8 9 0
 my_memcpy(arr1+2, arr1, 16);

  for (int i = 0; i < 10; i++)
  {
   printf("%d ", arr1[i]);
 }
}

实际为 1 2 1 2 1 2 7 8 9 0
因为到5 6 的时候3 4被改成了1 2 ,5 6也就被改成1 2。
也就是说被复制的元素在复制前被改变了,导致复制结果失败。
如果是这样指向同一个数组呢?

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

    for (int i = 0; i < 10; i++)
    {
     printf("%d ", arr1[i]);
   }
}

复制结果没有问题。

对于这种情况,c语言有一个更强大的函数memmove.

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

  • 与memcpy的功能一样,但是memmove可以指向同一块空间。

模拟实现memmove

**void* my_memmove(void* dest, void* src, size_t num)
{
 char* ret = dest;
 //如果指向同一块空间 判断地址大小,避免数据在被复制前被改变
 if ( (char*)dest-(char*)src< 0){
  while (num)
  {

   *((char*)dest)++ = *((char*)src)++;
   num--;
  }
 }
 else {
  while(num--){
   *((char*)dest+num) = *((char*)src+num);
  }
 }
 return ret;
}

2.比较 memcmp

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

从ptr1和ptr2的位置开始比较num个字节,当两个字节数据不同时就会返回。

  • ptr1>ptr2 返回值>0;
  • ptr1=ptr2 返回值=0;
  • ptr1<ptr2 返回值<0;

到此这篇关于c语言重要的字符串与内存函数的文章就介绍到这了,更多相关c语言的字符串与内存函数内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++实现扫雷经典小游戏

    C++实现扫雷经典小游戏

    这篇文章主要为大家详细介绍了C++实现扫雷经典小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • 使用C++制作GC Server过程详解

    使用C++制作GC Server过程详解

    这篇文章主要介绍了使用C++制作GC Server过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-09-09
  • C++详细讲解对象的构造

    C++详细讲解对象的构造

    当在参数化构造函数中声明对象时,必须将初始值作为参数传递给构造函数。对象声明的常规方法可能不起作用。构造函数可以显式或隐式调用,让我们一起了解对象的构造
    2022-04-04
  • C++list的模拟实现

    C++list的模拟实现

    list是数据结构中的链表,在C++的STL中,有list的模板,STL中的list的结构是带头双向循环链表,当然STL中还有一个forward_list的链表,这个链表是一个带头的单链表。为了更好的理解list,我们来对其进行模拟实现。,需要的朋友可以参考
    2023-04-04
  • 二叉树前序遍历的非递归算法

    二叉树前序遍历的非递归算法

    这篇文章主要介绍了二叉树前序遍历的非递归算法,需要的朋友可以参考下
    2014-02-02
  • C++九种排序具体实现代码

    C++九种排序具体实现代码

    这篇文章主要介绍了C++九种排序具体实现代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-04-04
  • C语言中炫酷的文件操作实例详解

    C语言中炫酷的文件操作实例详解

    内存中的数据都是暂时的,当程序结束时,它们都将丢失,为了永久性的保存大量的数据,C语言提供了对文件的操作,这篇文章主要给大家介绍了关于C语言中文件操作的相关资料,需要的朋友可以参考下
    2021-10-10
  • C++中std::vector的6种初始化方式

    C++中std::vector的6种初始化方式

    这篇文章主要介绍了C++中std::vector的6种初始化方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • Opencv处理图像之轮廓提取

    Opencv处理图像之轮廓提取

    这篇文章主要为大家详细介绍了Opencv处理图像之轮廓提取,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-01-01
  • C++和java设计模式之单例模式

    C++和java设计模式之单例模式

    这篇文章主要为大家详细介绍了C++和java设计模式之单例模式的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12

最新评论