【C语言】字符串+内存函数的介绍

news2025/1/11 13:58:31

🎈个人主页:.满船清梦压星河_-CSDN博客

🌂c/c++领域新星创作者

🎉欢迎👍点赞✍评论❤️收藏

😛😛😛希望我的文章能对你有所帮助,有不足的地方还请各位看官多多指教,大家一起学习交流!


0.前言

C 语言中对字符和字符串的处理很是频繁,但是 C 语言本身是没有字符串类型的,字符串通常放在 常量字符串中或者 字符数组 中。字符串常量适用于那些对它不做修改的字符串函数。

1.函数介绍

1.1 strlen
用来计算字符串长度
size_t  strlen ( const char * str );

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

字符串拷贝函数

char* strcpy ( char * destination , const char * source );
1. Copies the C string pointed by source into the array pointed by destination, including the
terminating null character (and stopping at that point).
2. 源字符串必须以 '\0' 结束。
3. 会将源字符串中的 '\0' 拷贝到目标空间。
4. 目标空间必须足够大,以确保能存放源字符串。
5. 目标空间必须可变。
1.3 strcat 
在字符串末尾追加字符串
char * strcat ( char * destination , const char * source );
1. Appends a copy of the source string to the destination string. The terminating null      character  in destination is overwritten by the first character of source, and a null-character   is included at the end of the new string formed by the concatenation of both in destination.
2. 源字符串必须以 '\0' 结束。
3. 目标空间必须有足够的大,能容纳下源字符串的内容。
4. 目标空间必须可修改。
1.4 strcmp 
比较字符串大小
int strcmp ( const char * str1 , const char * str2 );
标准规定:
                1.   第一个字符串大于第二个字符串,则返回大于 0 的数字
                2.   第一个字符串等于第二个字符串,则返回 0
                3.   第一个字符串小于第二个字符串,则返回小于 0 的数字
1.5 strncpy 
char * strncpy ( char * destination , const char * source , size_t num );
拷贝 num 个字符从源字符串到目标空间。
如果源字符串的长度小于 num ,则拷贝完源字符串之后,在目标的后边追加 0 ,直到 num 个。
1.6 strncat 
char * strncat ( char * destination , const char * source , size_t num );
1. destination要有足够的空间来容纳要拷贝的字符串
 
2. strncat会将dest字符串最后的’\0’覆盖掉,字符追加完成后,再追加’\0’
 
3. 如果num大于字符串src的长度,那么仅将src全部追加到dest的尾部;
4. 如果num小于字符串src的长度,该字符会将src的所有字符附加在dest末尾。无论哪种情        况,都会在新字符串的末尾添加空字符。
1.7 strncmp
int strncmp ( const char * str1 , const char * str2 , size_t num );
比较到出现另个字符不一样或者一个字符串结束或者num个字符全部比较完。返回值与strcmp类似。
1.8 strstr
char * strstr ( const char * str1 , const char * str2 );
返回str2在str1中第一次出现的位置,如果str1中不存在str1,则返回一个空指针。
1.9 strtok
char * strtok ( char * str , const char * sep );-----字符串分割函数
1. sep 参数是个字符串,定义了用作分隔符的字符集合
2. 第一个参数指定一个字符串,它包含了 0 个或者多个由 sep 字符串中一个或者多个分隔符分      割的标记。
3.  strtok 函数找到 str 中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注: strtok 函数会改变被操作的字符串,所以在使用 strtok 函数切分的字符串一般都是临时            拷贝的内容 并且可修改。)
4. strtok 函数的第一个参数不为 NULL ,函数将找到 str 中第一个标记, strtok 函数将保存它在      字符串 中的位置。
5. strtok 函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下        一个标记。
6. 如果字符串中不存在更多的标记,则返回 NULL 指针。

给大家看一个示例代码: 

#include <stdio.h>
int main()
{
	char* p = "houjindong@bjut.edu.cn";
	const char* sep = ".@";
	char arr[50];
	char* str = NULL;
	strcpy(arr, p);//将数据拷贝一份,处理arr数组的内容
	for (str = strtok(arr, sep); str != NULL; str = strtok(NULL, sep))
	{
		printf("%s\n", str);
	}
}

1.10 strerror
char * strerror ( int errnum );
返回错误码,所对应的错误信息。

示例代码: 

#define _CRT_SECURE_NO_WARNINGS 1
#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;
}

 

1.11 memcpy
void * memcpy ( void * destination , const void * source , size_t num );
1. 函数 memcpy source 的位置开始向后复制 num 个字节的数据到 destination 的内存位置。
 
2.  这个函数在遇到 '\0' 的时候并不会停下来。
3. 如果 source destination 有任何的重叠,复制的结果都是未定义的。
1.12 memmove
void * memmove ( void * destination , const void * source , size_t num );
1. 和 memcpy 的差别就是 memmove 函数处理的源内存块和目标内存块是可以重叠的。
2. 如果源空间和目标空间出现重叠,就得使用 memmove 函数处理。
1.13 memcmp  
int memcmp ( const void * ptr1 , const void * ptr2 , size_t num );
比较从 ptr1 ptr2 指针开始的 num 个字节。
返回值类型如下:

2.函数的模拟实现

2.1模拟实现strlen

求字符串的长度可以用三种方式来模拟实现:

①计数器方式

//计数器方式
int my_strlen(const char * str)
{
 int count = 0;
 while(*str)
 {
 count++;
 str++;
 }
 return count;
}

②递归方式

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

③指针运算的方式

//指针-指针的方式
int my_strlen(char *s)
{
       char *p = s;
       while(*p != ‘\0’ )
              p++;
       return p-s;
}
2.2模拟实现strcpy
char *my_strcpy(char *dest, const char*src)
{ 
  char *ret = dest;
  assert(dest != NULL);
  assert(src != NULL);
  while((*dest++ = *src++))
   {
      ;
   }
  return ret;
}
2.3模拟实现strcat
char *my_strcat(char *dest, const char*src)
{
 char *ret = dest;
 assert(dest != NULL);
 assert(src != NULL);
 while(*dest)
 {
 dest++;
 }
 while((*dest++ = *src++))
 {
 ;
 }
 return ret;
}
2.4模拟实现strstr
const char* my_strstr(const char* str1, const char* str2)
{
	const char* s1;//遍历str1指向的字符串
	const char* s2; //遍历str2指向的字符串
	const char* cp;//保存开始比较的位置
	assert(str1 && str2);
	if (*str2=='\0')
	{
		return str1;
	}
	cp=str1;
	while (*cp)
	{
		s1 = cp;
		s2 = str2;
		while (*s1 && *s2 && *s1 == *s2)
		{
			s1++;
			s2++;
		}
		if (*s2 == '\0')
		{
			return cp;
		}
		cp++;
	}
	return NULL;
}
2.5模拟实现strcmp
int my_strcmp (const char * src, const char * dst)
{
        int ret = 0 ;
 assert(src != NULL);
   assert(dest != NULL);
        while( ! (ret = *(unsigned char *)src - *(unsigned char *)dst) && *dst)
                ++src, ++dst;
        if ( ret < 0 )
                ret = -1 ;
        else if ( ret > 0 )
                ret = 1 ;
        return( ret );
}
2.6模拟实现memcpy
void* my_memcpy(void* dest, void* src, size_t sz)
{
	assert(dest && src);
	while (sz--)
	{
		*(char*)dest = *(char*)src;
		dest = (char*)dest + 1;
		src = (char*)src + 1;
	}
}
2.7模拟实现memmove
void* my_memmove(void* dest, void* src, size_t sz)
{
	assert(dest && src);
	void* start = dest;
	if (dest < src)//ǰ󿽱
	{
		while (sz--)
		{
			*(char*)dest = *(char*)src;
			dest = (char*)dest + 1;
			src = (char*)src + 1;
		}
	}
	else//Ӻǰ
	{
		while (sz--)
		{
			*((char*)dest + sz) = *((char*)src + sz);
		}
	}
	return start;
}

