本文所有代码均在仓库中,这是一个完整的由纯C语言实现的可以存储任意类型元素的数据结构的工程项目。
最后也是最重要的一点,数据结构的通用性和舒适的体验感,下面以平衡二叉树为例:
#include "tree-structure/balanced-binary-tree/BalancedBinaryTree.h"
#include "tree-structure/balanced-binary-tree/BalancedBinaryTree.h"
int dataCompare(void *, void *);
typedef struct People {
char *name;
int age;
} *People;
int main(int argc, char **argv) {
struct People dataList[] = {
{"张三", 15},
{"李四", 3},
{"王五", 7},
{"赵六", 10},
{"田七", 9},
{"周八", 8},
};
BalancedBinaryTree tree = balancedBinaryTreeConstructor(NULL, 0, dataCompare);
for (int i = 0; i < 6; ++i) {
balancedBinaryTreeInsert(&tree, dataList + i, dataCompare);
}
return 0;
}
/**
* 根据人的年龄比较
*/
int dataCompare(void *data1, void *data2) {
int sub = ((People) data1)->age - ((People) data2)->age;
if (sub > 0) {
return 1;
} else if (sub < 0) {
return -1;
} else {
return 0;
}
}
#include "tree-structure/balanced-binary-tree/BalancedBinaryTree.h"
int dataCompare(void *, void *);
void dataPrint(void *);
typedef struct People {
char *name;
int age;
} *People;
int main(int argc, char **argv) {
struct People dataList[] = {
{"张三", 15},
{"李四", 3},
{"王五", 7},
{"赵六", 10},
{"田七", 9},
{"周八", 8},
};
BalancedBinaryTree tree = balancedBinaryTreeConstructor(NULL, 0, dataCompare);
for (int i = 0; i < 6; ++i) {
balancedBinaryTreeInsert(&tree, dataList + i, dataCompare);
balancedBinaryTreePrint(tree, dataPrint);
printf("-------------\n");
}
return 0;
}
/**
* 根据人的年龄比较
*/
int dataCompare(void *data1, void *data2) {
int sub = ((People) data1)->age - ((People) data2)->age;
if (sub > 0) {
return 1;
} else if (sub < 0) {
return -1;
} else {
return 0;
}
}
/**
* 打印人的年龄
* @param data
*/
void dataPrint(void *data) {
People people = (People) data;
printf("%d", people->age);
}
打印的结果如下:
最后期待大佬们的点赞。
int compare (void * a, void * b)
函数用于待排元素的比较:
排序算法就是将结构中所有数据按照关键字有序的过程。排序的分类如下:
评价一个排序算法的指标通常有以下三种:
其中稳定性是指关键字相同的元素在排序前后相对位置是否改变,如果不变则称该排序算法是稳定的,否则就是不稳定的。
算法思想:每次将一个待排序的记录按其关键字大小插入前面已排好序的子序列,直到全部记录插入完成。
/**
* 直接插入排序
* @param dataList
* @param length
*/
void directInsert(void *dataList[], int length, int (*compare)(void *, void *)) {
for (int unOrderListIterator = 2; unOrderListIterator <= length; ++unOrderListIterator) {
void *sortedData = dataList[unOrderListIterator - 1];
int orderListIterator;
for (orderListIterator = unOrderListIterator - 1; orderListIterator >= 1 && compare(sortedData, dataList[orderListIterator - 1]) < 0; --orderListIterator) {
dataList[orderListIterator + 1 - 1] = dataList[orderListIterator - 1];
}
dataList[orderListIterator + 1 - 1] = sortedData;
}
}
/**
* 折半插入排序
* @param dataList
* @param length
*/
void binaryInsertSort(void *dataList[], int length, int (*compare)(void *, void *)) {
for (int unOrderListIterator = 2; unOrderListIterator <= length; ++unOrderListIterator) {
void *sortedData = dataList[unOrderListIterator - 1];
int mid, high = unOrderListIterator - 1, low = 1;
while (low <= high) {
mid = (high + low) / 2;
if (compare(dataList[mid - 1], sortedData) > 0) {
high = mid - 1;
} else {
low = mid + 1;
}
}
for (int orderListIterator = unOrderListIterator; orderListIterator > low; orderListIterator--) {
dataList[orderListIterator - 1] = dataList[orderListIterator - 1 - 1];
}
dataList[low - 1] = sortedData;
}
}
/**
* 希尔排序
* @param dataList
* @param length
* @param compare
*/
void shellSort(void *dataList[], int length, int (*compare)(void *, void *)) {
for (int p = length / 2; p >= 1; p /= 2) {
for (int unOrderListIterator = p + 1; unOrderListIterator <= length; ++unOrderListIterator) {
void *sortedData = dataList[unOrderListIterator - 1];
int orderListIterator;
for (orderListIterator = unOrderListIterator - p; orderListIterator >= 1 && compare(sortedData, dataList[orderListIterator - 1]) < 0; orderListIterator -= p) {
dataList[orderListIterator + p - 1] = dataList[orderListIterator - 1];
}
dataList[orderListIterator + p - 1] = sortedData;
}
}
}
算法思想:根据序列中两个元素关键字的比较结果来对换这两个元素在序列中的位置。
/**
* 冒泡排序
* @param dataList
* @param length
* @param compare
*/
void bubbleSort(void *dataList[], int length, int (*compare)(void *, void *)) {
for (int trip = 1; trip <= length - 1; trip++) {
bool isSwap = false;
for (int j = length; j > trip; j--) {
if (compare(dataList[j - 1], dataList[j - 1 - 1]) < 0) {
swap(dataList + j - 1, dataList + j - 1 - 1);
isSwap = true;
}
}
//如果没有交换则序列已有序
if (!isSwap) {
break;
}
}
}
快速排序算法的平均时间复杂度接近最好时间复杂度的排序算法,是最好的内部排序。
static int partition(void *dataList[], int low, int high, int (*compare)(void *, void *)) {
void *pivot = dataList[low - 1];
while (low < high) {
while (low < high && compare(dataList[high - 1], pivot) > 0) {
high--;
}
dataList[low - 1] = dataList[high - 1];
while (low < high && compare(dataList[low - 1], pivot) <= 0) {
low++;
}
dataList[high - 1] = dataList[low - 1];
}
dataList[low - 1] = pivot;
return low;
}
/**
* 快速排序
* @param dataList
* @param low
* @param high
* @param compare
*/
void quickSort(void *dataList[], int low, int high, int (*compare)(void *, void *)) {
if (low < high) {
int pivotPos = partition(dataList, low, high, compare);
quickSort(dataList, low, pivotPos - 1, compare);
quickSort(dataList, pivotPos + 1, high, compare);
}
}
算法思想:每一趟在待排序序列中选择关键字最小或最大的元素加入有序子序列。
/**
* 简单选择排序
* @param dataList
* @param length
* @param compare
*/
void simpleSelectSort(void *dataList[], int length, int (*compare)(void *, void *)) {
for (int orderListIterator = 1; orderListIterator < length; ++orderListIterator) {
int minIndex = orderListIterator;
for (int unOrderListIterator = orderListIterator + 1; unOrderListIterator <= length; ++unOrderListIterator) {
if (compare(dataList[unOrderListIterator - 1], dataList[minIndex - 1]) < 0) {
minIndex = unOrderListIterator;
}
}
if (minIndex != orderListIterator) {
swap(dataList + orderListIterator - 1, dataList + minIndex - 1);
}
}
}
当一个序列 L [ 1 , … , n ] L[1,\dots,n] L[1,…,n]满足:
可以将堆看成一棵线性存储的完全二叉树:
堆排序首要任务就是先构建一个堆(以大顶堆为例):
堆构建完后就可以进行堆排序了,堆排序的算法思想如下:
如果要在堆中插入或删除元素(以小顶堆为例),那么思想为:
static void heapAdjust(void **dataList, int rootIndex, int length, int (*compare)(void *, void *)) {
void *root = dataList[rootIndex - 1];
//i指向左孩子
for (int i = 2 * rootIndex; i <= length; i *= 2) {
//如果右孩子>左孩子,则让i指向右孩子
if (i < length && compare(dataList[i + 1 - 1], dataList[i - 1]) > 0) {
i++;
}
if (compare(root, dataList[i - 1]) > 0) {
break;
} else {
dataList[rootIndex - 1] = dataList[i - 1];
//调整完当前子树后接着向下调整,以免上一次的交换破坏了下一级的堆
rootIndex = i;
}
}
dataList[rootIndex - 1] = root;
}
static void maxHeapBuild(void **dataList, int length, int (*compare)(void *, void *)) {
for (int i = length / 2; i >= 1; i--) {
int a = *((int *) dataList[i - 1]);
heapAdjust(dataList, i, length, compare);
}
}
/**
* 堆排序
* @param dataList
* @param length
* @param compare
*/
void heapSort(void **dataList, int length, int (*compare)(void *, void *)) {
maxHeapBuild(dataList, length, compare);
for (int i = length; i > 1; i--) {
swap(dataList + i - 1, dataList + 1 - 1);
heapAdjust(dataList, 1, i - 1, compare);
}
}
static void merge(void *dataList[], int length, int low, int mid, int high, int (*compare)(void *, void *)) {
void *temp[length];
int i, j, k;
for (k = low; k <= high; ++k) {
temp[k-1] = dataList[k-1];
}
for (i = low, j = mid + 1, k = low; i <= mid && j <= high; k++) {
if (compare(temp[i - 1], temp[j - 1]) < 0) {
dataList[k - 1] = temp[i - 1];
i++;
} else {
dataList[k - 1] = temp[j - 1];
j++;
}
}
while (i <= mid) {
dataList[k - 1] = temp[i - 1];
k++;
i++;
}
for (; j <= high;) {
dataList[k - 1] = temp[j - 1];
k++;
j++;
}
}
/**
* 归并排序
* @param dataList
* @param length
* @param low
* @param high
* @param compare
*/
void mergeSort(void *dataList[], int length, int low, int high, int (*compare)(void *, void *)) {
if (low < high) {
int mid = (low + high) / 2;
mergeSort(dataList, length, low, mid, compare);
mergeSort(dataList, length, mid + 1, high, compare);
merge(dataList, length, low, mid, high, compare);
}
}
假设长度为
n
n
n的排序列表中每个结点
a
j
a_j
aj的关键字由
d
d
d元组
(
k
j
d
−
1
,
k
j
d
−
2
,
…
,
k
j
1
,
k
j
0
)
(k_j^{d-1},k_j^{d-2},\dots,k_j^1,k_j^0)
(kjd−1,kjd−2,…,kj1,kj0)组成,其中
0
≤
k
j
i
≤
r
−
1
,
(
0
≤
j
<
n
,
0
≤
i
≤
d
−
1
)
0\leq k_j^i\leq r-1,(0\leq j
以int dataList[] = {278, 109, 63, 930, 589, 184, 505, 269, 8, 83}
为例:
/**
* 基数排序
* @param dataList
* @param length
* @param maxLength
*/
void radixSort(int dataList[], int length, int maxLength) {
LinkedQueue queue = linkedQueueConstructor();
for (int i = 0; i < length; ++i) {
linkedQueueEnQueue(queue, dataList + i);
}
LinkedQueue queue0 = linkedQueueConstructor();
LinkedQueue queue1 = linkedQueueConstructor();
LinkedQueue queue2 = linkedQueueConstructor();
LinkedQueue queue3 = linkedQueueConstructor();
LinkedQueue queue4 = linkedQueueConstructor();
LinkedQueue queue5 = linkedQueueConstructor();
LinkedQueue queue6 = linkedQueueConstructor();
LinkedQueue queue7 = linkedQueueConstructor();
LinkedQueue queue8 = linkedQueueConstructor();
LinkedQueue queue9 = linkedQueueConstructor();
LinkedQueue queueList[] = {queue0, queue1, queue2, queue3, queue4, queue5, queue6, queue7, queue8, queue9};
for (int i = 1; i <= maxLength; ++i) {
while (!linkedQueueIsEmpty(queue)) {
void *data = linkedQueueDeQueue(queue);
int key = *(int *) data / (int) pow(10, i - 1) % 10;
linkedQueueEnQueue(queueList[key], data);
}
for (int j = 0; j < 10; ++j) {
LinkedQueue keyQueue = queueList[j];
while (!linkedQueueIsEmpty(keyQueue)) {
linkedQueueEnQueue(queue, linkedQueueDeQueue(keyQueue));
}
}
}
while (!linkedQueueIsEmpty(queue)) {
void *data = linkedQueueDeQueue(queue);
printf("%d,", *(int *) data);
}
}
操作系统以块为单位对磁盘存储空间进行管理,如果要修改磁盘块中的数据,就需要把对应磁盘块的内容读到内存中,在内存中修改后再写回磁盘。在对磁盘数据进行排序时,如果磁盘中的数据过多,那么无法一次将数据全部读到内存中,此时就应该使用外部排序。实现外部排序的思想是使用归并排序的的方法,最少只需要在内存中分配三块大小的缓冲区即可对任意一个大文件进行排序。
外部排序的步骤如下:
在每次构造归并段时都需要把所有的磁盘块读写一遍,并且还要进行内部排序,因此外部排序的时间开销由以下几部分构成:
外部排序的时间开销
=
读写外存的时间
+
内部排序所需时间
+
内部归并所需的时间
外部排序的时间开销=读写外存的时间+内部排序所需时间+内部归并所需的时间
外部排序的时间开销=读写外存的时间+内部排序所需时间+内部归并所需的时间
其中读写外存的时间是外部排序的主要开销,因此可以使用多路归并的方式来减少归并的趟数从而减少读写外存的次数。若对
r
r
r个初始归并段做
k
k
k路归并,则归并树可用
k
k
k叉树表示,若树高为
h
h
h,则归并趟数
n
n
n为:
n
=
h
−
1
=
⌈
l
o
g
k
r
⌉
n=h-1=⌈log_kr⌉
n=h−1=⌈logkr⌉
因此归并路数(增加缓冲区的个数)越多,初始归并段(增加缓冲区的长度)越少,读写磁盘的次数就越少。但多路归并同样存在着问题:
败者树可视为多一个根结点的完全二叉树, k k k个叶结点分别是当前参加比较的元素,非叶子结点用来记忆左右子树中的失败者,而让胜者往上继续进行比较,一直到根结点。
可以将败者树用于多路归并从而减少关键字的对比,从而解决问题二。对于 k k k路归并,第一次构造败者树需要对比关键字 k − 1 k-1 k−1次,有了败者树,选出最小元素,只需要对比关键字 ⌈ l o g 2 k ⌉ ⌈log_2k⌉ ⌈log2k⌉次。
对于传统归并段构造的方法,如果用于内部排序的输入缓冲区可容纳 l l l个记录,则每个初始归并段也只能包含 l l l个记录,若文件共有 n n n个记录,则初始归并段的数量为 r = n l r=\frac{n}{l} r=ln。而置换——选择排序可以构造比内存缓冲区长度长的归并段。置换——选择排序的思想为:
miniMax
),接着读入待排文件的下一记录填充输入缓冲区。miniMax
,那么就不可能将其放到归并段一的末尾,此时找到第二小且大于miniMax
的记录放到归并段。miniMax
时,第一个初始归并段就构造结束。如果采用置换——选择排序构造初始归并段,并将每一个初始归并段看作一个叶子结点,归并段的长度作为结点的权值,则归并树的带权路径长度
W
S
L
WSL
WSL有以下公式成立:
W
S
L
=
读磁盘的次数
=
写磁盘的次数
WSL=读磁盘的次数=写磁盘的次数
WSL=读磁盘的次数=写磁盘的次数
那么 W S L WSL WSL最小的树就是一棵哈夫曼树,从而可以通过构造一棵哈夫曼树以使存盘存取次数最小。在构造哈夫曼树树的过程中,如果初始归并段的数量无法构成严格的 k k k叉哈夫曼树,那么就需要补充长度为0的虚段,再进行构造。对于一棵 k k k叉归并树: