如何使用GLib的单向链表GSList

单向链表是一种基础的数据结构,也是一种简单而灵活的数据结构,本文讨论单向链表的基本概念及实现方法,并着重介绍使用GLib的GList实现单向链表的方法及步骤,本文给出了多个实际范例源代码,旨在帮助学习基于GLib编程的读者较快地掌握GSList的使用方法,本文程序在 ubuntu 20.04 下编译测试完成,gcc 版本号 9.4.0;本文适合初学者阅读。

1 单向链表及其实现

  • 在文章《使用GLib进行C语言编程的实例》中,简单介绍了 GLib,建议阅读本文前先阅读这篇文章;

  • 单向链表是一种基础的数据结构,‌它由一系列节点组成,‌每个节点都包含数据部分和指向下一个节点的指针;

  • 这种链表的特点是数据只能在一个方向上流动,‌即从头节点开始,‌通过每个节点的指针依次访问后续节点,‌直到链表的末尾;

  • 在单向链表中,‌头节点是链表的起始点,‌它存储了链表的第一个数据元素以及指向下一个节点的指针;

  • 随后的每个节点都存储了自己的数据和一个指向下一个节点的指针,‌这样形成了链表的结构;

  • 链表的最后一个节点,‌也就是尾节点,‌它的指向下一个节点的指针指向 NULL,‌表示链表的结束。‌

  • 单向链表的主要操作包括:‌

    1. 插入节点:‌可以在链表的头部、‌尾部或中间某个位置插入新的节点。‌
    2. 删除节点:‌可以删除链表中的任意节点,‌并通过调整指针来保持链表的完整性。‌
    3. 遍历链表:‌通过从头节点开始,‌依次访问每个节点,‌直到到达链表的末尾。‌
    4. 查找节点:‌根据特定的条件或值,‌在链表中查找节点。‌
  • 单向链表的优势在于其动态的内存分配和高效的插入与删除操作,‌特别是在链表中间或头部插入和删除节点时,‌只需调整指针,‌无需移动其他数据;

  • 然而,‌单向链表的访问效率较低,‌因为访问特定位置的元素需要从头节点开始遍历。‌

  • 总的来说,‌单向链表是一种简单而灵活的数据结构,‌适用于需要频繁插入和删除操作,‌但访问操作相对较少的场景。‌

  • 下面程序是一个简单的单向链表的 C 语言标准库实现,sllist-c.c(点击文件名下载源程序)

  • 编译:gcc -Wall -g sllist-c.c -o sllist-c

  • 运行:./sllist-c

  • 该程序实现了单向链表的插入、删除、遍历和查找;

  • 该程序首先建立一个单向链表,并在链表中加入 4 个节点,数据分别为:1、2、3、5,然后显示整个链表;

  • 在第 2 个节点(数据为 3,索引号为 2)的后面插入节点,数据为 4,然后显示整个链表;

  • 将第 2 个节点(数据为 3,索引号为 2)删除,然后显示整个链表;

  • 最后释放整个链表;

  • 运行截图:

    screenshot of sllist-c

2 GLib 中单向链表结构 GSList

  • GLib API version 2.0 手册 (点击查看手册)
  • GLib API 手册中 GSList 部分 (点击查看手册)
  • 在 GLib 中,‌单向链表是通过GSList结构体实现的。‌GSList是一个简单的单向链表结构,‌用于存储各种类型的数据;
  • GSList 定义如下:
    struct GSList {gpointer data;GSList *next;
    }
    
  • data 为单向链表的数据指针,可以指向任何类型或结构的数据;
  • next 为指向该单向链表下一个节点的指针;
  • GLib 为单向链表结构 GSList 的操作提供了大量的函数,本文仅就其中的一部分函数进行介绍;
  1. 添加、插入新节点
    • g_slist_append() 在单向链表的最后添加一个新节点;
      GSList *g_slist_append(GSList *list, gpointer data)
      
      • list - 指向单向链表的指针
      • data - 指向添加节点的数据
      • 返回指向单向链表的起始指针;
    • g_slist_prepend() 在单向链表的最前面添加一个新节点;
      GSList *g_slist_prepend(GSList *list, gpointer data)
      
      • list - 指向单向链表的指针
      • data - 指向添加节点的数据
      • 返回指向单向链表的指针,在单向链表的开头添加一个节点,单向链表的指针是肯定会变化的;
      • 返回该单向链表的起始指针;
    • g_slist_insert() 在单向链表的中间插入一个新节点;
      GSList *g_slist_insert(GSList *list, gpointer data, gint position)
      
      • list - 指向单向链表的指针
      • data - 指向添加节点的数据
      • position - 插入节点的位置,如果是负数或者超过了该单向链表的节点的数量,新节点将插到单向链表的最后;
      • 返回该单向链表的起始指针;
    • g_slist_insert_before() 在包含指定数据的节点之前插入一个新节点;
      GSList *g_slist_insert_before(GSList *slist, GSList *sibling, gpointer data)
      
      • slist - 指向单向链表的指针
      • data - 指向添加节点的数据
      • sibling - 指向一个节点的指针,将在这个节点前插入新节点
      • 返回该单向链表的起始指针;
  2. 删除节点
    • g_slist_remove_link() 从单向链表中删除一个节点,但并不释放该节点占用的内存
      GSList *g_slist_remove_link(GSList *list, GSList *link_)
      
      • list - 指向单向链表的指针;
      • link_ - 指向单向链表中一个节点的指针,该节点将被删除;
      • 返回该单向链表的起始指针;
      • 该函数并不释放被删除的节点内存,被删除的节点的 next 指针将指向 NULL,所以可以认为被删除的节点变成了一个只有一个节点的新的单向链表;
    • g_slist_delete_link() 从单向链表中删除一个节点,并释放该节点占用的内存;
      GSList *g_slist_delete_link(GSList *list, GSList *link_)
      
      • list - 指向单向链表的指针;
      • link_ - 指向单向链表中一个节点的指针,该节点将被删除;
      • 返回该单向链表的起始指针;
      • 该函数与 g_slist_remove_link() 的唯一区别是该函数在删除节点后释放了被删除节点占用的内存;
    • g_slist_remove() 从单向链表中删除指定数据的一个节点,如果链表中有指定数据的节点有多个,将只删除第一个;
      GSList *g_slist_remove(GSList *list, gconstpointer data)
      
      • list - 指向单向链表的指针
      • data - 指向要删除节点的数据
      • 返回该单向链表的起始指针;
    • g_slist_remove_all() 从单向链表中删除指定数据的所有节点;
      GSList *g_slist_remove_all(GSList *list, gconstpointer data)
      
      • list - 指向单向链表的指针
      • data - 指向要删除节点的数据
      • 返回该单向链表的起始指针;
  3. 遍历链表
    • g_slist_foreach() 遍历单向链表,每个节点都会调用一个指定函数;
      void g_slist_foreach(GSList *list, GFunc func, gpointer user_data)
      
      • list - 指向单向链表的指针
      • func - 一个指向函数的指针,遍历到单向链表的每个节点时,都会调用这个函数;
      • GFunc 的定义如下:
      void (* GFunc) (gpointer data, gpointer user_data)
      
      • GFunc 的定义表明,传递给 func 的参数有两个,一个是 data - 指向当前节点的节点数据指针,另一个就是指向自定义参数 user_data 的指针
      • user_data - 指针指向调用 func 时传递的用户参数;
  4. 查找节点
    • g_slist_find() 查找链表中包含给定数据的节点;
      GSList *g_slist_find(GSList *list, gconstpointer data)
      
      • list - 指向单向链表的指针
      • data - 指向要查找节点的数据
      • 返回在单向链表中找到的节点的指针,如果没有找到相应节点,返回 NULL;
    • g_slist_index() 获取包含给定数据的节点的位置(从 0 开始);
      gint g_slist_index(GSList *list, gconstpointer data)
      
      • list - 指向单向链表的指针;
      • data - 指向要查找节点的数据;
      • 返回数据为 data 的节点在单向链表中的位置(从 0 开始),如果没找到相应节点,则返回 -1;
    • g_slist_position() 获取给定节点在链表中的位置(从 0 开始);
      gint g_slist_position(GSList *list, GSList *llink)
      
      • list - 指向单向链表的指针;
      • llink - 指向单向链表中的一个节点的指针;
      • 返回 llink 指向的节点在单向链表中的位置(从 0 开始),如果没找到相应节点,则返回 -1;
  5. 释放链表
    • g_slist_free() 释放链表使用的所有内存,该函数不会释放节点中动态分配的内存;
      void g_slist_free(GSList *list)
      
      • list - 指向单向链表的指针;
      • 该函数仅释放 GSList 占用的内存,并不释放单向链表中各个节点动态申请的内存,如果链表中有动态申请内存,考虑使用 g_slist_free_full() 或手动释放内存;
    • g_slist_free_full() 释放链表使用的所有内存,并对每个节点的数据调用指定的销毁函数
      void g_slist_free_full(GSList *list, GDestroyNotify free_func)
      
      • list - 指向单向链表的指针;
      • free_func - 销毁函数,对单向链表中的每个节点数据将调用该函数,可用于释放节点中动态分配的内存;
      • GDestroyNotify 的定义如下:
      void (* GDestroyNotify) (gpointer data)
      
      • 所以在调用 free_func 时会将指向节点数据的指针传递给该函数;
  6. 其它
    • g_slist_length() 获取单向链表的长度;
      guint g_slist_length(GSList *list)
      
      • list - 指向单向链表的指针;
      • 返回单向链表中节点的数量。
    • g_slist_last() 获取单向链表的最后一个节点;
      GSList *g_slist_last(GSList *list)
      
      • list - 指向单向链表的指针;
      • 返回单向链表的最后一个节点的指针,如果单向链表没有节点,则返回 NULL;
    • g_slist_concat() 连接两个单向链表;
      GSList *g_slist_concat(GSList *list1, GSList *list2)
      
      • list1 - 指向第 1 个单向链表的指针;
      • list2 - 指向准备连接到第 1 个单向链表后面的单向链表的指针;
      • 返回连接好的单向链表的指针,

