《彻底理解C语言指针全攻略(6)-- qsort、sizeof和strlen》

《qsort、sizeof和strlen》

🔥小龙报:个人主页

❄️个人专栏:《C语言》《算法》KelpBar海带Linux智慧屏项目

永远相信美好的事情即将发生


文章目录


前言

继上篇文章后这篇文章将继续为大家分享指针的相关知识,本篇主要是qsort、sizeof和strlen三个知识点,让我们向着更优秀的自己出发go!


一、qsort

1.1引入

qsort函数是C语言中提供的一个排序函数(quick sort),是基于快速排序算法思想的一种排序算法
qsort函数的优点:

(1)现成的排序算法,学会了直接就可以使用不需要自己实现了。

(2)大部分的情况下效率都是比冒泡排序高的qsort函数可以排序任意类型的数据

qsort函数的原型及其参数解析:

c 复制代码
 #include<stdlib.h>;  //头文件
void qsort(void *base, size_t num, size_t size, int (*compar)(const void *, const void *));

参数说明:

**(1)void base* :指向要排序的数组首元素的指针。由于是 void * 类型,所以可以处理任意类型的数组。
(2)size_t num :数组中元素的个数。size_t 是一种无符号整数类型 ,通常用于表示大小或计数。
(4)size_t size:每个数组元素的大小(以字节为单位) 。这里有这个参数是因为,前面用的void*类型,只知道元素个数不知道每个元素多大是不行的。
(5)int (*compar)(const void *, const void *)一个指向比较函数的指针 。该比较函数用于比较数组中的两个元素,它接受两个 const void * 类型的参数,返回值为 int 类型: 如果进行升序排序,返回一个正整数。如果两个元素相等,返回0。如果进行降序排序,返回一个负整数。

补充说明一下:默认第一个参数比第二个大;

1.1. qsort使用举例

1.1.1使用qsort函数排序整型数据

c 复制代码
#include <stdio.h>

//qosrt函数的使⽤者得实现⼀个⽐较函数
int int_cmp(const void* p1,const void* p2)
{
	return (*((int*)p1) - *((int*)p2));
}

int main()
{
	int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };


	qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), int_cmp);

	for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
		printf("%d ", arr[i]);
	return 0;
}

运行结果:

1.1.2使用qsort排序结构数据

c 复制代码
#include <stdio.h>

struct Stu //学生
{
	char name[20];  //名字
	int age; //年龄
};


//假设按照年龄来⽐较
int cmp_stu_by_age(const void* e1, const void* e2)
{
	return ((struct Stu*)e1)->age - ((struct Stu*)e2)->age;
}

//strcmp - 是库函数,是专⻔⽤来⽐较两个字符串的⼤⼩的

//假设按照名字来⽐较
int cmp_stu_by_name(const void* e1, const void* e2)
{
	return strcmp(((struct Stu*)e1)->name, ((struct Stu*)e2)->name);
}

//按照名字来排序
void test1()
{
	struct Stu s[] = { {"zhangsan", 20}, {"lisi", 30}, {"wangwu", 15} ,{"zhuming",48}};
	int sz = sizeof(s) / sizeof(s[0]);
	qsort(s, sz, sizeof(s[0]), cmp_stu_by_name);
}

//按照年龄来排序
void test2()
{
	struct Stu s[] = { {"zhangsan", 20}, {"lisi", 30}, {"wangwu", 15} ,{"zhuming",48} };
	int sz = sizeof(s) / sizeof(s[0]);
	qsort(s, sz, sizeof(s[0]), cmp_stu_by_age);
}

int main()
{
	//test1();
	test2();
	return 0;
}

运行结果:

(1)按名字排序

(2)按年龄排序

1.2. qsort函数的模拟实现(冒泡排序)

c 复制代码
#include <stdio.h>

int int_cmp(const void* p1, const void* p2)
{
	return (*((int*)p1) - *((int*)p2));
}

void Swap(void* p1, void* p2, size_t size)
{
	//虽然不知道每个元素的大小具体是多少,但是可以一个一个字节的交换
	for (int i = 0; i < size; i++)
	{
		char temp = *((char*)p1 + i);
		*((char*)p1 + i) = *((char*)p2 + i);
		*((char*)p2 + i) = temp;
	}
}

void bubble(void* base, size_t count, size_t size, int(*cmp)(void*, void*))
{
	//虽然不知道具体元素类型,但是强制转换为char*类型是因为是它的大小是1,再+j*size就可以找到第j个元素了
	for (int i = 0; i < count - 1; i++)
	{
		for (int j = 0; j < count - i - 1; j++)
		{
			if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
				Swap((char*)base + j * size, (char*)base + (j + 1) * size, size);
		}
	}
}
int main()
{
	int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };
	int i = 0;
	bubble(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), int_cmp);
	for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");
}

运行结果:

回调函数的过程:

二、sizeof和strlen的对比

2.1 sizeof

在学习操作符的时候,我们学习了sizeof ,sizeof 计算变量所占内存内存空间大小的,单位是字节 ,如果操作数是类型的话,计算的是使用类型创建的变量所占内存空间的大小。
注: sizeof 只关注占⽤内存空间的大小,不在乎内存中存放什么数据。

例:

c 复制代码
#include <stdio.h>
int main()
{
	int a = 10;
	printf("%d\n", sizeof(a));
	printf("%d\n", sizeof a);
	printf("%d\n", sizeof(int));
	return 0;
}

运行结果:

2.2 strlen

strlen 是C语言库函数,功能是求字符串⻓度。函数原型如下:

c 复制代码
 size_t strlen ( const char * str );

:统计的是从strlen 函数的参数str 中这个地址开始向后,\0之前字符中字符的个数。strlen 函数会⼀直向后找\0 字符,直到找到\0为止,所以可能存在越界查找

例:

c 复制代码
#include <stdio.h>
int main()
{
	char arr1[3] = { 'a', 'b', 'c' };
	char arr2[] = "abc";
	printf("%d\n", strlen(arr1));
	printf("%d\n", strlen(arr2));

	printf("%d\n", sizeof(arr1));
	printf("%d\n", sizeof(arr2));
	return 0;
}

运行结果:

2.2 sizeof和 strlen比较

总结---每日励志时刻

相关推荐
郝学胜-神的一滴3 小时前
Three.js光照技术详解:为3D场景注入灵魂
开发语言·前端·javascript·3d·web3·webgl
fie88893 小时前
基于Matlab的深度堆叠自编码器(SAE)实现与分类应用
开发语言·分类
_w_z_j_3 小时前
C++11----列表初始化和initializer_list
开发语言·c++
冬天的雪20084 小时前
java内存性能优化工具Mat
java·开发语言
Le1Yu4 小时前
消息队列以及RabbitMQ的使用
java·开发语言
羚羊角uou4 小时前
【Linux】线程池
java·开发语言
Fcy6484 小时前
C++ vector容器的解析和使用
开发语言·c++·vector
无限进步_5 小时前
C语言文件操作全面解析:从基础概念到高级应用
c语言·开发语言·c++·后端·visual studio
_OP_CHEN5 小时前
C++基础:(十五)queue的深度解析和模拟实现
开发语言·c++·stl·bfs·queue·容器适配器·queue模拟实现