近日周立功教授公開了數年的心血之作《程序設計與數據結構》,電子版已無償性分享到電子工程師與高校群體下載,經周立功教授授權,特對本書內容進行連載。
由于使用迭代器可以輕松地實現指針的前移或后移,因此可以使用迭代器接口實現冒泡排序算法。其函數原型為:
void iter_sort(iterator_if_t *p_if, iterator_t begin, iterator_t end, compare_t compare, swap_t swap) ;
其中,p_if表示算法使用的迭代器接口。begin與end是一對迭代器,表示算法的操作范圍,但不一定是容器的首尾迭代器,因此算法可以處理任何范圍的數據。為了判定范圍的有效性,習慣采用前閉后開范圍表示法,即使用begin和end表示的范圍為[begin,end),表示范圍涵蓋bigen到end(不含end)之間的所有元素。當begin==end時,上述所表現的便是一個空范圍。
compare同樣也是比較函數,但比較的類型發生了變化,用于比較兩個迭代器所對應的值。其類型compare_t定義如下:
typedef int (*compare_t)(iterator_t it1, iterator_t it2);
swap函數用于交換兩個迭代器所對應的數據,其類型swap_t定義如下:
typedef void (*swap_t)(iterator_t it1, iterator_t it2);
由此可見,接口中只有迭代器,根本沒有容器的蹤影,從而做到了容器與冒泡排序算法徹底分離,基于迭代器的冒泡排序算法詳見程序清單3.56。
程序清單3.56冒泡排序算法函數
1 void iter_sort(iterator_if_t *p_if, iterator_t begin, iterator_t end, compare_t compare, swap_t swap)
2 {
3 int flag = 1; // flag = 1,表示指針的內容未交換
4 iterator_t it1 = begin; // it1指向數組變量的首元素
5 iterator_t it2 = end;
67 iterator_t it_next; // pNext指向p1所指向的元素的下一個元素
8 if (begin == end) { //沒有需要算法處理的迭代器
9 return;
10 }
11 iterator_prev(p_if, &it2); // it2指向需要排序的最后一個元素12 while (it2 != begin){
13 it1 = begin;
14 flag = 1;
15 while(it1 != it2){
16 it_next = it1; //暫存
17 iterator_next(p_if, &it_next); // it_next為 it1 的下一個元素
18 if(compare(it1, it_next) > 0){
19 swap(it1, it_next); //交換內容
20 flag = 0; // flag = 0,表示指針的內容已交換
21 }22 it1 = it_next; // it1的下一個元素
23 }
24 if(flag) return; //沒有交換,表示已經有序,則直接返回
25 iterator_prev(p_if, &it2); // it2向前移
26 }
27 }下面以一個簡單的例子來測試驗證基于迭代器的冒泡排序算法,詳見程序清單3.57。將整數存放到雙向鏈表中,首先將5、4、3、2、1分別加在鏈表的尾部,接著調用dlist_foreach()遍歷鏈表,看是否符合預期,然后再調用算法庫的iter_sort()排序。當排序完畢后鏈表的元素應該是從小到大排列的,再次調用算法庫的dilst_foreach()遍歷鏈表,看是否符合預期。
程序清單3.57使用雙向鏈表、算法和迭代器
1 #include
2 #include "iterator.h"
3
4 typedef struct _dlist_int{
5 dlist_node_t node; //包含鏈表結點
6 int data; // int類型數據
7 }dlist_int_t;
8
9 int list_node_process(void *p_arg, dlist_node_t *p_node)
10 {
11 printf("%d ", ((dlist_int_t *)p_node) -> data);12 return 0;
13 }
14
15 static int __compare(iterator_t it1, iterator_t it2)
16 {
17 return ((dlist_int_t *)it1) -> data - ((dlist_int_t *)it2) -> data;
18 }
19
20 static void __swap(iterator_t it1, iterator_t it2)
21 {
22 int data = ((dlist_int_t *)it2) -> data;
23 ((dlist_int_t *)it2) -> data = ((dlist_int_t *)it1) -> data;
24 ((dlist_int_t *)it1) -> data = data;
25 }
2627 int main(int argc, char *argv[])
28 {
29 iterator_if_t iterator_if;
30 dlist_head_t head; //定義鏈表頭結點
31 dlist_int_t node[5]; //定義5個結點空間
32 int i;
3334 dlist_init(&head);
35
36 for (i = 0; i < 5; i++) {? ???????????????????????? //?將5個結點添加至鏈表尾部
37 node[i].data = 5 - i; //使值的順序為 5 ~ 1
38 dlist_add_tail(&head, &(node[i].node));
39 }
40 dlist_iterator_if_get(&iterator_if);
4142 printf("\nBefore bubble sort:\n");
43 dlist_foreach (&head, list_node_process, NULL); //打印排序前的情況
44
45 iter_sort(&iterator_if, dlist_begin_get(&head), dlist_end_get(&head),__compare, __swap);
46
47 printf("\nAfter bubble sort:\n");
48 dlist_foreach (&head, list_node_process, NULL); //打印排序后的情況
49 return 0;
50 }在這里,使用了dlist_foreach()遍歷函數,既然通過迭代器能夠實現冒泡排序,那么也能通過迭代器實現簡單的遍歷算法,此時遍歷算法與具體容器無關。遍歷函數的原型如下:
void iter_foreach(iterator_if_t *p_if, iterator_t begin, iterator_t end, visit_t visit, void *p_arg);
其中,p_if表示算法使用的迭代器接口,begin與end表示算法需要處理的迭代器范圍,visit是用戶自定義的遍歷迭代器的函數。其類型visit_t定義如下:
typedef int (*visit_t)(void *p_arg, iterator_t it);
visit_t的參數是p_arg指針和it迭代器,其返回值為int類型的函數指針。每遍歷一個結點均會調用visit指向的函數,傳遞給p_arg的值即為用戶參數,其值為iter_foreach()函數的p_arg參數,p_arg的值完全是由用戶決定的,傳遞給it迭代器的值即為指向當前遍歷的迭代器,iter_foreach()函數的實現詳見程序清單3.58。
程序清單3.58遍歷算法函數
1 void iter_foreach(iterator_if_t *p_if, iterator_t begin, iterator_t end, visit_t visit, void *p_arg)
2 {
3 iterator_t it = begin;
4 while(it != end){
5 if (visit(p_arg, it) < 0) {???? ?????????????????? //?若返回值為負值,表明用戶終止了遍歷6 return;
7 }
8 iterator_next(p_if, &it); //讓迭代器向后移動
9 }
10 }現在可以將程序清單3.57中的第43行和第48行中的dlist_foreach()函數修改為使用iter_foreach()函數,看能否得到相同的效果?
如果將數據保存在數組變量中,那么將如何使用已有的冒泡排序算法呢?由于數組也是容器,因此只要實現基于數組的迭代器即可,詳見程序清單3.59。
程序清單3.59使用數組實現迭代器接口
1 typedef int element_type_t;
2
3 static void __array_iterator_next(iterator_t *p_iter)
4 {
5 (*(element_type_t **)(p_iter))++; //讓迭代器指向下一個數據
6 }
7
8 static void __array_iterator_prev(iterator_t *p_iter)
9 {10 (*(element_type_t **)(p_iter))--; //讓迭代器指向前一個數據
11 }
12
13 void array_iterator_if_get(iterator_if_t *p_if)
14 {
15 iterator_if_init(p_if, __array_iterator_next, __array_iterator_prev);
16 }基于新的迭代器,同樣可以直接使用冒泡排序算法實現排序,詳見程序清單3.60。
程序清單3.60使用數組、算法和迭代器
1 #include
2 #include "iterator.h"
3
4 static int __visit(void *p_arg, iterator_t it)
5 {
6 printf("%d ", *(int *)it);
7 return 0;
8 }
9
10 static int __compare(iterator_t it1, iterator_t it2)
11 {12 return *(int *)it1 - *(int *)it2;
13 }
14
15 static void __swap(iterator_t it1, iterator_t it2)
16 {
17 int data = *(int *)it2;
18 *(int *)it2 = *(int *)it1;
19 *(int *)it1 = data;
20 }
2122 int main(int argc, char *argv[])
23 {
24 iterator_if_t iterator_if;
25 int a[] = {5, 3, 2, 4, 1};
26 array_iterator_if_get(&iterator_if);
27
28 printf("\nBefore bubble sort:\n");
29 iter_foreach(&iterator_if, a, a + 5, __visit, NULL);
30
31 iter_sort(&iterator_if, a, a + 5, __compare, __swap);
32
33 printf("\nAfter bubble sort:\n");34 iter_foreach(&iterator_if, a, a + 5, __visit, NULL);
35 return 0;
36 }由此可見,通過迭代器冒泡排序算法也得到了復用。如果算法庫里有幾百個函數,那么只要實現迭代器接口的2個函數即可,從而達到復用代碼的目的。顯然,迭代器是一種更靈活的遍歷行為,它可以按任意順序訪問容器中的元素,而且不會暴露容器的內部結構。
-
算法
+關注
關注
23文章
4681瀏覽量
94323 -
指針
+關注
關注
1文章
484瀏覽量
70896 -
周立功
+關注
關注
38文章
130瀏覽量
38028 -
迭代器
+關注
關注
0文章
45瀏覽量
4426
原文標題:周立功:迭代器——算法的接口
文章出處:【微信號:Zlgmcu7890,微信公眾號:周立功單片機】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
Python高級特性:迭代器切片的應用
什么是void指針?void指針有何功能
python迭代器
OpenHarmony中的HDF單鏈表及其迭代器
OpenHarmony中的HDF單鏈表及其迭代器
人工智能抗疫 AI技術實現了篩查的時間窗口前移
【C和指針】指針

Python中的迭代器介紹 迭代器在scoreboard中的應用有哪些?

C++智能指針的底層實現原理

評論