c语言链表升级
之前的链表就是一个普通的带头的单向链表,我们不自觉的会发现这样的链表有缺陷,有关链表的删除新增查找跟链表的结构体内容耦合性太强
什么意思呢?
比如我们之前的链表的结构体
typedef struct _Teacher
{
int age;
struct _Teacher *next;
}Teacher;
我们有关链表所有的操作都跟这个结构体紧密的相连,如果此刻我们有另一个结构体,里面的字段都跟这个不一样,那么,我们可能还需要对这个新的结构体写一套链表操作?
相当于下面的图,呵呵,有点丑

那么我们的解决方案是什么呢?,我们能不能关于不同结构体的所有操作都有一套公共方法呢?
当然是可以的!
在这之前我们必须有一个概念,结构体中的第一个元素的地址就是代表结构体的地址。
我们设计一个单纯代表数据结构的结构体,这个结构体只有下个对象的地址的指针成员

typedef struct _tag_LinkListNode
{
void * next;
}LinkListNode;
比如此刻我们有一个教师结构体,那么我们只需要在结构体的第一个成员是上面的LinkListNode对象。
形成下面的数据结构:

typedef struct _Teacher
{
LinkListNode listNode;
int age;
char name[];
}Teacher;
那么,此刻我们只要在我们的对链表的操作将传过来的Teacher * 指针强转为LinkListNode *类型,查询出来的LinkListNode * 指针变量再强转为Teacher * 对象,从而对LinkListNode形成一套插入删除查询的api就可以了
以下是实现的代码:
接口:
#ifndef _MYLINKLIST_H_
#define _MYLINKLIST_H_ typedef void LinkList; typedef struct _tag_LinkListNode
{
struct _tag_LinkListNode* next;
}LinkListNode; LinkList* LinkList_Create(); void LinkList_Destroy(LinkList* list); void LinkList_Clear(LinkList* list); int LinkList_Length(LinkList* list); int LinkList_Insert(LinkList* list, LinkListNode* node, int pos); LinkListNode* LinkList_Get(LinkList* list, int pos); LinkListNode* LinkList_Delete(LinkList* list, int pos); #endif
实现:
#include "stdio.h"
#include "stdlib.h"
#include "linklist.h" typedef struct _tag_LinkList
{
//头节点
LinkListNode header;
int length;
}TLinkList; /************************************************************************/
/* 创建list并初始化一个头节点 */
/************************************************************************/
LinkList* LinkList_Create()
{
TLinkList *tlist = (TLinkList *)malloc(sizeof(TLinkList));
if (tlist == NULL)
{
return NULL;
}
tlist->length = ;
tlist->header.next = NULL;
return tlist;
} void LinkList_Destroy(LinkList* list)
{
if (list == NULL)
{
return;
}
free(list);
}
/************************************************************************/
/* 清空list链表 */
/************************************************************************/
void LinkList_Clear(LinkList* list)
{
if (list == NULL)
{
return;
}
TLinkList *tlinkList = (TLinkList *)list;
tlinkList->length = ;
tlinkList->header.next = NULL; } int LinkList_Length(LinkList* list)
{
if (list == NULL)
{
return ;
}
TLinkList *tlinkList = (TLinkList *)list; return tlinkList->length;
} int LinkList_Insert(LinkList* list, LinkListNode* node, int pos)
{
LinkListNode *pre, *cur;
int i;
if (list == NULL || node == NULL)
{
return -;
}
//校验下标
if (pos<)
{
return -;
}
TLinkList *tlinkList = (TLinkList *)list;
if (pos>tlinkList->length - )
{
pos = tlinkList->length;
}
pre = (LinkListNode *)list;//初始化指向头节点
cur = tlinkList->header.next;//初始化指向第一个节点,如果空链表指向空
for (i = ; i < pos; i++)
{
pre = cur;
cur = cur->next;//最终让当前指针指向要插入的位置
}
pre->next = node;
node->next = cur; tlinkList->length++;
return ;
} LinkListNode* LinkList_Get(LinkList* list, int pos)
{
LinkListNode *pre, *cur;
int i;
if (list == NULL)
{
return NULL;
}
TLinkList *tlinkList = (TLinkList *)list;
//校验下标
if (pos >= tlinkList->length || pos < )
{
return NULL;
}
cur = tlinkList->header.next;//初始化指向第一个节点,如果空链表指向空
for (i = ; i < pos; i++)
{
cur = cur->next;
} return cur;
} LinkListNode* LinkList_Delete(LinkList* list, int pos)
{
LinkListNode *pre, *cur;
int i;
if (list == NULL)
{
return NULL;
}
TLinkList *tlinkList = (TLinkList *)list;
//校验下标
if (pos >= tlinkList->length || pos < )
{
return NULL;
}
pre = (LinkListNode *)list;//初始化指向头节点
cur = tlinkList->header.next;//初始化指向第一个节点,如果空链表指向空
for (i = ; i < pos; i++)
{
pre = cur;
cur = cur->next;
}
pre->next = cur->next;
LinkListNode* curnode = cur;
tlinkList->length--;
return curnode;
}
测试代码
#include "stdio.h"
#include "stdlib.h"
#include "linklist.h" typedef struct _Teacher
{
LinkListNode listNode;
int age;
char name[];
}Teacher;
void main()
{
LinkList* linkList;
Teacher t1, t2, t3;
int len;
int i;
linkList = LinkList_Create();
t1.age = ;
t2.age = ;
t3.age = ;
LinkList_Insert(linkList, (LinkListNode*)&t1, );
LinkList_Insert(linkList, (LinkListNode*)&t2, );
LinkList_Insert(linkList, (LinkListNode*)&t3, ); len = LinkList_Length(linkList); for (i = ; i < len; i++)
{
Teacher * t = (Teacher *)LinkList_Get(linkList, i);
printf("cur teacher age=%d\n", t->age);
} LinkList_Delete(linkList, );
LinkList_Delete(linkList, );
len = LinkList_Length(linkList);
for (i = ; i < len; i++)
{
Teacher * t = (Teacher *)LinkList_Get(linkList, i);
printf("cur teacher age=%d\n", t->age);
}
system("pause"); }
接下来,如果我们有新的结构体,只要在测试代码那边做修改,而不需要动我们的核心代码了。
c语言链表升级的更多相关文章
- C语言 链表
原文:C语言 链表 最近在复习数据结构,想把数据结构里面涉及的都自己实现一下,完全是用C语言实现的. 自己编写的不是很好,大家可以参考,有错误希望帮忙指正,现在正处于编写阶段,一共将要实现19个功能. ...
- C语言链表操作模板(添加,删除,遍历,排序)
C语言链表操作模板,摘自郝斌的C语言视频教程,简单的修改成了纯C格式.当年照着视频学习的时候记录下来的,在使用的时候直接拿来修改修改修改能节约不少时间的. /********************* ...
- ZT C语言链表操作(新增单向链表的逆序建立)
这个不好懂,不如看 转贴:C语言链表基本操作http://www.cnblogs.com/jeanschen/p/3542668.html ZT 链表逆序http://www.cnblogs.com/ ...
- C语言链表结构体(学习笔记)
#include <stdio.h> #define LENTEST 100 // 采取逐步删除的方法求的素数 //先假设1-100都是素数,然后剔除2的倍数, //3的倍数,直到剔除所有 ...
- C语言链表实例--玩转链表
下图为最一简单链表的示意图: 第 0 个结点称为头结点,它存放有第一个结点的首地址,它没有数据,只是一个指针变量.以下的每个结点都分为两个域,一个是数据域,存放各种实际的数据,如学号 num,姓名 n ...
- c语言-链表VS数组
数组和链表的区别 数组是将元素在内存中连续存放,由于每个元素占用内存相同,可以通过下标迅速访问数组中任何元素.但是如果要在数组中增加一个元素,需要移动大量元素,在内存中空出一个元素的空间,然后将要 ...
- 再次复习数据结构:c语言链表的简单操作
最近呢,又要面临多次的数据结构与算法方面的试题了,而我呢,大概也重新温习c语言的基本要点快一个月了,主要是针对指针这货的角度在研究c语言,感觉又学到了不少. 现在c指针感觉知道点了,也就匆忙开展数据结 ...
- [数据结构]C语言链表实现
我学数据结构的时候也是感觉很困难,当我学完后我发现了之所以困难时因为我没有系统的进行学习,而且很多教授都只是注重数据结构思想,而忽略了代码方面,为此我写了这些博文给那些试图自学数据结构的朋友,希望你们 ...
- [C语言]链表实现贪吃蛇及部分模块优化
在继上篇[C语言]贪吃蛇_结构数组实现大半年后,链表实现的版本也终于出炉了.两篇隔了这么久除了是懒癌晚期的原因外,对整个游戏流程的改进,模块的精简也花了一些时间(都是借口). 优化模块的前沿链接: · ...
随机推荐
- Java虚拟机15:再谈四种引用状态
JVM的四种引用状态 在Java虚拟机5:Java垃圾回收(GC)机制详解一文中,有简单提到过JVM的四种引用状态,当时只是简单学习,知道有这么一个概念,对四种引用状态理解不深.这两天重看虚拟机这部分 ...
- php session_start()报错 解决办法
1.php.ini中的output_buffering=off 改成output_buffering=4096 2.php.ini中的session.save_path是否设置好了 3.php.ini ...
- LODOP之票据连续套打笔记<一>
之前项目中需要使用套打,费了半天劲,最后找到LODOP,整体感觉还是不错,简单,满足大多数web套打 这是我项目中需要打印的票据 该票据每张做多显示6条数据,数据超过6条的时候需要进行分页打印,当时做 ...
- 【Android Developers Training】 104. 接受地点更新
注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...
- CUDA零内存拷贝 疑问考证
今天思考了一下CUDA零内存拷贝的问题,感觉在即将设计的程序中会派上用场,于是就查了一下相关信息. 以下是一些有帮助的链接: cuda中的零拷贝用法--针对二维指针 cuda中的零拷贝用法--针对一维 ...
- 使用JDK自带的MessageDigest计算消息摘要
使用JDK自带的MessageDigest计算消息摘要 上代码 /** * 使用JDK自带MessageDigest */ public class MessageDigestUtils { /** ...
- bash远程代码执行漏洞
博客园新闻:http://news.cnblogs.com/n/504506/(如果以下有说错的地方请不吝指出,谢谢~) 详情可围观上面的链接.因为我们的服务器都是私有网环境,即使要修复也得等到下次安 ...
- JS弹出下载对话框以及实现常见文件类型的下载
写在前面 JS要实现下载功能,一般都是这么几个过程:生成下载的URL,动态创建一个A标签,并将其href指向生成的URL,然后触发A标签的单击事件,这样就会弹出下载对话框,从而实现了一个下载的功能. ...
- c#编程-线程同步
线程同步 上一篇介绍了如何开启线程,线程间相互传递参数,及线程中本地变量和全局共享变量区别. 本篇主要说明线程同步. 如果有多个线程同时访问共享数据的时候,就必须要用线程同步,防止共享数据被破坏.如果 ...
- ELK日志框架(2):log4net.ElasticSearch+ Kibana实现日志记录和显示
环境说明 1. windows server 2012 R2 64位 2. log4net.ElasticSearch 3. kibana-5.5.0-windows-x86.zip 架构说明 数据采 ...