🛻🛻🛻 文章到此,我也就基本上给大家介绍完了。我们下篇文章再见!

                希望路过的各位点点关注点点赞👍

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1128255.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【概率论教程01】对贝叶斯定理的追忆

一、说明 贝叶斯定理&#xff0c;是一个需要反复体悟的道理&#xff0c;不是说公式解释清除就算Grasp&#xff0c;而是需要反复在实际项目中发挥&#xff0c;才能算掌握了。而实际应用中&#xff0c;并不是简单给出条件就可以套用&#xff0c;而是隐藏在迷雾一样的事实中&#…

『C语言进阶』自定义类型详解

&#x1f525;博客主页&#xff1a; 小羊失眠啦. &#x1f516;系列专栏&#xff1a; C语言、Linux、Cpolar ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 文章目录 前言一、结构体1.1 结构体的基础知识1.2 结构体的声明1.3 结构体的特殊声明1.4 结构体的自引用1.5 结构体的初…

Unity开发过程中的一些小知识点

1、如何查询挂载了指定脚本的游戏物体 可以直接在Hierarchy面板上&#xff0c;搜索想要找的脚本名 2、如何将Unity生成的多个相同游戏物体获得序号 可以使用Unity的API Transform.GetSiblingIndex() 实现。 Transform.GetSiblingIndex()gameobject.idTransform.GetSiblingI…

PDF Reader Pro v2.9.8(pdf编辑阅读器)

PDF Reader Pro是一款PDF阅读和编辑软件&#xff0c;具有以下特点&#xff1a; 界面设计简洁&#xff0c;易于上手。软件界面直观清晰&#xff0c;用户可以轻松浏览文档&#xff0c;编辑注释和填写表单。功能强大&#xff0c;提供了多种PDF处理工具&#xff0c;包括阅读、注释…

高压放大器在扫描显微镜中的应用及优势是什么

随着科技的不断进步&#xff0c;扫描显微镜成为了现代显微镜技术的重要组成部分。它能够提供更高的分辨率、更广的视场和更强的功能&#xff0c;用于研究各种微观结构和材料的特性。而高压放大器在扫描显微镜系统中发挥着至关重要的作用。下面安泰电子Aigtek将详细介绍高压放大…

数据分享 I 第一至第四批专精特新“小巨人”企业数据

数据地址&#xff1a; 专精特新“小巨人”企业数据https://www.xcitybox.com/datamarketview/#/Productpage?id355 基本信息. 数据名称: 第一至第四批专精特新“小巨人”企业数据 数据格式: Shpxlsx 数据时间: 2022年 数据几何类型: 点 数据坐标系: WGS84坐标系 数据字…

c语言进阶部分详解(详细解析自定义类型——枚举,联合(共用体))

上篇文章介绍了结构体相关的内容&#xff0c;大家可以点击链接进行浏览&#xff1a;c语言进阶部分详解&#xff08;详细解析自定义类型——结构体&#xff0c;内存对齐&#xff0c;位段&#xff09;-CSDN博客 各种源码大家可以去我的gitee主页进行查找&#xff1a;唔姆 (Nerow…

Centos使用war文件部署jenkins

部署jenkins所需要的jdk环境如下&#xff1a; 这里下载官网最新的版本&#xff1a; 选择jenkins2.414.3版本&#xff0c;所以jdk环境最低得是java11 安装java11环境 这里直接安装open-jdk yum -y install java-11-openjdk.x86_64 java-11-openjdk-devel.x86_64下载jenkins最新…

python实验12_中文文本分析

实验12&#xff1a;中文文本分析 1.实验目标及要求 &#xff08;1&#xff09;掌握jieba库的常用方法。 &#xff08;2&#xff09;掌握使用WordCloud库绘制词云的方法。 &#xff08;3&#xff09;掌握使用networkx库绘制关系图的方法。 2. 实验主要内容 (1)分析文本“水浒…

ConcurrentHashMap底层具体实现知道吗?实现原理是什么

