linux内核数据结构之链表-实现

代码:list.h

  1 /*
  2  * @file list.h
  3  * @author PF
  4  * @date 2017/05/1
  5  *
  6  * port from linux kernel list.h: https://github.com/torvalds/linux/raw/master/include/linux/list.h
  7  *
  8  * Here is a recipe to cook list.h for user space program.
  9  * 1. copy list.h from linux/include/list.h
 10  * 2. remove
 11  *     - #ifdef __KERNE__ and its #endif
 12  *     - all #include line
 13  *     - prefetch() and rcu related functions
 14  * 3. add macro offsetof() and container_of
 15  */
 16 
 17 #ifndef LIST_H_
 18 #define LIST_H_ (1)
 19 
 20 // import from include/linux/types.h
 21 struct list_head {
 22     struct list_head *next, *prev;
 23 };
 24 
 25 struct hlist_head {
 26     struct hlist_node *first;
 27 };
 28 
 29 struct hlist_node {
 30     struct hlist_node *next, **pprev;
 31 };
 32 
 33 // import from include/linux/poison.h
 34 
 35 /*
 36  * Architectures might want to move the poison pointer offset
 37  * into some well-recognized area such as 0xdead000000000000,
 38  * that is also not mappable by user-space exploits:
 39  */
 40 #ifdef CONFIG_ILLEGAL_POINTER_VALUE
 41 # define POISON_POINTER_DELTA _AC(CONFIG_ILLEGAL_POINTER_VALUE, UL)
 42 #else
 43 # define POISON_POINTER_DELTA (0)
 44 #endif
 45 
 46 /*
 47  * These are non-NULL pointers that will result in page faults
 48  * under normal circumstances, used to verify that nobody uses
 49  * non-initialized list entries.
 50  */
 51 #define LIST_POISON1  ((void *) 0x00100100 + POISON_POINTER_DELTA)
 52 #define LIST_POISON2  ((void *) 0x00200200 + POISON_POINTER_DELTA)
 53 
 54 // import from include/linux/stddef.h
 55 #undef offsetof
 56 #ifdef __compiler_offsetof
 57 #define offsetof(TYPE,MEMBER) __compiler_offsetof(TYPE,MEMBER)
 58 #else
 59 #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
 60 #endif
 61 
 62 // import from include/linux/kernel.h
 63 /**
 64 * container_of - cast a member of a structure out to the containing structure
 65 * @ptr:        the pointer to the member.
 66 * @type:       the type of the container struct this is embedded in.
 67 * @member:     the name of the member within the struct.
 68 *
 69 */
 70 #define container_of(ptr, type, member) ({                      \
 71         const typeof( ((type *)0)->member ) *__mptr = (ptr);    \
 72         (type *)( (char *)__mptr - offsetof(type,member) );})
 73 
 74 /*
 75  * Simple doubly linked list implementation.
 76  *
 77  * Some of the internal functions ("__xxx") are useful when
 78  * manipulating whole lists rather than single entries, as
 79  * sometimes we already know the next/prev entries and we can
 80  * generate better code by using them directly rather than
 81  * using the generic single-entry routines.
 82  */
 83 
 84 #define LIST_HEAD_INIT(name) { &(name), &(name) }
 85 
 86 #define LIST_HEAD(name) \
 87     struct list_head name = LIST_HEAD_INIT(name)
 88 
 89 static inline void INIT_LIST_HEAD(struct list_head *list) {
 90     list->next = list;
 91     list->prev = list;
 92 }
 93 
 94 /*
 95  * Insert a new entry between two known consecutive entries.
 96  *
 97  * This is only for internal list manipulation where we know
 98  * the prev/next entries already!
 99  */
100 #ifndef CONFIG_DEBUG_LIST
101 static inline void __list_add(struct list_head *new,
102     struct list_head *prev,
103     struct list_head *next) {
104     next->prev = new;
105     new->next = next;
106     new->prev = prev;
107     prev->next = new;
108 }
109 #else
110 extern void __list_add(struct list_head *new,
111                   struct list_head *prev,
112                   struct list_head *next);
113 #endif
114 
115 /**
116  * list_add - add a new entry
117  * @new: new entry to be added
118  * @head: list head to add it after
119  *
120  * Insert a new entry after the specified head.
121  * This is good for implementing stacks.
122  */
123 static inline void list_add(struct list_head *new, struct list_head *head) {
124     __list_add(new, head, head->next);
125 }
126 
127 /**
128  * list_add_tail - add a new entry
129  * @new: new entry to be added
130  * @head: list head to add it before
131  *
132  * Insert a new entry before the specified head.
133  * This is useful for implementing queues.
134  */
135 static inline void list_add_tail(struct list_head *new, struct list_head *head) {
136     __list_add(new, head->prev, head);
137 }
138 
139 /*
140  * Delete a list entry by making the prev/next entries
141  * point to each other.
142  *
143  * This is only for internal list manipulation where we know
144  * the prev/next entries already!
145  */
146 static inline void __list_del(struct list_head *prev, struct list_head *next) {
147     next->prev = prev;
148     prev->next = next;
149 }
150 
151 /**
152  * list_del - deletes entry from list.
153  * @entry: the element to delete from the list.
154  * Note: list_empty() on entry does not return true after this, the entry is
155  * in an undefined state.
156  */
157 #ifndef CONFIG_DEBUG_LIST
158 
159 static inline void __list_del_entry(struct list_head *entry) {
160     __list_del(entry->prev, entry->next);
161 }
162 
163 static inline void list_del(struct list_head *entry) {
164     __list_del(entry->prev, entry->next);
165     entry->next = LIST_POISON1;
166     entry->prev = LIST_POISON2;
167 }
168 
169 #else
170 extern void __list_del_entry(struct list_head *entry);
171 extern void list_del(struct list_head *entry);
172 #endif
173 
174 /**
175  * list_replace - replace old entry by new one
176  * @old : the element to be replaced
177  * @new : the new element to insert
178  *
179  * If @old was empty, it will be overwritten.
180  */
181 static inline void list_replace(struct list_head *old,
182         struct list_head *new) {
183     new->next = old->next;
184     new->next->prev = new;
185     new->prev = old->prev;
186     new->prev->next = new;
187 }
188 
189 static inline void list_replace_init(struct list_head *old,
190         struct list_head *new) {
191     list_replace(old, new);
192     INIT_LIST_HEAD(old);
193 }
194 
195 /**
196  * list_del_init - deletes entry from list and reinitialize it.
197  * @entry: the element to delete from the list.
198  */
199 static inline void list_del_init(struct list_head *entry) {
200     __list_del_entry(entry);
201     INIT_LIST_HEAD(entry);
202 }
203 
204 /**
205  * list_move - delete from one list and add as another's head
206  * @list: the entry to move
207  * @head: the head that will precede our entry
208  */
209 static inline void list_move(struct list_head *list, struct list_head *head) {
210     __list_del_entry(list);
211     list_add(list, head);
212 }
213 
214 /**
215  * list_move_tail - delete from one list and add as another's tail
216  * @list: the entry to move
217  * @head: the head that will follow our entry
218  */
219 static inline void list_move_tail(struct list_head *list,
220         struct list_head *head) {
221     __list_del_entry(list);
222     list_add_tail(list, head);
223 }
224 
225 /**
226  * list_is_last - tests whether @list is the last entry in list @head
227  * @list: the entry to test
228  * @head: the head of the list
229  */
230 static inline int list_is_last(const struct list_head *list,
231         const struct list_head *head) {
232     return list->next == head;
233 }
234 
235 /**
236  * list_empty - tests whether a list is empty
237  * @head: the list to test.
238  */
239 static inline int list_empty(const struct list_head *head) {
240     return head->next == head;
241 }
242 
243 /**
244  * list_empty_careful - tests whether a list is empty and not being modified
245  * @head: the list to test
246  *
247  * Description:
248  * tests whether a list is empty _and_ checks that no other CPU might be
249  * in the process of modifying either member (next or prev)
250  *
251  * NOTE: using list_empty_careful() without synchronization
252  * can only be safe if the only activity that can happen
253  * to the list entry is list_del_init(). Eg. it cannot be used
254  * if another CPU could re-list_add() it.
255  */
256 static inline int list_empty_careful(const struct list_head *head) {
257     struct list_head *next = head->next;
258     return (next == head) && (next == head->prev);
259 }
260 
261 /**
262  * list_rotate_left - rotate the list to the left
263  * @head: the head of the list
264  */
265 static inline void list_rotate_left(struct list_head *head) {
266     struct list_head *first;
267 
268     if (!list_empty(head)) {
269         first = head->next;
270         list_move_tail(first, head);
271     }
272 }
273 
274 /**
275  * list_is_singular - tests whether a list has just one entry.
276  * @head: the list to test.
277  */
278 static inline int list_is_singular(const struct list_head *head) {
279     return !list_empty(head) && (head->next == head->prev);
280 }
281 
282 static inline void __list_cut_position(struct list_head *list,
283         struct list_head *head, struct list_head *entry) {
284     struct list_head *new_first = entry->next;
285     list->next = head->next;
286     list->next->prev = list;
287     list->prev = entry;
288     entry->next = list;
289     head->next = new_first;
290     new_first->prev = head;
291 }
292 
293 /**
294  * list_cut_position - cut a list into two
295  * @list: a new list to add all removed entries
296  * @head: a list with entries
297  * @entry: an entry within head, could be the head itself
298  *  and if so we won't cut the list
299  *
300  * This helper moves the initial part of @head, up to and
301  * including @entry, from @head to @list. You should
302  * pass on @entry an element you know is on @head. @list
303  * should be an empty list or a list you do not care about
304  * losing its data.
305  *
306  */
307 static inline void list_cut_position(struct list_head *list,
308         struct list_head *head, struct list_head *entry) {
309     if (list_empty(head)) {
310         return;
311     }
312     if (list_is_singular(head) &&
313             (head->next != entry && head != entry)) {
314         return;
315     }
316     if (entry == head) {
317         INIT_LIST_HEAD(list);
318     } else {
319         __list_cut_position(list, head, entry);
320     }
321 }
322 
323 static inline void __list_splice(const struct list_head *list,
324         struct list_head *prev,
325         struct list_head *next) {
326     struct list_head *first = list->next;
327     struct list_head *last = list->prev;
328 
329     first->prev = prev;
330     prev->next = first;
331 
332     last->next = next;
333     next->prev = last;
334 }
335 
336 /**
337  * list_splice - join two lists, this is designed for stacks
338  * @list: the new list to add.
339  * @head: the place to add it in the first list.
340  */
341 static inline void list_splice(const struct list_head *list,
342         struct list_head *head) {
343     if (!list_empty(list)) {
344         __list_splice(list, head, head->next);
345     }
346 }
347 
348 /**
349  * list_splice_tail - join two lists, each list being a queue
350  * @list: the new list to add.
351  * @head: the place to add it in the first list.
352  */
353 static inline void list_splice_tail(struct list_head *list,
354         struct list_head *head) {
355     if (!list_empty(list)) {
356         __list_splice(list, head->prev, head);
357     }
358 }
359 
360 /**
361  * list_splice_init - join two lists and reinitialise the emptied list.
362  * @list: the new list to add.
363  * @head: the place to add it in the first list.
364  *
365  * The list at @list is reinitialised
366  */
367 static inline void list_splice_init(struct list_head *list,
368         struct list_head *head) {
369     if (!list_empty(list)) {
370         __list_splice(list, head, head->next);
371         INIT_LIST_HEAD(list);
372     }
373 }
374 
375 /**
376  * list_splice_tail_init - join two lists and reinitialise the emptied list
377  * @list: the new list to add.
378  * @head: the place to add it in the first list.
379  *
380  * Each of the lists is a queue.
381  * The list at @list is reinitialised
382  */
383 static inline void list_splice_tail_init(struct list_head *list,
384         struct list_head *head) {
385     if (!list_empty(list)) {
386         __list_splice(list, head->prev, head);
387         INIT_LIST_HEAD(list);
388     }
389 }
390 
391 /**
392  * list_entry - get the struct for this entry
393  * @ptr:    the &struct list_head pointer.
394  * @type:   the type of the struct this is embedded in.
395  * @member: the name of the list_head within the struct.
396  */
397 #define list_entry(ptr, type, member) \
398     container_of(ptr, type, member)
399 
400 /**
401  * list_first_entry - get the first element from a list
402  * @ptr:    the list head to take the element from.
403  * @type:   the type of the struct this is embedded in.
404  * @member: the name of the list_head within the struct.
405  *
406  * Note, that list is expected to be not empty.
407  */
408 #define list_first_entry(ptr, type, member) \
409     list_entry((ptr)->next, type, member)
410 
411 /**
412  * list_last_entry - get the last element from a list
413  * @ptr:    the list head to take the element from.
414  * @type:   the type of the struct this is embedded in.
415  * @member: the name of the list_head within the struct.
416  *
417  * Note, that list is expected to be not empty.
418  */
419 #define list_last_entry(ptr, type, member) \
420     list_entry((ptr)->prev, type, member)
421 
422 /**
423  * list_first_entry_or_null - get the first element from a list
424  * @ptr:    the list head to take the element from.
425  * @type:   the type of the struct this is embedded in.
426  * @member: the name of the list_head within the struct.
427  *
428  * Note that if the list is empty, it returns NULL.
429  */
430 #define list_first_entry_or_null(ptr, type, member) \
431     (!list_empty(ptr) ? list_first_entry(ptr, type, member) : NULL)
432 
433 /**
434  * list_next_entry - get the next element in list
435  * @pos:    the type * to cursor
436  * @member: the name of the list_head within the struct.
437  */
438 #define list_next_entry(pos, member) \
439     list_entry((pos)->member.next, typeof(*(pos)), member)
440 
441 /**
442  * list_prev_entry - get the prev element in list
443  * @pos:    the type * to cursor
444  * @member: the name of the list_head within the struct.
445  */
446 #define list_prev_entry(pos, member) \
447     list_entry((pos)->member.prev, typeof(*(pos)), member)
448 
449 /**
450  * list_for_each    -   iterate over a list
451  * @pos:    the &struct list_head to use as a loop cursor.
452  * @head:   the head for your list.
453  */
454 #define list_for_each(pos, head) \
455     for (pos = (head)->next; pos != (head); pos = pos->next)
456 
457 /**
458  * list_for_each_prev   -   iterate over a list backwards
459  * @pos:    the &struct list_head to use as a loop cursor.
460  * @head:   the head for your list.
461  */
462 #define list_for_each_prev(pos, head) \
463     for (pos = (head)->prev; pos != (head); pos = pos->prev)
464 
465 /**
466  * list_for_each_safe - iterate over a list safe against removal of list entry
467  * @pos:    the &struct list_head to use as a loop cursor.
468  * @n:      another &struct list_head to use as temporary storage
469  * @head:   the head for your list.
470  */
471 #define list_for_each_safe(pos, n, head) \
472     for (pos = (head)->next, n = pos->next; pos != (head); \
473         pos = n, n = pos->next)
474 
475 /**
476  * list_for_each_prev_safe - iterate over a list backwards safe against removal of list entry
477  * @pos:    the &struct list_head to use as a loop cursor.
478  * @n:      another &struct list_head to use as temporary storage
479  * @head:   the head for your list.
480  */
481 #define list_for_each_prev_safe(pos, n, head) \
482     for (pos = (head)->prev, n = pos->prev; \
483          pos != (head); \
484          pos = n, n = pos->prev)
485 
486 /**
487  * list_for_each_entry  -   iterate over list of given type
488  * @pos:    the type * to use as a loop cursor.
489  * @head:   the head for your list.
490  * @member: the name of the list_head within the struct.
491  */
492 #define list_for_each_entry(pos, head, member)                \
493     for (pos = list_first_entry(head, typeof(*pos), member);    \
494          &pos->member != (head);                    \
495          pos = list_next_entry(pos, member))
496 
497 /**
498  * list_for_each_entry_reverse - iterate backwards over list of given type.
499  * @pos:    the type * to use as a loop cursor.
500  * @head:   the head for your list.
501  * @member: the name of the list_head within the struct.
502  */
503 #define list_for_each_entry_reverse(pos, head, member)            \
504     for (pos = list_last_entry(head, typeof(*pos), member);        \
505          &pos->member != (head);                    \
506          pos = list_prev_entry(pos, member))
507 
508 /**
509  * list_prepare_entry - prepare a pos entry for use in list_for_each_entry_continue()
510  * @pos:    the type * to use as a start point
511  * @head:   the head of the list
512  * @member: the name of the list_head within the struct.
513  *
514  * Prepares a pos entry for use as a start point in list_for_each_entry_continue().
515  */
516 #define list_prepare_entry(pos, head, member) \
517     ((pos) ? : list_entry(head, typeof(*pos), member))
518 
519 /**
520  * list_for_each_entry_continue - continue iteration over list of given type
521  * @pos:    the type * to use as a loop cursor.
522  * @head:   the head for your list.
523  * @member: the name of the list_head within the struct.
524  *
525  * Continue to iterate over list of given type, continuing after
526  * the current position.
527  */
528 #define list_for_each_entry_continue(pos, head, member)        \
529     for (pos = list_next_entry(pos, member);            \
530          &pos->member != (head);                    \
531          pos = list_next_entry(pos, member))
532 
533 /**
534  * list_for_each_entry_continue_reverse - iterate backwards from the given point
535  * @pos:    the type * to use as a loop cursor.
536  * @head:   the head for your list.
537  * @member: the name of the list_head within the struct.
538  *
539  * Start to iterate over list of given type backwards, continuing after
540  * the current position.
541  */
542 #define list_for_each_entry_continue_reverse(pos, head, member)        \
543     for (pos = list_prev_entry(pos, member);            \
544          &pos->member != (head);                    \
545          pos = list_prev_entry(pos, member))
546 
547 /**
548  * list_for_each_entry_from - iterate over list of given type from the current point
549  * @pos:    the type * to use as a loop cursor.
550  * @head:   the head for your list.
551  * @member: the name of the list_head within the struct.
552  *
553  * Iterate over list of given type, continuing from current position.
554  */
555 #define list_for_each_entry_from(pos, head, member)            \
556     for (; &pos->member != (head);                    \
557          pos = list_next_entry(pos, member))
558 
559 /**
560  * list_for_each_entry_safe - iterate over list of given type safe against removal of list entry
561  * @pos:    the type * to use as a loop cursor.
562  * @n:      another type * to use as temporary storage
563  * @head:   the head for your list.
564  * @member: the name of the list_head within the struct.
565  */
566 #define list_for_each_entry_safe(pos, n, head, member)            \
567     for (pos = list_first_entry(head, typeof(*pos), member),    \
568         n = list_next_entry(pos, member);            \
569          &pos->member != (head);                    \
570          pos = n, n = list_next_entry(n, member))
571 
572 /**
573  * list_for_each_entry_safe_continue - continue list iteration safe against removal
574  * @pos:    the type * to use as a loop cursor.
575  * @n:      another type * to use as temporary storage
576  * @head:   the head for your list.
577  * @member: the name of the list_head within the struct.
578  *
579  * Iterate over list of given type, continuing after current point,
580  * safe against removal of list entry.
581  */
582 #define list_for_each_entry_safe_continue(pos, n, head, member)        \
583     for (pos = list_next_entry(pos, member),                \
584         n = list_next_entry(pos, member);                \
585          &pos->member != (head);                        \
586          pos = n, n = list_next_entry(n, member))
587 
588 /**
589  * list_for_each_entry_safe_from - iterate over list from current point safe against removal
590  * @pos:    the type * to use as a loop cursor.
591  * @n:      another type * to use as temporary storage
592  * @head:   the head for your list.
593  * @member: the name of the list_head within the struct.
594  *
595  * Iterate over list of given type from current point, safe against
596  * removal of list entry.
597  */
598 #define list_for_each_entry_safe_from(pos, n, head, member)            \
599     for (n = list_next_entry(pos, member);                    \
600          &pos->member != (head);                        \
601          pos = n, n = list_next_entry(n, member))
602 
603 /**
604  * list_for_each_entry_safe_reverse - iterate backwards over list safe against removal
605  * @pos:    the type * to use as a loop cursor.
606  * @n:      another type * to use as temporary storage
607  * @head:   the head for your list.
608  * @member: the name of the list_head within the struct.
609  *
610  * Iterate backwards over list of given type, safe against removal
611  * of list entry.
612  */
613 #define list_for_each_entry_safe_reverse(pos, n, head, member)        \
614     for (pos = list_last_entry(head, typeof(*pos), member),        \
615         n = list_prev_entry(pos, member);            \
616          &pos->member != (head);                    \
617          pos = n, n = list_prev_entry(n, member))
618 
619 /**
620  * list_safe_reset_next - reset a stale list_for_each_entry_safe loop
621  * @pos:    the loop cursor used in the list_for_each_entry_safe loop
622  * @n:      temporary storage used in list_for_each_entry_safe
623  * @member: the name of the list_head within the struct.
624  *
625  * list_safe_reset_next is not safe to use in general if the list may be
626  * modified concurrently (eg. the lock is dropped in the loop body). An
627  * exception to this is if the cursor element (pos) is pinned in the list,
628  * and list_safe_reset_next is called after re-taking the lock and before
629  * completing the current iteration of the loop body.
630  */
631 #define list_safe_reset_next(pos, n, member)                \
632     n = list_next_entry(pos, member)
633 
634 /*
635  * Double linked lists with a single pointer list head.
636  * Mostly useful for hash tables where the two pointer list head is
637  * too wasteful.
638  * You lose the ability to access the tail in O(1).
639  */
640 
641 #define HLIST_HEAD_INIT { .first = NULL }
642 #define HLIST_HEAD(name) struct hlist_head name = {  .first = NULL }
643 #define INIT_HLIST_HEAD(ptr) ((ptr)->first = NULL)
644 
645 static inline void INIT_HLIST_NODE(struct hlist_node *h) {
646     h->next = NULL;
647     h->pprev = NULL;
648 }
649 
650 static inline int hlist_unhashed(const struct hlist_node *h) {
651     return !h->pprev;
652 }
653 
654 static inline int hlist_empty(const struct hlist_head *h) {
655     return !h->first;
656 }
657 
658 static inline void __hlist_del(struct hlist_node *n) {
659     struct hlist_node *next = n->next;
660     struct hlist_node **pprev = n->pprev;
661     *pprev = next;
662     if (next)
663         next->pprev = pprev;
664 }
665 
666 static inline void hlist_del(struct hlist_node *n) {
667     __hlist_del(n);
668     n->next = LIST_POISON1;
669     n->pprev = LIST_POISON2;
670 }
671 
672 static inline void hlist_del_init(struct hlist_node *n) {
673     if (!hlist_unhashed(n)) {
674         __hlist_del(n);
675         INIT_HLIST_NODE(n);
676     }
677 }
678 
679 static inline void hlist_add_head(struct hlist_node *n, struct hlist_head *h) {
680     struct hlist_node *first = h->first;
681     n->next = first;
682     if (first) {
683         first->pprev = &n->next;
684     }
685     h->first = n;
686     n->pprev = &h->first;
687 }
688 
689 /* next must be != NULL */
690 static inline void hlist_add_before(struct hlist_node *n,
691         struct hlist_node *next) {
692     n->pprev = next->pprev;
693     n->next = next;
694     next->pprev = &n->next;
695     *(n->pprev) = n;
696 }
697 
698 static inline void hlist_add_behind(struct hlist_node *n,
699         struct hlist_node *prev) {
700     n->next = prev->next;
701     prev->next = n;
702     n->pprev = &prev->next;
703 
704     if (n->next) {
705         n->next->pprev = &n->next;
706     }
707 }
708 
709 /* after that we'll appear to be on some hlist and hlist_del will work */
710 static inline void hlist_add_fake(struct hlist_node *n) {
711     n->pprev = &n->next;
712 }
713 
714 /*
715  * Move a list from one list head to another. Fixup the pprev
716  * reference of the first entry if it exists.
717  */
718 static inline void hlist_move_list(struct hlist_head *old,
719         struct hlist_head *new) {
720     new->first = old->first;
721     if (new->first) {
722         new->first->pprev = &new->first;
723     }
724     old->first = NULL;
725 }
726 
727 #define hlist_entry(ptr, type, member) container_of(ptr,type,member)
728 
729 #define hlist_for_each(pos, head) \
730     for (pos = (head)->first; pos ; pos = pos->next)
731 
732 #define hlist_for_each_safe(pos, n, head) \
733     for (pos = (head)->first; pos && ({ n = pos->next; 1; }); \
734          pos = n)
735 
736 #define hlist_entry_safe(ptr, type, member) \
737     ({ typeof(ptr) ____ptr = (ptr); \
738        ____ptr ? hlist_entry(____ptr, type, member) : NULL; \
739     })
740 
741 /**
742  * hlist_for_each_entry - iterate over list of given type
743  * @pos:    the type * to use as a loop cursor.
744  * @head:   the head for your list.
745  * @member: the name of the hlist_node within the struct.
746  */
747 #define hlist_for_each_entry(pos, head, member)                \
748     for (pos = hlist_entry_safe((head)->first, typeof(*(pos)), member);\
749          pos;                            \
750          pos = hlist_entry_safe((pos)->member.next, typeof(*(pos)), member))
751 
752 /**
753  * hlist_for_each_entry_continue - iterate over a hlist continuing after current point
754  * @pos:    the type * to use as a loop cursor.
755  * @member: the name of the hlist_node within the struct.
756  */
757 #define hlist_for_each_entry_continue(pos, member)            \
758     for (pos = hlist_entry_safe((pos)->member.next, typeof(*(pos)), member);\
759          pos;                            \
760          pos = hlist_entry_safe((pos)->member.next, typeof(*(pos)), member))
761 
762 /**
763  * hlist_for_each_entry_from - iterate over a hlist continuing from current point
764  * @pos:    the type * to use as a loop cursor.
765  * @member: the name of the hlist_node within the struct.
766  */
767 #define hlist_for_each_entry_from(pos, member)                \
768     for (; pos;                            \
769          pos = hlist_entry_safe((pos)->member.next, typeof(*(pos)), member))
770 
771 /**
772  * hlist_for_each_entry_safe - iterate over list of given type safe against removal of list entry
773  * @pos:    the type * to use as a loop cursor.
774  * @n:      another &struct hlist_node to use as temporary storage
775  * @head:   the head for your list.
776  * @member: the name of the hlist_node within the struct.
777  */
778 #define hlist_for_each_entry_safe(pos, n, head, member)        \
779     for (pos = hlist_entry_safe((head)->first, typeof(*pos), member);\
780          pos && ({ n = pos->member.next; 1; });            \
781          pos = hlist_entry_safe(n, typeof(*pos), member))
782 
783 #endif // LIST_H_

这个代码不是我写的,就是用别人现成的,具体:https://www.cnblogs.com/muahao/p/8109733.html博主的代码

测试代码:testLi.c

 1 #include <stdio.h>                                                                                    
 2 #include <stdlib.h>
 3 #include <stdbool.h>
 4 #include "list.h"
 5 
 6 #define MaxSize 10
 7 
 8 typedef struct _stu{
 9     int id;
10     bool sex;
11     char name[MaxSize];
12 
13     struct list_head list;
14 } Student;
15 
16 void creatNumStudent(Student *mylist){
17     int num = 0;
18     printf("please input you want create the number, not zero: ");
19     scanf("%d", &num);
20 
21     for(int i=num; i!=0; --i){
22         Student *tmp= (Student *)malloc(sizeof(Student));
23         printf("enter id sex and name:");
24         scanf("%d %d %s", &tmp->id, &tmp->sex, &tmp->name);
25         list_add(&(tmp->list), &(mylist->list));
26     }
27     printf("\n");
28 
29     return;
30 }
31 
32 int main(int argc, char **argv){
33 
34     Student *tmp;
35     struct list_head *pos, *q;
36     //unsigned int i;
37 
38     Student mylist;
39     INIT_LIST_HEAD(&mylist.list);
40 
41     creatNumStudent(&mylist);
42 
43     printf("traversing the list using list_for_each_entry()\n");
44     list_for_each_entry(tmp, &mylist.list, list)
45         if(tmp->sex){
46             printf("id= %d sex= %s name= %s\n", tmp->id, "female", tmp->name);
47         }else{
48             printf("id= %d sex= %s name= %s\n", tmp->id, "male", tmp->name);
49         }
50     printf("\n");
51 
52     printf("deleting the list using list_for_each_safe()\n");
53     list_for_each_safe(pos, q, &mylist.list){
54         tmp= list_entry(pos, Student, list);
55 
56     if(tmp->sex){
57         printf("id= %d sex= %s name= %s\n", tmp->id, "female", tmp->name);
58     }else{
59         printf("id= %d sex= %s name= %s\n", tmp->id, "male", tmp->name);
60     }
61         list_del(pos);
62         free(tmp);
63     }
64 
65     return 0;
66 }

测试结果:

please input you want create the number, not zero: 3   
enter id sex and name:101 0 zhangsan
enter id sex and name:102 1 lisi
enter id sex and name:103 0 wangwu

traversing the list using list_for_each_entry()
id= 103 sex= male name= wangwu
id= 102 sex= female name= lisi
id= 101 sex= male name= zhangsan

deleting the list using list_for_each_safe()
id= 103 sex= male name= wangwu
id= 102 sex= female name= lisi
id= 101 sex= male name= zhangsan

这个也是一个链表的实现,但是几乎没有实现任何的函数就实现了一个链表,主要是使用了linux的内核链表,无需自己实现,直接使用即可:由于这里的list.h文件不是自己改造的,所以目前不多说明。以下针对使用linux你和的链表使用说明如下:

1、list.h文件中几乎实现了链表的所有操作,无论是你能想到的,或者是想不到的,安全的不安全都有了,几乎不需要添加,根据个人需求,可以把不需要的删除即可

2、头文件中仅仅是实现了纯粹的链表,没有关联到数据,什么意思,也就是说改变数据的结构和类型,不影响链表的使用方法。

3、内核链表打破了普通链表中的链表和数据结合很深的缺点,大大增强了链表的使用的灵活性,一般链表的实现中数据中有链表,链表中有数据,相互包含,相互引用;但是内核链表中数据和链表是经纬分明的,可以自由组合,随时能够分离的,大大提高了链表和数据使用的灵活性,增强了链表的使用的范围,缺点就是不太容易理解。

改变数据类型:测试函数为:

 1 #include <stdio.h>
 2 #include <stdlib.h>
 3 #include <stdbool.h>
 4 #include <string.h>
 5 #include "list.h"
 6 
 7 #define MaxSize 10
 8 
 9 typedef struct _doc{
10     int id;
11     bool sex;
12     char name[MaxSize];
13     char typed[MaxSize];
14 
15     struct list_head list;
16 } Doctor;
17 
18 void creatNumStudent(Doctor *mylist){
19     int num = 0;
20     printf("please input you want create the number, not zero: ");
21     scanf("%d", &num);
22 
23     for(int i=num; i!=0; --i){
24         Doctor *tmp= (Doctor *)malloc(sizeof(Doctor));
25         printf("enter id sex and name  and typed:");
26         scanf("%d %d %s %s", &tmp->id, &tmp->sex, &tmp->name, &tmp->typed);
27         list_add(&(tmp->list), &(mylist->list));
28     }
29     printf("\n");
30 
31     return;
32 }
33 
34 void outputInfo(bool sex, Doctor *tmp){
35     char str[MaxSize];
36 
37     if(sex){
38         strcpy(str, "female");
39     }else{
40         strcpy(str, "male");
41     }
42 
43     printf("id= %d sex= %s name= %s  typed= %s\n", tmp->id, str, tmp->name, tmp->typed);
44 
45     return;
46 }
47 
48 int main(int argc, char **argv){
49 
50     Doctor *tmp;
51     struct list_head *pos, *q;
52     //unsigned int i;
53 
54     Doctor mylist;
55     INIT_LIST_HEAD(&mylist.list);
56 
57     creatNumStudent(&mylist);
58 
59     printf("traversing the list using list_for_each_entry()\n");
60     list_for_each_entry(tmp, &mylist.list, list){
61         outputInfo(tmp->sex, tmp);
62     }
63     printf("\n");                                                                                                
64 
65     printf("deleting the list using list_for_each_safe()\n");
66     list_for_each_safe(pos, q, &mylist.list){
67         tmp= list_entry(pos, Doctor, list);
68 
69         outputInfo(tmp->sex, tmp);
70 
71         list_del(pos);
72         free(tmp);
73     }
74 
75     return 0;
76 }

