专栏首页Pou光明单向链表的一点儿感悟

单向链表的一点儿感悟

最近一段时间学习了挺多的,数据结构看了一点点,略有感悟,和感兴趣的同志分享一下,欢迎大家不吝评论。

除了关于链表的一点感悟,还有最近了解到的工程中遇到的几个实际问题:

①libevent由于阻塞,将所在进程挂起

②使用线程池时由于线程属性没有设置为分离属性,造成内存泄漏

③Linux的共享内存与C++ STL中共享内存的比较

仅供大家参考。

一、链表的分类

学习前先分类,从大的方面来分,链表属于线性表;线性表从存储方式来分可分为顺序存储结构与链式存储结构——即链表。链表根据特点又可以再具体分为单向链表、循环链表和双向链表等。

二、链表的操作

那按照不同的分法简直太多了,20来个。。。这次简单介绍几个,其中重点介绍如何逆转一个链表。

贴程序前先熟悉数据类型:

typedef int ElemType;

typedef struct node{
 ElemType data;
 struct node *link;
}LNode, *LinkList;

1. 创建一个链表

LinkList Create(int n)
{
 int LinearTable[7] = {0,1,2,3,5,8,10};
 LinkList p, r, head = NULL;
 ElemType tmp;
 int i;

 for(i=0; i<n; i++)
 {
  tmp = LinearTable[i];
  p = (LinkList)malloc(sizeof(LNode));
  p->data = tmp;
  p->link = NULL;

  if(NULL == head)
   head = p;
  else
   r->link = p;

  r = p;
//  printf("--%p \n",p);
 }

 return head;
}

把数组中的元素分别放到链表中节点的数据域,注意将表头存储返回。

2. 遍历一个链表

void Traverse(LinkList list)
{
 LinkList p = list;

 while (NULL != p)
 {
  printf(">> %d \n",p->data);
  printf("pointer %p \n",p);
  p = p->link;
//  printf(">> %p",p->link);  //Segmentation fault (core dumped)
 }
}

这个大家应该比较熟悉了。

3. 计算链表的长度

int Length(LinkList list)
{
 LinkList p = list;
 int length = 0;

 while (NULL != p)
 {
  length++;
  p = p->link;
 }

 return length;
}

这个也还好。

4. 查找元素所在地址

LinkList Find(LinkList list, const ElemType item)
{
 LinkList p = list;

 while (NULL != p && item != p->data)
  p = p->link;

 return p;
}

5. 在非空链表第一个节点前插入一个节点

/*************************************************
  名称:
  描述:    insert a item before the first node
         in a not empty linklist
  输入参数:
  输出参数:
  返回值:
*************************************************/
void InsertLinkOne(LinkList *list, const ElemType item)
{
 LinkList p = (LinkList)malloc(sizeof(LNode));

 p->data = item;
 p->link = *list;

 *list = p;
// list = &p;
}

注意如何修改头节点。

6. 在非空链表表尾插入一个节点

/*************************************************
  名称:
  描述:    insert a item after the tail node
         in a not empty linklist
  输入参数:
  输出参数:
  返回值:
*************************************************/
void InsertLinkTwo(LinkList list, const ElemType item)
{
 LinkList p, r;
 r = list;

 while (NULL != r->link)
  r = r->link;

 p = (LinkList)malloc(sizeof(LNode));
 p->data = item;
 p->link = NULL;

 r->link = p;
// list = &p;
}

这个使用到了遍历,因为链表不能随机访问节点,想下哪些操作还需要使用到遍历?

7. 逆转一个链表

void Invert(LinkList *list)
{
 LinkList oriList, newList, tmpList;
 oriList = *list;
 newList = NULL;

 while (NULL != oriList)
 {
  tmpList = newList;
  newList = oriList;
  oriList = oriList->link;
  newList->link = tmpList;
 }

 *list = newList;
}

设计的挺巧妙的,光看就看了好一会儿。

可以类比如何交换两个数的程序,需要使用一个中间变量来进行临时存储:

a, b, c,

c = a;

a = b;

b = c;

第一次执行:

通过tmpList节点来临时存储新链表的节点,

新的节点指向原来链表的头结点,

原来链表移动到下一个节点,

新链表节点的link链向新链表——

第二次执行:

此时tmpList节点存储的是新的链表的指针,此时有一个节点,

获取原来链表的第二个节点,

原来链表移动到下一个节点(功能不变),

新节点的link指向新的链表,此时新链表有两个节点了,且链表尾端是原来的链表的头结点。

多琢磨琢磨,还是挺有意思的。

链表什么样的操作需要用到遍历?

三、总结

拼尽全力去学习,学习,再学习。

上面那句是废话。

学习是要讲究方法的,

由于最近大量的学习一些东西,会迫使自己不断去梳理,去寻找知识点之间的关系,这个过程迫使你去“找规律”、“寻求联系”;相反写程序反而是最简单的,只不过把流程用编程语言表示出来。

我觉得一个知识学会的标志是:很久都不会忘,即使忘了也会记住一些“自己总结出来的易用的技巧”,那个知识点最后是变成了技巧。数学尤其如此,见到一个东西,脑海里立马想出4条路线,发现一个走不通立马换下一个,肯定会有走通的那一条。

本文分享自微信公众号 - Pou光明(pou0230),作者:PouG

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2020-07-14

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Qt ListView 配合Model 显示文件与删除文件

    表格、列表和树型窗口部件是 GUI 开发中经常会用到的窗口部件。这些窗口部件有两种不同的方式来获取数据。传统的方式是窗口部件本身包含用于存储数据的内置容器。这种...

    用户5908113
  • Qt ModelView教程——设置表头与可编辑Table

    点击上方蓝字可直接关注!方便下次阅读。如果对你有帮助,可以点个在看,让它可以帮助到更多老铁~

    用户5908113
  • Qt创建一个OpenGL窗口

    点击上方蓝字可直接关注!方便下次阅读。如果对你有帮助,可以点个在看,让它可以帮助到更多同志~

    用户5908113
  • 数据结构 | 每日一练(46)

    1.已知不带头结点的线性链表 list,链表中结点构造为(data、link),其中 data 为数据域,link 为指

    C语言入门到精通
  • [Redis] list底层的数据结构

    压缩列表(ziplist)是Redis为了节省内存而开发的,是由一系列特殊编码的连续内存块组成的顺序型数据结构,一个压缩列表可以包含任意多个节点(entry),...

    陶士涵
  • 关于HashMap的一些理解

    本文主要补充对HashMap的一些理解、分析。相信大家对HashMap都很熟悉,但是其中的一些细节上的设计、思想,往往会被大家忽略,这些都是构成HashMap的...

    Erwin
  • Yann LeCun:CNN已解决CIFAR-10,目标 ImageNet

    用户1737318
  • Python爬虫(九)_非结构化数据与结构化数据

    爬虫的一个重要步骤就是页面解析与数据提取。更多内容请参考:Python学习指南 页面解析与数据提取 实际上爬虫一共就四个主要步骤: 定(要知道你准备在哪个...

    用户1174963
  • 手把脚看看密码学No.72

    咳咳,不可否认,有时候总会出现,请证明你爸是你爸的这种事情,所以会有了身份认证的密码。但是这个密码不是你们想的那个登录密码,银行卡密码,QQ密码,IQ密码,更不...

    大蕉
  • 【每天一道编程系列-2018.2.2】(Ans)

    You are given two non-empty linked lists representing two non-negative integers...

    yesr

扫码关注云+社区

领取腾讯云代金券