3 如何使用 GSList 实现单向链表

  • 文章的一开始有一个使用标准 C 语言函数库的单向链表的实例,使用 GLib 的 GSList 操作单向链表要容易得多;

  • 下面程序是使用 C 语言,基于 GLib 实现的单向链表,sllist-glib.c(点击文件名下载源程序)

  • 该程序实现的功能与文章开头的程序 sllist-c.c 完全一样,但程序看上去要简洁很多,我们不妨把源程序列在这里

    #include <stdio.h>
    #include <glib.h>void print_node(gpointer data, gpointer user_data) {printf("%d -> ", GPOINTER_TO_INT(data));
    }void print_list(GSList *list) {g_slist_foreach(list, &print_node, NULL);printf("NULL\n");
    }int main() {GSList *list = NULL;printf("Append 4 nodes, the data are 1, 2, 3, 5.\n");list = g_slist_append(list, GINT_TO_POINTER(1));list = g_slist_append(list, GINT_TO_POINTER(2));list = g_slist_append(list, GINT_TO_POINTER(3));list = g_slist_append(list, GINT_TO_POINTER(5));print_list(list);printf("Insert a new node after node with the data 3.\n");list = g_slist_insert(list, GINT_TO_POINTER(4), 3);print_list(list);printf("Remove node with the data 3.\n");list = g_slist_remove(list, GINT_TO_POINTER(3));print_list(list);// Free the listg_slist_free(list);return 0;
    }
    
  • 编译:

    gcc -Wall -g sllist-glib.c -o sllist-glib `pkg-config --cflags --libs glib-2.0`
    
  • 其中,pkg-config --cflags --libs glib-2.0 的含义在文章《使用GLib进行C语言编程的实例》中做过介绍;

  • 运行:./sllist-glib

  • 该程序实现了单向链表的插入、删除、遍历和查找;

  • print_list() 中使用 g_slist_foreach() 对链表进行遍历,对链表中的每个节点数据,将调用函数 print_node()

  • 运行截图:

    screenshot of sllist-glib

