c语言List头文件和应用

c语言List头文件和应用util_list.h#ifndef__UTIL_LIST__#define__UTIL_LIST__/*双链节点*/typedefstructlist_node{ list_node*prev; list_node*next;}LIST_NODE;/*单链节点*/typedefstructslist_node{ slist_node*ne

大家好,又见面了,我是你们的朋友全栈君。

util_list.h

#ifndef __UTIL_LIST__
#define __UTIL_LIST__


/*双链节点*/
typedef struct list_node
{
	struct list_node * prev;
	struct list_node * next;
} LIST_NODE;

/*单链节点*/
typedef struct slist_node
{
	struct slist_node * next;
} SLIST_NODE;


#define lst_offsetof( type, member )  ( unsigned int )( &((type*)0)->member )

/* 初始化双链节点*/
#define lst_init( list )       { (list)->next = (list)->prev = list; }

/* 判断双链是否为空,1:为空,0:不为空 */
#define lst_empty( list )      ( ( (list) == (list)->next ) ? 1 : 0 )

/* 在链表list头部增加节点 */
#define lst_add(list, node)    { (list)->next->prev = (node); (node)->next = (list)->next; (list)->next = (node); (node)->prev = (list); }

/* 在链表list尾部增加节点 */
#define lst_insert(list, node) { (list)->prev->next = (node); (node)->prev = (list)->prev; (list)->prev = (node); (node)->next = (list); }

/* 删除双链节点 */
#define lst_del(node)          { (node)->prev->next = (node)->next; (node)->next->prev = (node)->prev; }

/* 从双链节点得到实例节点 */
#define lst_entity(node, type, member) ((type*)((char *)(node) - lst_offsetof( type, member )))

/* 初始化双链节点*/
#define slst_init(list)              { (list)->next =  (list); }

/* 判断单链是否为空,1:为空,0:不为空 */
#define slst_empty(list)             (((list) == (list)->next ) ? 1 : 0 )

/* 在链表list头部增加节点 */
#define slst_add(list, node)         { (node)->next = (list)->next; (list)->next = (node); }

/* 删除单链节点,prev是node的前节点,由调用者保证 */
#define slst_del_prev( prev, node )  { (prev)->next = (node)->next; }

/* 删除单链节点 */
#define slst_del( node ) \
{ \
	slist_node * p    = (node)->next; \
	slist_node * prev = (node); \
    while (( node ) != p)\
		{ prev = p; p = p->next; }\
	slst_del_prev( prev, node );\
}

/* 从单链节点得到实例节点 */
#define slst_entity(node, type, member) ((type*)((char *)(node) - lst_offsetof(type, member)))

#endif  /* __UTIL_LIST__ */

实际应用

TestList2.cpp

// TestList2.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include "stdlib.h"
#include "util_list.h"

LIST_NODE   g_TestNodeList     = { 0 };
SLIST_NODE  g_slstTestNodeList = { 0 };

typedef struct test_node
{
	LIST_NODE  ltNeigbor;
	SLIST_NODE sltNeigbor;

	int test_number;
} TEST_NODE;


