【C语言航路】第十一站:字符串、字符和内存函数(上)

简介: 【C语言航路】第十一站:字符串、字符和内存函数

一、字符串函数

1.strlen

(1)strlen的库函数文档

如下图所示,是strlen在库函数中的文档。

它的参数是const char*类型的,这是考虑到它不会被改变的原因

返回类型是size_t类型的,这是考虑到它计算的是长度不会是负数的原因

这个函数返回的是一个字符串的长度,传入一个地址,计算的是\0字符之前的长度

(2)strlen的模拟实现

在这里我们采用三种方式来实现:循环计数、递归、指针减指针

#include<stdio.h>
#include<assert.h>
//循环计数
int my_strlen1(const char* str)
{
  assert(str);
  int count = 0;
  while (*str != '\0')
  {
    count++;
    str++;
  }
  return count;
}
//递归
int my_strlen2(const char* str)
{
  assert(str);
  if (*str == NULL)
  {
    return 0;
  }
  return 1 + my_strlen2(str + 1);
}
//指针减指针
int my_strlen3(const char* str)
{
  assert(str);
  const char* start = str;
  while (*str != '\0')
  {
    str++;
  }
  return str - start;
}
int main()
{
  char arr[] = "abcdef";
  int len = my_strlen3(arr);
  printf("%d\n", len);
  return 0;
}

(3)strlen的注意事项

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

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

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

关于第三点注意事项,我们在这里着重说一下

#include<stdio.h>
#include<string.h>
int main()
{
  if (strlen("abc") - strlen("abcdef") > 0)
  {
    printf(">\n");
  }
  else
  {
    printf("<=\n");
  }
  return 0;
}

这段代码的运行结果是

肯定与我们的设想是不一样的,因为strlen("abc")计算出来的是3,strlen("abcdef")计算出来的是6,虽然看上去好像3-6是-3,应该是<=,但是要主要这里的都是无符号的,计算出来的结果也是无符号的。所以这个数其实在内存中来看是一个很大的数。还是大于0的,所以是>

2.strcpy

(1)strcpy的库函数文档

它的参数是char* destination 和const char* source,destination的意思是目标空间的地址,

source的意思是源头空间的地址,这个地址里面的值是不可以被修改的。

功能是将source处的字符串拷贝到destination处

返回类型是char*,意思是将destination处的地址返回

(2)strcpy的使用以及注意事项

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

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

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

4.目标空间必须可变。

在下面这段代码中

#include<stdio.h>
#include<string.h>
int main()
{
  char arr1[] = "xxxxxxxxxxxxxxxxxx";
  char arr2[] = "hello world";
  strcpy(arr1, arr2);
  printf("%s\n", arr1);
  return 0;
}

我们直接从调试里面查看数组的值

在下面这段代码中

#include<stdio.h>
#include<string.h>
int main()
{
  char arr1[] = "xxxxxxxxxxxxxxxxxx";
  char arr2[] = "hello world";
  strcpy(arr2, arr1);
  printf("%s\n", arr1);
  return 0;
}

出现了错误,这是因为,arr2的空间不是足够大。

在下面这段代码中

#include<stdio.h>
#include<string.h>
int main()
{
  char* p = "xxxxxxxxxxxxxxxxxx";
  char arr2[] = "hello world";
  strcpy(p, arr2);
  printf("%s\n", p);
  return 0;
}

程序直接崩溃,这是因为char* p是不可以被修改的

(3)strcpy的模拟实现

#include<stdio.h>
#include<assert.h>
char* my_strcpy(char* dest, const char* src)
{
  assert(dest && src);
  char* ret = dest;
  while (*dest++ = *src++)
  {
    ;
  }
  return ret;
}
int main()
{
  char arr1[] = "xxxxxxxxxxxxx";
  char arr2[] = "hello world";
  char* p = my_strcpy(arr1, arr2);
  printf("%s\n", p);
  return 0;
}

运行结果为

3.strcat

(1)strcat的库函数文档

1.它的参数是char* destination和const char* source,与strcpy是一样的。

2.它的功能是将source处的字符串追加到destination后面

3.它的返回类型是char*,返回destination的地址

(2)strcat的使用以及注意事项

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

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

3.目标空间必须可修改。

4.字符串自己给自己追加,会陷入死循环

#include<stdio.h>
#include<string.h>
int main()
{
  char arr[20] = "hello ";
  char* p = strcat(arr, "world");
  printf("%s\n", arr);
  return 0;
}

运行结果为

(3)strcat的模拟实现

思路就两步:

1.找到\0字符的位置

2.追加

#include<stdio.h>
#include<assert.h>
char* my_strcat(char* dest, const char* src)
{
  assert(dest && src);
  char* ret = dest;
  //1.找到'\0'字符
  while (*dest != '\0')
  {
    dest++;
  }
  //2.追加
  while (*dest++ = *src++)
  {
    ;
  }
  return ret;
}
int main()
{
  char arr[20] = "hello ";
  char* p = my_strcat(arr, "world");
  printf("%s\n", arr);
  return 0;
}

(4)strcat自己对自己追加

这个其实是不可以的,会造成死循环的现象

假如说下面的字符串自己对自己追加的话

\0会被修改为a,后面的都会覆盖过去,这样我们就永远也找不到\0了,最终导致分配的空间用完, 使用了未分配的空间而崩溃

下图是我们模拟出来的strcat的自己对自己追加后的效果

如果是官方的库里面的话,似乎没有出现这个问题,这个是visual studio对这个问题进行了优化,在一些其他的ide上可能就会出现这个死循环问题,导致程序出现bug

相关文章
|
1月前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
70 10
|
1月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
53 9
|
1月前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
53 6
|
1月前
|
存储 C语言 开发者
【C语言】格式化输出占位符及其标志字符详解(基于ISO/IEC 9899:2024)
在C语言中,格式化输出通过 `printf` 函数等格式化输出函数来实现。格式说明符(占位符)定义了数据的输出方式,标准ISO/IEC 9899:2024(C23)对这些格式说明符进行了详细规定。本文将详细讲解格式说明符的组成部分,包括标志字符、宽度、精度、长度修饰符和类型字符,并适当增加表格说明。
44 6
|
1月前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
298 6
|
1月前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
67 6
|
2月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
402 1
|
1月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
2月前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
2月前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
27 3