从这三个方面来回答&#xff1a; ConcurrentHashMap 的整体架构 ConcurrentHashMap 的基本功能 ConcurrentHashMap 在性能方面的优化 ConcurrentHashMap 的整体架构 这个是 ConcurrentHashMap 在 JDK1.8 中的存储结构&#xff0c;它是由数组、单向链表、红黑树组成. 当我们初始…

谈谈你对Spring的理解

谈谈你对Spring的理解 一&#xff0c;什么是Spring 1.介绍 Spring是一个用于开发Java应用程序的工具集合&#xff0c;它提供了许多方便的组件和工具&#xff0c;可以帮助开发者更轻松地构建企业级应用程序。 Spring Framework是Spring的核心部分&#xff0c;它可以帮助开发者…

windows的adb环境安装

需要配置androidhome和一些path

基于情感分析的网络舆情热点分析系统 计算机竞赛

文章目录 0 前言1 课题背景2 数据处理3 文本情感分析3.1 情感分析-词库搭建3.2 文本情感分析实现3.3 建立情感倾向性分析模型 4 数据可视化工具4.1 django框架介绍4.2 ECharts 5 Django使用echarts进行可视化展示5.1 修改setting.py连接mysql数据库5.2 导入数据5.3 使用echarts…

2023年信息科学与工程学院学生科协第二次软件培训

2023年信息科学与工程学院学生科协第二次软件培训 文章目录 2023年信息科学与工程学院学生科协第二次软件培训一维数组数组的概念定义格式一维数组的访问例题&#xff1a;练习题&#xff1a; 数组元素数量一维数组的初始化 二维数组定义格式二维数组的访问二维数组的存储结构二…

实现多余内容变成省略号

实现效果 代码 <p class"item-content">{{ item.content }}</p>样式 .item-content {white-space: nowrap;/* 禁止换行 */overflow: hidden;/* 隐藏溢出部分 */text-overflow: ellipsis;/* 使用省略号表示溢出部分 */ }

21.4 Python 使用GeoIP2地图定位

GeoIP2是一种IP地址定位库&#xff0c;它允许开发人员根据IP地址查找有关位置和地理位置的信息。它使用MaxMind公司的IP地址数据库&#xff0c;并提供一个方便的Python API。GeoIP2可以用于许多不同的应用程序&#xff0c;例如网站分析、广告定位和身份验证。GeoIP2提供了许多不…

达梦管理工具报错“结果集不可更新,请确认查询列是否出自同一张表,并且包含值唯一的列。”

在使用达梦数据库管理工具时&#xff0c;我们测试过程中时常需要更新表数据&#xff0c;有时为了便捷&#xff0c;会直接使用管理工具修改表数据的值&#xff0c;但偶尔会遇到“结果集不可更新&#xff0c;请确认查询列是否出自同一张表&#xff0c;并且包含值唯一的列。”的报…

天津五家受欢迎的python培训机构 python凭什么这么火?

在目前的编程语言中&#xff0c;Python的抽象程度是最高的&#xff0c;是最接近自然语言的&#xff0c;非常容易上手&#xff0c;许多想学编程的小伙伴都从python入手&#xff0c;今天我们就来聊聊关于大家非常困惑的一些点&#xff0c;下面就一起来看看吧。 python火的原因 …

Java的反射(reflection)机制的简单使用

目录 一、定义 二、用途 三、反射基本信息 四、反射相关的类 五、反射示例 六、反射的优点和缺点 一、定义 Java的反射机制是运行时的状态&#xff0c;可以通过反射来调用类里面的属性和方法&#xff0c;私有的属性和方法也可以调用&#xff0c;也可以对它们进行修改。 二…

React 核心与实战2023版

课程亮点: 完整的前后台项目(PC+移动;完成业务;)React 最新企业标准技术栈(React 18 + Redux + ReactRouter + AntD)React + TypeScript (为大型项目奠定了基础)课程内容安排: React 介绍 React 是什么? React 是由Meta公司研发,是一个用于 构建Web和原生交互界面…