1.qsort函数的分析
qsort()函数是C语言库函数中的一种排序函数,排序方法为快速排序。其可以排序任意类型的数组元素。 同时,我们也知道,冒泡排序的核心思想就是:两个相邻的元素进行比较,再根据需要进行排序。
所以,在qsort函数的模拟实现中,我们大致需要三个函数:比较函数:int cmp(const void* p1, const void* p2)、交换函数:void swap(char* buf1,char* buf2,size_t width)、冒泡排序:void bubble(void* base, size_t sz, size_t width, int(*cmp)(const void* p1, const void* p2))。
下面我们以模拟整形数组排序为例,来分析一下我们的代码。
2.qsort模拟函数代码实现
2.1模拟整形数组
#include <stdio.h>
#include <stdlib.h>
int cmp(const void* p1, const void* p2)
{
return *(int*)p1 - *(int*)p2;
}
void swap(char* buf1,char* buf2,size_t width)
{
int i = 0;
for (i = 0; i < width; i++)
{
char tmp = *buf1;
*buf1 = *buf2;
*buf2 = tmp;
buf1++;
buf2++;
}
}
void bubble(void* base, size_t sz, size_t width, int(*cmp)(const void* p1, const void* p2))
{
int i = 0, j = 0;
for (i = 0; i < sz - 1; i++)
{
for (j = 0; j < sz - 1 - i; j++)
{
if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0)
{
swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
}
}
}
}
int main()
{
int arr[10] = { 3, 7, 1, 9, 4, 10, 6, 5, 2, 8 };
int sz = sizeof(arr) / sizeof(arr[0]);
bubble(arr, sz, sizeof(arr[0]), cmp);
int i = 0;
for (i = 0; i < sz; i++)
{
printf("%d ", arr[i]);
}
return 0;
}
结果为:
在swap()函数中,我们使用的参数类型为char*类型,由于不知道传入的数据为什么类型,所以使用char*类型最为合适,这样无论传参为什么类型,都可以一个字节一个字节进行交换。
2.2模拟字符数组
#include <stdio.h>
#include <stdlib.h>
int cmp(const void* p1, const void* p2)
{
return *(char *)p1 - *(char *)p2;
}
void swap(char* buf1, char* buf2, size_t width)
{
int i = 0;
for (i = 0; i < width; i++)
{
char tmp = *buf1;
*buf1 = *buf2;
*buf2 = tmp;
buf1++;
buf2++;
}
}
void bubble(void* base, size_t sz, size_t width, int(*cmp)(const void* p1, const void* p2))
{
int i = 0, j = 0;
for (i = 0; i < sz - 1; i++)
{
for (j = 0; j < sz - 1 - i; j++)
{
if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0)
{
swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
}
}
}
}
int main()
{
char arr[10] = { 'c', 'g', 'e', 'b', 'j', 'd', 'a', 'f', 'i', 'h'};
int sz = sizeof(arr) / sizeof(arr[0]);
bubble(arr, sz, sizeof(arr[0]), cmp);
int i = 0;
for (i = 0; i < sz; i++)
{
printf("%c ", arr[i]);
}
return 0;
}
结果为:
2.3模拟字符串排序
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int cmp(const void* p1, const void* p2)
{
return strcmp(*(char**)p1,*(char**)p2);
}
void swap(char* buf1, char* buf2, size_t width)
{
int i = 0;
for (i = 0; i < width; i++)
{
char tmp = *buf1;
*buf1 = *buf2;
*buf2 = tmp;
buf1++;
buf2++;
}
}
void bubble(void* base, size_t sz, size_t width, int(*cmp)(const void* p1, const void* p2))
{
int i = 0, j = 0;
for (i = 0; i < sz - 1; i++)
{
for (j = 0; j < sz - 1 - i; j++)
{
if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0)
{
swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
}
}
}
}
int main()
{
char arr1[10] = "niko";
char arr2[10] = "simple";
char arr3[10] = "zywoo";
char arr4[10] = "device";
char* arr[4] = { arr1,arr2,arr3,arr4 };
int sz = sizeof(arr) / sizeof(arr[0]);
bubble(arr, sz, sizeof(arr[0]), cmp);
int i = 0;
for (i = 0; i < sz; i++)
{
printf("%s ", arr[i]);
}
return 0;
}
结果为:
2.4模拟结构体排序
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
struct student
{
char name[20];
int age;
};
int cmp_name(const void* p1, const void* p2)
{
return strcmp(((struct student*)p1)->name, ((struct student*)p2)->name);
}
void swap(char* buf1, char* buf2, size_t width)
{
int i = 0;
for (i = 0; i < width; i++)
{
char tmp = *buf1;
*buf1 = *buf2;
*buf2 = tmp;
buf1++;
buf2++;
}
}
void bubble(void* base, size_t sz, size_t width, int(*cmp)(const void* p1, const void* p2))
{
int i = 0, j = 0;
for (i = 0; i < sz - 1; i++)
{
for (j = 0; j < sz - 1 - i; j++)
{
if (cmp_name((char*)base + j * width, (char*)base + (j + 1) * width) > 0)
{
swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
}
}
}
}
int main()
{
struct student s[] = { {"zhang san",18},{"li si",30},{"wang wu",46} };
int sz = sizeof(s) / sizeof(s[0]);
bubble(s, sz, sizeof(s[0]), cmp_name);
for (int i = 0; i < sz; i++)
{
printf("%s %d\n", s[i].name, s[i].age);
}
return 0;
}
结果为: