练习33:链表算法

原文:Exercise 33: Linked List Algorithms

译者:飞龙

我将想你介绍涉及到排序的两个算法,你可以用它们操作链表。我首先要警告你,如果你打算对数据排序,不要使用链表,它们对于排序十分麻烦,并且有更好的数据结构作为替代。我向你介绍这两种算法只是因为它们难以在链表上完成,并且让你思考如何高效操作它们。

为了编写这本书,我打算将算法放在两个不同的文件中,list_algos.hlist_algos.c,之后在list_algos_test.c中编写测试。现在你要按照我的结构,因为它足以把事情做好,但是如果你使用其它的库要记住这并不是通用的结构。

这个练习中我打算给你一些额外的挑战,并且希望你不要作弊。我打算先给你单元测试,并且让你打下来。之后让你基于它们在维基百科中的描述,尝试实现这个两个算法,之后看看你的代码是否和我的类似。

冒泡排序和归并排序

互联网的强大之处,就是我可以仅仅给你冒泡排序归并排序的链接,来让你学习它们。是的,这省了我很多字。现在我要告诉你如何使用它们的伪代码来实现它们。你可以像这样来实现算法:

  • 阅读描述,并且观察任何可视化的图表。
  • 使用方框和线条在纸上画出算法,或者使用一些带有数字的卡片(比如扑克牌),尝试手动执行算法。这会向你形象地展示算法的执行过程。
  • list_algos.c文案总创建函数的主干,并且创建list_algos.h文件,之后创建测试代码。
  • 编写第一个测试并且编译所有东西。
  • 回到维基百科页面,复制粘贴伪代码到你创建的函数中(不是C代码)。
  • 将伪代码翻译成良好的C代码,就像我教你的那样,使用你的单元测试来保证它有效。
  • 为边界情况补充一些测试,例如空链表,排序号的链表,以及其它。
  • 对下一个算法重复这些过程并测试。

我只是告诉你理解大多数算法的秘密,直到你碰到一些更加麻烦的算法。这里你只是按照维基百科来实现冒泡排序和归并排序,它们是一个好的起始。

单元测试

下面是你应该通过的单元测试:

  1. #include "minunit.h"
  2. #include <lcthw/list_algos.h>
  3. #include <assert.h>
  4. #include <string.h>
  5. char *values[] = {"XXXX", "1234", "abcd", "xjvef", "NDSS"};
  6. #define NUM_VALUES 5
  7. List *create_words()
  8. {
  9. int i = 0;
  10. List *words = List_create();
  11. for(i = 0; i < NUM_VALUES; i++) {
  12. List_push(words, values[i]);
  13. }
  14. return words;
  15. }
  16. int is_sorted(List *words)
  17. {
  18. LIST_FOREACH(words, first, next, cur) {
  19. if(cur->next && strcmp(cur->value, cur->next->value) > 0) {
  20. debug("%s %s", (char *)cur->value, (char *)cur->next->value);
  21. return 0;
  22. }
  23. }
  24. return 1;
  25. }
  26. char *test_bubble_sort()
  27. {
  28. List *words = create_words();
  29. // should work on a list that needs sorting
  30. int rc = List_bubble_sort(words, (List_compare)strcmp);
  31. mu_assert(rc == 0, "Bubble sort failed.");
  32. mu_assert(is_sorted(words), "Words are not sorted after bubble sort.");
  33. // should work on an already sorted list
  34. rc = List_bubble_sort(words, (List_compare)strcmp);
  35. mu_assert(rc == 0, "Bubble sort of already sorted failed.");
  36. mu_assert(is_sorted(words), "Words should be sort if already bubble sorted.");
  37. List_destroy(words);
  38. // should work on an empty list
  39. words = List_create(words);
  40. rc = List_bubble_sort(words, (List_compare)strcmp);
  41. mu_assert(rc == 0, "Bubble sort failed on empty list.");
  42. mu_assert(is_sorted(words), "Words should be sorted if empty.");
  43. List_destroy(words);
  44. return NULL;
  45. }
  46. char *test_merge_sort()
  47. {
  48. List *words = create_words();
  49. // should work on a list that needs sorting
  50. List *res = List_merge_sort(words, (List_compare)strcmp);
  51. mu_assert(is_sorted(res), "Words are not sorted after merge sort.");
  52. List *res2 = List_merge_sort(res, (List_compare)strcmp);
  53. mu_assert(is_sorted(res), "Should still be sorted after merge sort.");
  54. List_destroy(res2);
  55. List_destroy(res);
  56. List_destroy(words);
  57. return NULL;
  58. }
  59. char *all_tests()
  60. {
  61. mu_suite_start();
  62. mu_run_test(test_bubble_sort);
  63. mu_run_test(test_merge_sort);
  64. return NULL;
  65. }
  66. RUN_TESTS(all_tests);

建议你从冒泡排序开始,使它正确,之后再测试归并。我所做的就是编写函数原型和主干,让这三个文件能够编译,但不能通过测试。之后你将实现填充进入之后才能够工作。

实现

你作弊了吗?之后的练习中,我只会给你单元测试,并且让自己实现它。对于你来说,不看这段代码知道你自己实现它是一种很好的练习。下面是list_algos.clist_algos.h的代码:

  1. #ifndef lcthw_List_algos_h
  2. #define lcthw_List_algos_h
  3. #include <lcthw/list.h>
  4. typedef int (*List_compare)(const void *a, const void *b);
  5. int List_bubble_sort(List *list, List_compare cmp);
  6. List *List_merge_sort(List *list, List_compare cmp);
  7. #endif
  1. #include <lcthw/list_algos.h>
  2. #include <lcthw/dbg.h>
  3. inline void ListNode_swap(ListNode *a, ListNode *b)
  4. {
  5. void *temp = a->value;
  6. a->value = b->value;
  7. b->value = temp;
  8. }
  9. int List_bubble_sort(List *list, List_compare cmp)
  10. {
  11. int sorted = 1;
  12. if(List_count(list) <= 1) {
  13. return 0; // already sorted
  14. }
  15. do {
  16. sorted = 1;
  17. LIST_FOREACH(list, first, next, cur) {
  18. if(cur->next) {
  19. if(cmp(cur->value, cur->next->value) > 0) {
  20. ListNode_swap(cur, cur->next);
  21. sorted = 0;
  22. }
  23. }
  24. }
  25. } while(!sorted);
  26. return 0;
  27. }
  28. inline List *List_merge(List *left, List *right, List_compare cmp)
  29. {
  30. List *result = List_create();
  31. void *val = NULL;
  32. while(List_count(left) > 0 || List_count(right) > 0) {
  33. if(List_count(left) > 0 && List_count(right) > 0) {
  34. if(cmp(List_first(left), List_first(right)) <= 0) {
  35. val = List_shift(left);
  36. } else {
  37. val = List_shift(right);
  38. }
  39. List_push(result, val);
  40. } else if(List_count(left) > 0) {
  41. val = List_shift(left);
  42. List_push(result, val);
  43. } else if(List_count(right) > 0) {
  44. val = List_shift(right);
  45. List_push(result, val);
  46. }
  47. }
  48. return result;
  49. }
  50. List *List_merge_sort(List *list, List_compare cmp)
  51. {
  52. if(List_count(list) <= 1) {
  53. return list;
  54. }
  55. List *left = List_create();
  56. List *right = List_create();
  57. int middle = List_count(list) / 2;
  58. LIST_FOREACH(list, first, next, cur) {
  59. if(middle > 0) {
  60. List_push(left, cur->value);
  61. } else {
  62. List_push(right, cur->value);
  63. }
  64. middle--;
  65. }
  66. List *sort_left = List_merge_sort(left, cmp);
  67. List *sort_right = List_merge_sort(right, cmp);
  68. if(sort_left != left) List_destroy(left);
  69. if(sort_right != right) List_destroy(right);
  70. return List_merge(sort_left, sort_right, cmp);
  71. }

冒泡排序并不难以理解,虽然它非常慢。归并排序更为复杂,实话讲如果我想要牺牲可读性的话,我会花一点时间来优化代码。

归并排序有另一种“自底向上”的实现方式,但是它太难了,我就没有选择它。就像我刚才说的那样,在链表上编写排序算法没有什么意思。你可以把时间都花在使它更快,它比起其他可排序的数据结构会相当版。链表的本质决定了如果你需要对数据进行排序,你就不要使用它们(尤其是单向的)。

你会看到什么

如果一切都正常工作,你会看到这些:

  1. $ make clean all
  2. rm -rf build src/lcthw/list.o src/lcthw/list_algos.o tests/list_algos_tests tests/list_tests
  3. rm -f tests/tests.log
  4. find . -name "*.gc*" -exec rm {} \;
  5. rm -rf `find . -name "*.dSYM" -print`
  6. cc -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG -fPIC -c -o src/lcthw/list.o src/lcthw/list.c
  7. cc -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG -fPIC -c -o src/lcthw/list_algos.o src/lcthw/list_algos.c
  8. ar rcs build/liblcthw.a src/lcthw/list.o src/lcthw/list_algos.o
  9. ranlib build/liblcthw.a
  10. cc -shared -o build/liblcthw.so src/lcthw/list.o src/lcthw/list_algos.o
  11. cc -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG build/liblcthw.a tests/list_algos_tests.c -o tests/list_algos_tests
  12. cc -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG build/liblcthw.a tests/list_tests.c -o tests/list_tests
  13. sh ./tests/runtests.sh
  14. Running unit tests:
  15. ----
  16. RUNNING: ./tests/list_algos_tests
  17. ALL TESTS PASSED
  18. Tests run: 2
  19. tests/list_algos_tests PASS
  20. ----
  21. RUNNING: ./tests/list_tests
  22. ALL TESTS PASSED
  23. Tests run: 6
  24. tests/list_tests PASS
  25. $

这个练习之后我就不会向你展示这样的输出了,除非有必要向你展示它的工作原理。你应该能知道我运行了测试,并且通过了所有测试。

如何改进

退回去查看算法描述,有一些方法可用于改进这些实现,其中一些是很显然的:

  • 归并排序做了大量的链表复制和创建操作,寻找减少它们的办法。
  • 归并排序的维基百科描述提到了一些优化,实现它们。
  • 你能使用List_splitList_join(如果你实现了的话)来改进归并排序嘛?
  • 浏览所有防御性编程原则,检查并提升这一实现的健壮性,避免NULL指针,并且创建一个可选的调试级别的不变量,在排序后实现is_sorted的功能。

附加题

  • 创建单元测试来比较这两个算法的性能。你需要man 3 time来查询基本的时间函数,并且需要运行足够的迭代次数,至少以几秒钟作为样本。
  • 改变需要排序的链表中的数据总量,看看耗时如何变化。
  • 寻找方法来创建不同长度的随机链表,并且测量需要多少时间,之后将它可视化并与算法的描述对比。
  • 尝试解释为什么对链表排序十分麻烦。
  • 实现List_insert_sorted(有序链表),它使用List_compare,接收一个值,将其插入到正确的位置,使链表有序。它与创建链表后再进行排序相比怎么样?
  • 尝试实现维基百科上“自底向上”的归并排序。上面的代码已经是C写的了,所以很容易重新创建,但是要试着理解它的工作原理,并与这里的低效版本对比。