测试结果为:

please input you want create the number, not zero: 4
enter id sex and name  and typed:101 0 zhangsan tooth
enter id sex and name  and typed:102 1 lisi nose
enter id sex and name  and typed:103 0 wangwu eye
enter id sex and name  and typed:104 1 zhaoliu mouth

traversing the list using list_for_each_entry()
id= 104 sex= female name= zhaoliu  typed= mouth
id= 103 sex= male name= wangwu  typed= eye
id= 102 sex= female name= lisi  typed= nose
id= 101 sex= male name= zhangsan  typed= tooth

deleting the list using list_for_each_safe()
id= 104 sex= female name= zhaoliu  typed= mouth
id= 103 sex= male name= wangwu  typed= eye
id= 102 sex= female name= lisi  typed= nose
id= 101 sex= male name= zhangsan  typed= tooth

说明:1、很明显看到,更换了数据的类型,由Student变成了Doctor,成员增加了一个,

      2、优化了性别判别的方法,代码稍微重构一下

   3、list.h中的链表的实现丝毫没有任何的改变,就让链表重新实现了新的数据的应用,是不是觉得很神奇,很不可思议,这就对了,普通人怎么能和linux的内核开发大神想的一样的呢

      4、还有一个要注意的是,这里其实也实现了数据和链表的互相隔离,从而能达到使用更加灵活的目的了。

posted @ 2021-08-20 21:28  叕叒双又  阅读(195)  评论(0编辑  收藏  举报