4 单向链表的应用场景

  • 单向链表是一种基础的数据结构,具有节点之间按顺序相连的特性,在特定场景下非常有用,以下是一些典型的应用场景:

    1. 动态数据集

      当数据量不确定且频繁增删时,单向链表比数组更适用,它可以方便地在任意位置插入或删除节点,而不需要像数组那样移动大量元素;

    2. 队列和栈的实现

      单向链表常用于实现队列(FIFO)和栈(LIFO),因为它支持高效的插入和删除操作,尤其在头部或尾部进行操作时性能更好;

    3. 浏览历史记录或撤销操作

      在一些应用程序中,如浏览器的历史记录,单向链表可以用来保存用户的浏览路径或操作步骤,方便逐步返回或撤销;

    4. 分配器管理内存块

      操作系统的内存管理器中,单向链表经常被用于管理空闲的内存块(free lists),通过链表可以快速地找到可用的内存块;

  • 由于单向链表的简单结构,它在上述场景下既灵活又高效,特别是当增删操作频繁时。

5 基于 GLib 的 GSList 实现的 FIFO 队列

  • FIFO(First Input First Output)队列,也就是先进先出队列,是一种简单的机制,操作一个 FIFO 队列需要队列的头指针和尾指针;

  • 当向 FIFO 队列中加入数据时,数据添加到队列的尾指针处,当从队列中取出数据时,要从队列的头指针处取;

  • FIFO 队列的重要参数是队列的最大长度,当队列中数据的数量达到队列的最大长度时,则不能再向队列中添加数据;

  • FIFO 队列的两个重要判断就是判断队列为空(队列中没有数据)或者队列已满(数据数量达到最大长度);

  • 源程序 queue-glib.c(点击文件名下载源程序) 基于 GLib 的 GSList 实现了一个简单的 FIFO 队列;

  • 该程序实现了 FIFO 队列的两个基本操作:入队操作和出队操作,基于 GLib 使得程序相当的简单;

    #include <stdio.h>
    #include <glib.h>#define QUEUE_MAX_LEN           10
    GSList *queue_head, *queue_tail;        // head and tail pointers of the queue
    guint32 queue_max_len;                  // Max. length of the queuevoid queue_init(int maxn) {queue_head = queue_tail = NULL;queue_max_len = maxn;
    }gboolean queue_put(gpointer data) {guint queue_len = g_slist_length(queue_head);           // length of the queueif (queue_len >= queue_max_len) {// the queue is fullreturn FALSE;} else {queue_head = g_slist_append(queue_head, data);      // append a node with data to the queuequeue_tail = g_slist_last(queue_head);              // get the pointer of last node}return TRUE;
    }gpointer queue_get() {guint queue_len = g_slist_length(queue_head);           // length of the queueif (queue_len == 0) {// the queue is emptyreturn NULL;}gpointer queue_data = queue_tail->data;                 // data pointer of the last nodequeue_head = g_slist_delete_link(queue_head, queue_tail);   // delete the last nodeif (queue_head == NULL) {queue_tail = queue_head;                            // the queue is empty} else {queue_tail = g_slist_last(queue_head);              // get the pointer of last node}return queue_data;
    }int main(int argc, char **argv) {guint64 len;if (argc >= 2) {len = g_ascii_strtoll(argv[1], NULL, 10);           // Convert string to intif (len <= 0 || len > (QUEUE_MAX_LEN * 10)) {len = QUEUE_MAX_LEN;}} else {len = QUEUE_MAX_LEN;}printf("Max. length of the queue is %ld.\n", len);queue_init(len);            // Initialize the queueguint16 i;// append some data to the queuefor (i = 0; i < (queue_max_len << 1); ++i) {if (queue_put(GINT_TO_POINTER(i + 1))) {printf("Put data %d into the queue.\n", i + 1);} else {printf("The queue is full.\n");break;}}// get some data from the queuefor (i = 0; i < (queue_max_len << 1); ++i) {gpointer queue_data = queue_get();if (queue_data != NULL) {printf("Get data %d from the queue.\n", GPOINTER_TO_INT(queue_data));} else {printf("The queue is empty.\n");break;}}return 0;
    }
    
  • 可以看出,用 GLib 实现的 FIFO 队列非常简洁;

  • 编译:

    gcc -Wall -g queue-glib.c -o queue-glib `pkg-config --cflags --libs glib-2.0`
    
  • 其中,pkg-config --cflags --libs glib-2.0 的含义在文章《使用GLib进行C语言编程的实例》中做过介绍;

  • 运行:./queue-glib 8

  • 运行截图:

    screenshot of queue-glib

  • 该程序并不完整,如果实际运用,至少要加一个互斥锁,以保证 FIFO 队列的线程安全;

  • 使用 GLib 的 GSList 实现的 FIFO 队列,其中的数据并不需要是相同的数据类型,因为队列中存储的数据的指针,这一点在某些应用场景下会带来一些方便,但也会增加开销,而且在数据使用完成后有可能需要释放额外申请的内存空间。


email: hengch@163.com

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/430439.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

uni-app快速入门

目录 一、什么是 uni-app二、快速创建 uni-app 项目1.创建 uni-app2.运行 uni-app 三、uni-app 相对传统 H5 的变化1.网络模型的变化2.文件类型变化3.文件内代码架构的变化4.外部文件引用方式变化5.组件/标签的变化6.js的变化&#xff08;1&#xff09;运行环境从浏览器变成v8引…

Leetcode—329. 矩阵中的最长递增路径【困难】

2024每日刷题&#xff08;165&#xff09; Leetcode—329. 矩阵中的最长递增路径 dfs dp实现代码 class Solution { public:int longestIncreasingPath(vector<vector<int>>& matrix) {// 9 9 4// 6 6 8// 2 1 1// 1 1 2// 2 2 1// 3 4 2int m …

fastadmin 根据选择数据来传参给selectpage输入框

文章目录 js代码php代码&#xff1a;完结 js代码 $(document).on(change,#table .bs-checkbox [type"checkbox"],function(){let url$(#chuancan).attr(data-url)urlurl.split(?)[0]let idsTable.api.selectedids(table)if(ids.length){let u_id[]ids.forEach(eleme…

CSS文档流以及脱离文档流的方法

文档流 文档流是文档中可显示对象在排列时占用的位置/空间。例如&#xff1a;块元素自上而下摆放&#xff0c;内联元素从左到右摆放。&#xff08;文档流中限制非常的多&#xff0c;导致很多页面效果无法实现)。 常见文档流限制 高低不齐&#xff0c;底边对齐 <head>&…

02【Matlab系统辨识】白噪声

1.白噪声与有色噪声 1.1 白噪声(white noise) 系统辨识中所用到的数据通常都含有噪声。从工程实际出发&#xff0c;这种噪声往往可以视为具有有理谱密度的平稳随机过程。白噪声是一种最简单的随机过程&#xff0c;是由一系列不相关的随机变量组成的理想化随机过程。白噪声的数…

构建数据分析模型,及时回传各系统监控监测数据进行分析反馈响应的智慧油站开源了。

AI视频监控平台简介 AI视频监控平台是一款功能强大且简单易用的实时算法视频监控系统。它的愿景是最底层打通各大芯片厂商相互间的壁垒&#xff0c;省去繁琐重复的适配流程&#xff0c;实现芯片、算法、应用的全流程组合&#xff0c;从而大大减少企业级应用约95%的开发成本。增…

【深度学习】03-神经网络2-1损失函数

在神经网络中&#xff0c;不同任务类型&#xff08;如多分类、二分类、回归&#xff09;需要使用不同的损失函数来衡量模型预测和真实值之间的差异。选择合适的损失函数对于模型的性能至关重要。 这里的是API 的注意⚠️&#xff0c;但是在真实的公式中&#xff0c;目标值一定是…

面向对象 vs 面向过程

Java 和 C 语言的区别&#xff1a;面向对象 vs 面向过程 在编程世界中&#xff0c;不同的编程语言承载着不同的编程范式。C 语言作为一门经典的面向过程编程语言&#xff0c;注重函数的调用和操作&#xff1b;而Java则是典型的面向对象编程语言&#xff0c;重视对象与类的设计…

科技云报到:以数据“价值三角”为擎,探索数据治理实践路径

科技云报到原创。 过去四十年&#xff0c;经济发展主要来自于土地、劳动力、农业技术、工业技术等要素的充分释放。面向数字经济时代&#xff0c;无论是大模型、自动驾驶还是具身智能、人形机器人&#xff0c;数据已然成为继土地、劳动、资本和技术之后的又一种战略资产和新型…

【机器学习】11——矩阵求导

机器学习11——矩阵求导 打公式不太好标注&#xff0c;全图警告&#xff01;&#xff01;&#xff01; 文章目录 机器学习11——矩阵求导1.1标量对向量1.2标量对矩阵2.1向量对标量2.2向量对向量2.3向量对矩阵 1.1标量对向量 1.2标量对矩阵 X是m*n的矩阵&#xff0c;不严谨&am…

在vue中嵌入vitepress,基于markdown文件生成静态网页从而嵌入社团周报系统的一些想法和思路

什么是vitepress vitepress是一种将markdown文件渲染成静态网页的技术 其使用仅需几行命令即可 //在根目录安装vitepress npm add -D vitepress //初始化vitepress&#xff0c;添加相关配置文件&#xff0c;选择主题&#xff0c;描述&#xff0c;框架等 npx vitepress init //…

隐匿发案:David律所代理艺术家Ina Tomecek的两张青蛙版权画维权

案件基本情况&#xff1a;起诉时间&#xff1a;2024-8-14案件号&#xff1a;2024-cv-07196原告&#xff1a;Ina Tomecek原告律所&#xff1a;Law Office of David Gulbransen起诉地&#xff1a;伊利诺伊州北部法院涉案商标/版权&#xff1a;原告品牌简介&#xff1a;Ina Tomece…

内核是如何接收网络包的

1、数据如何从网卡到网络协议栈 1.1内核收包的过程 1、数据帧从外部网络到达网卡 2、网卡把数据帧从自己的缓存DMA(拷贝到)和内核共有的RingBuffer上 3、网卡发出硬中断通知CPU 4、CPU响应硬中断&#xff0c;简单处理后发出软中断 5、k’softirqd线程处理软中断&#xff0c;调…

颍川陈氏——平民崛起的典范

园子说颍川 广州有一处老建筑“陈家祠”&#xff0c;豪华精美堪比皇宫&#xff0c;誉为“岭南建筑艺术明珠”、“新世纪羊城八景”之一&#xff0c;是全国文保单位&#xff0c;4A 级景区。主体建筑以中轴线三座厅堂为中心&#xff0c;由大小十九座单体建筑组成&#xff0c;占地…

Windows如何查看已缓存的DNS信息

Windows server 2016如何查看已缓存的DNS信息 在Windows server 2016系统下&#xff0c;如何查看已缓存的DNS信息呢? 1.打开“运行”&#xff0c;输入cmd&#xff0c;点击“确定” 2.在命令行界面输入ipconfig /displaydns&#xff0c;按回车即可查看已缓存的dns信息

在vue中:style 的几种使用方式

在日常开发中:style的使用也是比较常见的&#xff1a; 亲测有效 1.最通用的写法 <p :style"{fontFamily:arr.conFontFamily,color:arr.conFontColor,backgroundColor:arr.conBgColor}">{{con.title}}</p> 2.三元表达式 <a :style"{height:…

Android15之源码分支qpr、dp、beta、r1含义(二百三十二)

简介&#xff1a; CSDN博客专家、《Android系统多媒体进阶实战》一书作者 新书发布&#xff1a;《Android系统多媒体进阶实战》&#x1f680; 优质专栏&#xff1a; Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a; 多媒体系统工程师系列【…

Java面向对象——内部类(成员内部类、静态内部类、局部内部类、匿名内部类,完整详解附有代码+案例)

文章目录 内部类17.1概述17.2成员内部类17.2.1 获取成员内部类对象17.2.2 成员内部类内存图 17.3静态内部类17.4局部内部类17.5匿名内部类17.5.1概述 内部类 17.1概述 写在一个类里面的类叫内部类,即 在一个类的里面再定义一个类。 如&#xff0c;A类的里面的定义B类&#x…

江协科技STM32学习- P14 示例程序(定时器定时中断和定时器外部时钟)

&#x1f680;write in front&#x1f680; &#x1f50e;大家好&#xff0c;我是黄桃罐头&#xff0c;希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流 &#x1f381;欢迎各位→点赞&#x1f44d; 收藏⭐️ 留言&#x1f4dd;​…