字符串函数和内存操作函数

前言

本篇文章主要介绍了一些字符串函数和内存操作函数,在C语言中对字符和字符串的处理还是很频繁的,但在C语言中是没有字符串类型的,字符串通常放在字符数组或常量字符串中。常量字符串里的内容是不可以被更改的,所以通常对不做修改的字符串放到常量字符串中。

字符串函数

1. strlen函数

1.1 strlen函数的基本用法

strlen函数是用来计算字符串长度的。

size_t strlen ( const char * str );

#include <stdio.h>
#include <string.h>`
int main()
{
	char arr1[20] = "abcdef";
	char arr2[] = { 'a','b','c','d','e','f'};
	char arr3[20] = "abc\0def";

	printf("%d\n", strlen(arr1));
	printf("%d\n", strlen(arr2));
	printf("%d\n", strlen(arr3));

	if (strlen(arr3) - strlen(arr1) > 0)
	{
		printf(">\n");
	}
	else
	{
		printf("<=\n");
	}
	return 0;
}

在这里插入图片描述
从上面的运行结果我们可以发现
1.arr1和arr3中的结果我们可以得知字符串以 ‘\0’ 作为结束标志,strlen函数返回的是在字符串中 ‘\0’ 前面出现的字符个数(不包含 ‘\0’ )。
2.arr2得到结果是个随机值,strlen函数会一直向后面查找’\0’,直到找到’\0’为止,所以参数指向的字符串必须要以 ‘\0’ 结束。
3.打印结果是’>‘是因为3-6=-3,应该打印’<=’,但strlen函数的返回值类型是size_t类型的,会把-3的补码当作原码来输出。

1.2 strlen函数的模拟实现

#include <stdio.h>
#include <assert.h>
size_t my_strlen(const char* str)
{
	assert(str);
	int count = 0;
	while (*str++)
	{
		count++;
	}
	return count;
}
int main()
{
	char arr[20] = "abcdef";
	printf("%u\n", my_strlen(arr));
	return 0;
}

在这里插入图片描述
用计数器的方式来模拟实现,从字符串的起始位置开始查找,没有遇到字符串结束标志’\0’计数器就加一次,直到遇到‘\0’为止。
assert用来判断参数不为NULL。

2. strcpy函数

2.1 strcpy函数的基本用法

strcpy是用来拷贝字符串的。

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

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20] = {0};
	char* p = "abcdef";
	char arr2[20] = "abcdef";
	char arr3[20] = "ghijklmn";

	printf("%s\n", strcpy(arr1, arr2));
	printf("%s\n", strcpy(arr3, arr2));
	printf("%s\n", strcpy(p, arr2));

	return 0;
}

在这里插入图片描述
在这里插入图片描述

第三种strcpy(p, arr2);这种使用方式是不对的,程序会挂掉,目标空间必须是可以改变的,不能是常量字符串。
strcpy(arr3, arr2)中源字符串必须以 ‘\0’ 结束,会将源字符串中的 ‘\0’ 拷贝到目标空间。
目标空间必须足够大,以确保能存放源字符串。

2.2 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[20] = "abcdef";
	char arr2[20];
	printf("%s\n", my_strcpy(arr2, arr1));
	return 0;
}

在这里插入图片描述
将源字符串拷贝到目标字符串中,直到遇到’\0’为止(‘\0’也会拷贝,所以源字符串中必须有’\0’),'\0’拷贝完就结束。

3. strcat函数

3.1 strcat函数的基本用法

strcat函数是用来字符串追加的。

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

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

在这里插入图片描述
源字符串必须以 ‘\0’ 结束。目标空间必须有足够大的空间,能容纳下源字符串的内容。目标空间必须可修改。

3.2 strcat函数的模拟实现

#include <stdio.h>
#include <assert.h>
char* my_strcat(char* dest, const char* src)
{
	assert(dest && src);
	char* ret = dest;
	//while (*dest++);//*dest找到'\0'后,又后置++指向了下一个位置也是'\0'
	while (*dest)
		dest++;
	while (*dest++ = *src++);
	return ret;
}
int main()
{
	char arr1[20] = "abcdef";
	char arr2[20] = "ghijkl";
	printf("%s\n", my_strcat(arr1, arr2));
	return 0;
}

在这里插入图片描述
实现的基本思想是先找到目标字符串的字符串结束标志’\0’,然后再将源字符串从第一个字符开始,将源字符串的第一个字符拷贝到目标字符串的’\0’位置开始拷贝,直到源字符串为’\0’拷贝到目标字符串上为止。后面的实现过程和strcpy的模拟实现基本一样。

strcat函数不能自己追加自己,如果自己追加自己的话,字符串末尾的’\0’会被自己的第一个字符覆盖掉,这样的话字符串中就找不到’\0’了,程序会死循环。

4. strcmp函数

4.1 strcmp函数的基本用法

strcmp函数是用来比较两个字符串大小的,比较的是两个字符串中对应字符的ASCII值。

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

标准规定:
第一个字符串大于第二个字符串,则返回大于0的数字
第一个字符串等于第二个字符串,则返回0
第一个字符串小于第二个字符串,则返回小于0的数字

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20] = "abcdef";
	char arr2[20] = "abcdef";
	char arr3[20] = "abcdef";
	char arr4[20] = "abcde";
	char arr5[20] = "abcde";
	char arr6[20] = "abcdef";

	printf("%d\n", strcmp(arr1, arr2));
	printf("%d\n", strcmp(arr3, arr4));
	printf("%d\n", strcmp(arr5, arr6));
	return 0;
}

在这里插入图片描述
比较字符串的大小时,不能直接用字符串的名字(例如arr1>arr2)来进行比较,字符串的名字是首元素的地址,比较的是地址,而不是字符串中字符的ASCII码值。

4.2 strcmp函数的模拟实现

#include <stdio.h>
#include <assert.h>
int my_strcmp(const char* str1, const char* str2)
{
	assert(str1 && str2);
	while (*str1 == *str2)
	{
		if (*str1 == '\0')
			return 0;
		str1++;
		str2++;
	}
	if (*str1 > *str2)
	{
		return 1;
	}
	else
	{
		return -1;
	}
}
int main()
{
	char arr1[20] = "abcdef";
	char arr2[20] = "abcdef";
	char arr3[20] = "abcdef";
	char arr4[20] = "abcde";
	char arr5[20] = "abcde";
	char arr6[20] = "abcdef";

	printf("%d\n", my_strcmp(arr1, arr2));
	printf("%d\n", my_strcmp(arr3, arr4));
	printf("%d\n", my_strcmp(arr5, arr6));
	return 0;
}

5. strncpy函数

5.1 strncpy函数的基本用法

strncpy函数也是用来字符串拷贝的,只不过在strcpy函数的基础上可以对字符长度进行了设置。

char * strncpy ( char * destination, const char * source, size_t num );

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20] = "abcdef";
	char arr2[20] = "bcdef";
	char arr3[20] = "abcef";
	char arr4[10] = "ab";
	printf("%s\n", strncpy(arr1, arr2, 4));
	printf("%s\n", strncpy(arr3, arr4, 4));
	return 0;
}

在这里插入图片描述
在这里插入图片描述
这里需要注意的是,如果需要拷贝的字符个数大于源字符串的字符个数的话,多出的字符会用’\0’代替。

6. strncat函数

6.1 strncat函数的基本用法

strncat函数也是用来追加字符串的,只不过在strcat函数的基础上可以对字符长度进行了设置。

char * strncat ( char * destination, const char * source, size_t num );

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20] = "abcd";
	char arr2[10] = "efgh";
	printf("%s\n", strncat(arr1, arr2, 2));
	return 0;
}

在这里插入图片描述

7. strncmp函数

7.1 strncmp函数的基本用法

strncmp函数是比较字符串的,在strcmp函数的基础上可以设置字符比较的个数。

int strncmp ( const char * str1, const char * str2, size_t num );

比较到出现另个字符不一样或者一个字符串结束或者num个字符全部比较完。

标准规定:
第一个字符串大于第二个字符串,则返回大于0的数字
第一个字符串等于第二个字符串,则返回0
第一个字符串小于第二个字符串,则返回小于0的数字

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[10] = "abcdef";
	char arr2[10] = "abcdeg";
	char arr3[10] = "abcdee";

	printf("%d\n", strncmp(arr1, arr2, 5));
	printf("%d\n", strncmp(arr1, arr2, 6));
	printf("%d\n", strncmp(arr1, arr3, 6));
	return 0;
}

在这里插入图片描述

8.strstr函数

8.1 strstr函数的基本用法

strstr函数是用来判断一个字符串是否为另一个字符串的子串的,如果是被查找字符串的子串,就返回这个子串在被查找字符串的第一次位置的地址,不是子串的话就返回空指针。

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

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20] = "abcdefdefghi";
	char arr2[10] = "def";
	char* p = strstr(arr1, arr2);
	if (p == NULL)
	{
		printf("找不到子串\n");
	}
	else
	{
		printf("%s\n", p);
	}
	return 0;
}

在这里插入图片描述

8.2 strstr函数的模拟实现

#include <stdio.h>
#include <assert.h>
char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 && str2);
	const char* s1 = str1;//用s1和s2来进行比较
	const char* s2 = str2;

	const char* cur = str1;//记录被查找字符串当前比较的位置
	while (*cur)
	{
		s1 = cur;
		s2 = str2;
		while (*s2 && *s1 == *s2)
		{
			s1++;
			s2++;
		}
		if (*s2 == '\0')//*s2='\0'说明子串已经全部被查找完
		{
			return (char*)cur;
		}
		cur++;//如果*s2不等于'\0',说明子串没有全部被匹配上,cur++从下一个位置继续查找
	}
	return NULL;
}
int main()
{
	char arr1[20] = "aabcabcd";
	char arr2[10] = "abc";
	char* p = my_strstr(arr1, arr2);
	if (p == NULL)
	{
		printf("找不到子串\n");
	}
	else
	{
		printf("%s\n", p);
	}
	return 0;
}

在这里插入图片描述

9. strtok函数

9.1 strtok函数的基本用法

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 arr1[20] = "qiqi123456@163.com";
	char sep[10] = "@.";

	printf("%s\n", strtok(arr1, sep));
	printf("%s\n", strtok(NULL, sep));
	printf("%s\n", strtok(NULL, sep));
	printf("%s\n", strtok(NULL, sep));

	//for (char* ret = strtok(arr1, sep); ret != NULL; ret = strtok(NULL, sep))//简便打印
	//{
	//	printf("%s\n", ret);
	//}
	
	return 0;
}

在这里插入图片描述

10. strerror函数

10.1 strerror函数的基本用法

strerror函数是查看错误信息的,返回错误码,所对应的错误信息,错误码是放在errno这个全局变量中的。

char * strerror ( int errnum );

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
	printf("%s\n", strerror(0));
	printf("%s\n", strerror(1));
	printf("%s\n", strerror(2));
	printf("%s\n", strerror(3));
	printf("%s\n", strerror(4));

	FILE* pf = fopen("text.txt", "r");
	printf("%s\n", strerror(errno));
	return 0;
}

在这里插入图片描述
前5行是对应的错误码信息,最后一行是遇到错误后errno的错误信息。

以上的函数只能对字符串进行操作,下面的内存操作函数可以对任意类型的数据进行操作,是以字节为单位进行操作的。

内存操作函数

1. memcpy函数

1.1 memcpy函数的基本用法

memcpy函数是以字节为单位来进行内存拷贝的。

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

函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。
这个函数在遇到 ‘\0’ 的时候并不会停下来。
如果source和destination有任何的重叠,复制的结果都是未定义的。

#include <stdio.h>
#include <string.h>
int main()
{
	int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[10] = { 0 };

	float arr3[10] = { 1.1,1.2,1.3,1.4,1.5,1.6,1.7,1.8 };
	float arr4[10] = { 0 };

	memcpy(arr2, arr1, 20);
	memcpy(arr4, arr3, 20);
	return 0;
}

在这里插入图片描述
在这里插入图片描述

1.2 memcopy函数的模拟实现

#include <stdio.h>
#include <assert.h>
void* my_memcpy(void* dest,const void* src, size_t num)
{
	assert(dest && src);
	void* ret = dest;//记录要返回的起始位置
	while (num--)
	{
		*(char*)dest = *(char*)src;
		dest = (char*)dest + 1;//不能dest++,(char*)dest只是暂时强制转换成char*,dest的类型还是void*
		src = (char*)src + 1;
	}
	return ret;
}
int main()
{
	int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[10] = { 0 };

	float arr3[10] = { 1.1,1.2,1.3,1.4,1.5,1.6,1.7,1.8 };
	float arr4[10] = { 0 };

	my_memcpy(arr2, arr1, 20);
	my_memcpy(arr4, arr3, 20);
	return 0;
}

2. memmove函数

2.1 memmove函数的基本用法

和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
如果源空间和目标空间出现重叠,就得使用memmove函数处理。

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

#include <string.h>
int main()
{
	int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };//3 4 5 6 7 6 7 8 9 10
	int arr2[10] = { 1,2,3,4,5,6,7,8,9,10 };//1 2 1 2 3 4 5 8 9 10

	memmove(arr1, arr1 + 2, 20);
	memmove(arr2 + 2, arr2, 20);

	return 0;
}

在这里插入图片描述
在这里插入图片描述

2.2 memmove函数的模拟实现

#include <assert.h>
void* my_memmove(void* dest, void* src, size_t num)
{
	assert(dest && src);
	void* ret = dest;
	if (dest < src)
	{
		while (num--)
		{
			*(char*)dest = *(char*)src;
			dest = (char*)dest + 1;
			src = (char*)src + 1;
		}
		return dest;
	}
	else
	{
		while (num--)
		{
			*((char*)dest + num) = *((char*)src + num);
		}
	}
	return dest;
}
int main()
{
	int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };//3 4 5 6 7 6 7 8 9 10
	int arr2[10] = { 1,2,3,4,5,6,7,8,9,10 };//1 2 1 2 3 4 5 8 9 10

	my_memmove(arr1, arr1 + 2, 20);//dest<src 从前向后拷贝数据不会被覆盖
	my_memmove(arr2 + 2, arr2, 20);//src<dest 从后向前拷贝数据不会被覆盖

	return 0;
}

3. memcpy函数

3.1 memcpy函数的基本用法

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

比较从ptr1和ptr2指针开始的num个字节。
相等的话,返回0。ptr1>ptr2,返回大于0的数字。ptr1<ptr2,返回小于0的数字。

#include <stdio.h>
#include <string.h>
int main()
{
	int arr1[10] = { 1,2,3,4,5,6,7 };
	int arr2[10] = { 1,2,3,4,5,7 };

	printf("%d\n", memcmp(arr1, arr2, 20));
	printf("%d\n", memcmp(arr1, arr2, 21));
	return 0;
}

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

4. memset函数

4.1 memset函数的基本用法

void *memset( void *dest, int c, size_t count );

memset函数是内存设置函数,将num个字节的空间设置成c。

#include <string.h>
int main()
{
	int arr[10] = { 0 };
	memset(arr, 1, 40);
	return 0;
}

在这里插入图片描述
在这里插入图片描述

总结

常用的字符串函数和内存操作函数就先介绍到这里了,希望对大家能有所帮助。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
GeoPandas是一个开源的Python库,旨在简化地理空间数据的处理和分析。它结合了Pandas和Shapely的能力,为Python用户提供了一个强大而灵活的工具来处理地理空间数据。以下是关于GeoPandas的详细介绍: 一、GeoPandas的基本概念 1. 定义 GeoPandas是建立在Pandas和Shapely之上的一个Python库,用于处理和分析地理空间数据。 它扩展了Pandas的DataFrame和Series数据结构,允许在其中存储和操作地理空间几何图形。 2. 核心数据结构 GeoDataFrame:GeoPandas的核心数据结构,是Pandas DataFrame的扩展。它包含一个或多个列,其中至少一列是几何列(geometry column),用于存储地理空间几何图形(如点、线、多边形等)。 GeoSeries:GeoPandas中的另一个重要数据结构,类似于Pandas的Series,但用于存储几何图形序列。 二、GeoPandas的功能特性 1. 读取和写入多种地理空间数据格式 GeoPandas支持读取和写入多种常见的地理空间数据格式,包括Shapefile、GeoJSON、PostGIS、KML等。这使得用户可以轻松地从各种数据源中加载地理空间数据,并将处理后的数据保存为所需的格式。 2. 地理空间几何图形的创建、编辑和分析 GeoPandas允许用户创建、编辑和分析地理空间几何图形,包括点、线、多边形等。它提供了丰富的空间操作函数,如缓冲区分析、交集、并集、差集等,使得用户可以方便地进行地理空间数据分析。 3. 数据可视化 GeoPandas内置了数据可视化功能,可以绘制地理空间数据的地图。用户可以使用matplotlib等库来进一步定制地图的样式和布局。 4. 空间连接和空间索引 GeoPandas支持空间连接操作,可以将两个GeoDataFrame按照空间关系(如相交、包含等)进行连接。此外,它还支持空间索引,可以提高地理空间数据查询的效率。
SQLAlchemy 是一个 SQL 工具包和对象关系映射(ORM)库,用于 Python 编程语言。它提供了一个高级的 SQL 工具和对象关系映射工具,允许开发者以 Python 类和对象的形式操作数据库,而无需编写大量的 SQL 语句。SQLAlchemy 建立在 DBAPI 之上,支持多种数据库后端,如 SQLite, MySQL, PostgreSQL 等。 SQLAlchemy 的核心功能: 对象关系映射(ORM): SQLAlchemy 允许开发者使用 Python 类来表示数据库表,使用类的实例表示表中的行。 开发者可以定义类之间的关系(如一对多、多对多),SQLAlchemy 会自动处理这些关系在数据库中的映射。 通过 ORM,开发者可以像操作 Python 对象一样操作数据库,这大大简化了数据库操作的复杂性。 表达式语言: SQLAlchemy 提供了一个丰富的 SQL 表达式语言,允许开发者以 Python 表达式的方式编写复杂的 SQL 查询。 表达式语言提供了对 SQL 语句的灵活控制,同时保持了代码的可读性和可维护性。 数据库引擎和连接池: SQLAlchemy 支持多种数据库后端,并且为每种后端提供了对应的数据库引擎。 它还提供了连接池管理功能,以优化数据库连接的创建、使用和释放。 会话管理: SQLAlchemy 使用会话(Session)来管理对象的持久化状态。 会话提供了一个工作单元(unit of work)和身份映射(identity map)的概念,使得对象的状态管理和查询更加高效。 事件系统: SQLAlchemy 提供了一个事件系统,允许开发者在 ORM 的各个生命周期阶段插入自定义的钩子函数。 这使得开发者可以在对象加载、修改、删除等操作时执行额外的逻辑。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值