int _tmain(int argc, _TCHAR* argv[]) 
{
	lst_init(&g_TestNodeList);
	slst_init(&g_slstTestNodeList);

	TEST_NODE * pstTestNode1 = (TEST_NODE *)malloc(sizeof(TEST_NODE));
	pstTestNode1->test_number = 100;	
	lst_add( &g_TestNodeList, &(pstTestNode1->ltNeigbor));
	//lst_insert(&g_TestNodeList, &(pstTestNode1->ltNeigbor));
	slst_add(&g_slstTestNodeList, &(pstTestNode1->sltNeigbor));

	TEST_NODE * pstTestNode2 = (TEST_NODE *)malloc(sizeof(TEST_NODE));
	pstTestNode2->test_number = 101;
	//lst_add( &g_TestNodeList, &(pstTestNode2->ltNeigbor));
	lst_insert(&g_TestNodeList, &(pstTestNode2->ltNeigbor));
	slst_add(&g_slstTestNodeList, &(pstTestNode2->sltNeigbor));

	TEST_NODE * pstTestNode3 = (TEST_NODE *)malloc(sizeof(TEST_NODE));
	pstTestNode3->test_number = 102;
	lst_add(&g_TestNodeList, &(pstTestNode3->ltNeigbor));
	//lst_insert(&g_TestNodeList, &(pstTestNode3->ltNeigbor));
	slst_add(&g_slstTestNodeList, &(pstTestNode3->sltNeigbor));

	LIST_NODE * pHead = &g_TestNodeList;
	LIST_NODE * pNext = pHead->next;

	while ( pNext != pHead )
	{
		TEST_NODE * pstTestNode100 = lst_entity( pNext, TEST_NODE, ltNeigbor );	
		lst_del(&( pstTestNode100->ltNeigbor ));
		pNext = pNext->next;
	}

	if (lst_empty( &g_TestNodeList ))
	{
		printf( "test ok" );
	}

	SLIST_NODE * psltHead = &g_slstTestNodeList;
	SLIST_NODE * psltNext = psltHead->next;
	SLIST_NODE * psltPrev = psltHead;

	while ( psltNext != psltHead )
	{
		TEST_NODE * pstTestNode101 = slst_entity( psltNext, TEST_NODE, sltNeigbor);

		if ( pstTestNode101->test_number == 102 )
		{
			slst_del_prev(psltPrev, &(pstTestNode101->sltNeigbor));
		}
		else
		{
			psltPrev = psltNext;
		}

		psltNext = psltNext->next;
	}

	slst_del( &( pstTestNode1->sltNeigbor ));
	slst_del( &( pstTestNode2->sltNeigbor ));

	if (slst_empty( &g_slstTestNodeList ))
	{
		printf("test2 ok");
	}

	free( pstTestNode1 );
	free( pstTestNode2 );
	free( pstTestNode3 );
	return 0;
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/157946.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • pycharm 多行注释_注释代码快捷键是什么

    pycharm 多行注释_注释代码快捷键是什么多行注释选中代码后快捷键Ctrl+/单行注释选中代码或者光标停留在该行,然后使用快捷键Ctrl+/多行代码缩进选中代码后,快捷键Tab多行代码取消缩进选中代码后,快捷键shift+Tab…

  • 作用域插槽(作用域插槽传值)

    作用域插槽一:假设第一个场景,需要你写一个商品卡片组件,并通过循环去展示多个卡片,并且要求能响应每个卡片上的图片或者其他内容的点击事件而跳转到商品详情页,你会怎么写?我会使用如下的处理方式,首先将商品卡片写成一个组件Commodity.vue,而在CommodityList.vue中用一个v-for来处理商品卡片列表的展示。<commodityv-for=”(item,index)incommodities”@clickCommodity=”onCommodityClick”>&l

  • jrtplib介绍[通俗易懂]

    jrtplib介绍[通俗易懂]程序流程发送:获得接收端的IP地址和端口号创建RTP会话指定RTP数据接收端设置RTP会话默认参数发送流媒体数据接收:获得用户指定的端口号创建RTP会话设置接收模式接受RTP数据检索RTP数据源获取RTP数据报删除RTP数据报1.初始化I、在使用JRTPLIB进行实时流媒体数据传输之前,首先应该生成R…

  • Linux上传下载命令_linux常用的20个命令

    Linux上传下载命令_linux常用的20个命令1、从服务器上下载文件scpusername@servername:/path/filename/var/www/local_dir(本地目录)例如scproot@192.168.0.101:/var/www/test.txt把192.168.0.101上的/var/www/test.txt的文件下载到/var/www/local_dir(本地目录)2、上传本地文件到服务器scp/p…

  • 激光SLAM流程_激光打眼

    激光SLAM流程_激光打眼基于环境自然导航激光叉车:基于环境自然导航的激光导航叉车AGV中,机器人在运动过程中通过编码器结合IMU计算得到里程计信息,运用机器人的运动模型得到机器人的位姿初估计,然后通过机器人装载的激光传感器获取的激光数据结合观测模型(激光的扫描匹配)对机器人位姿进行精确修正,得到机器人的精确定位,最后在精确定位的基础上,将激光数据添加到栅格地图中,反复如此,机器人在环境中运动,最终完成整个场景地图的构建…

  • ICMP数据包分析_Wireshark数据包分析实战

    ICMP数据包分析_Wireshark数据包分析实战一.实验目的1.学习和掌握ICMP协议的基本作用和报文格式2.理解ICMP协议与IP协议的封装关系3.学习和掌握ICMP协议的应用和报文格式4.理解tracertoute工作过程二.实验拓扑三.实验工具GNS3和Wireshark抓包分析软件四.ICMP协议的封装格式(1)Type类型值,标识ICMP分组类型(2)Code代码值,标识ICMP分组类型的某一种具体分组(3)Checksum校验和,用于检验数据包是否完整或是否被修改(4)Identifier标识符,标识本进程

    2022年10月21日

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号