原题:

双向链表中,需要三个基本数据,一个携带具体数据,一个携带指向上一环节的prev指针,一个携带指向下一环节的next指针。请改写双向链表,仅用一个指针np实现双向链表的功能。定义np为next XOR prev,请根据表头提供的信息,为双向链表编写插入函数、删除函数和查找函数,并在O(1)时间内实现链表的翻转。

分析:

问题的关键,在于怎样利用prev指针和next指针的异或结果,来获得上一节点或下一节点的地址值。也就是说,如何利用异或来算出具体的prev及next值。我们注意到两点:

1、A XOR B = C  =>  C XOR B = A AND C XOR A = B。A、B两个元素的异或结果C,同A异或后得B,同B异或后得A。也就是说,如果我们能获取到next或者prev其中的一个值,我们就能根据np计算出另一个值。

2、双向链表中,头结点的prev值为0,尾结点的next值为0,因此,可以根据这一点,计算出全部的结点prev及next地址。依据第一点,表头信息需要提供头结点的地址。

实现:

C语言中,指针变量中存储的数值不能直接参与异或运算,需要转换为int型变量参与运算,再转换为指针来指定特定的地址。

根据前述分析,定义两个结构体如下:

typedef struct Node

{

int num;

int np;

}Node;

作为结点的结构体,包含结点必须的两个数据:类型为int的num数据单元,和类型为int的np指针,其中np指针中包含了next和prev两个结点的地址。

typedef struct double_train

{

Node *head;

int length;

}double_train;

作为双向链表的表头,提供了链表的大小、头结点两个数据,以供他人操作该链表。

插入函数void Insert(double_train *L , int value):

该函数接受两个参数——链表L和数据value,该函数的功能为将数据value插入表头。具体代码如下:

void

Insert(double_train *L , int value)

{

Node *p = (Node *)malloc(sizeof(Node));

int  temp2;

int  temp1;

p->num = value;

temp2 = (int) L->head;

temp2 = 0 ^ temp2;

p->np = temp2;

//插入一个新节点P,设置prev为0,next为L->head,并表示为np。

if(L->head != NULL)

{

temp1 = (int) p;

temp2 = L->head->np;

L->head->np = temp1 ^ temp2;

}

//若链表非空,则更新原先头结点的np值。

L->head = p;

L->length ++;

}

展示函数 void View(double_train *L):

该函数接受一个参数——链表L,该函数的功能为列出链表L中的所有数据单元。具体代码如下:

void

View(double_train *L)

{

int temp1,Last = 0;

Node *Next;

Node *top = L->head;

int count = L->length;

while(count != 0)        //从头结点开始,根据每个节点的np值遍历链表

{

temp1 = L->head->np ^ Last;

Next = (Node *) temp1;

Last = (int) L->head;

L->head = Next;

count --;

}

L->head = top;

}

删除函数和查找函数代码从略。下面是主函数测试代码,在windows gcc下编译通过并输出正确结果:

int

main(void)             //programmed by wmydx

{

double_train test;

double_train *L = &test;

L->length = 0;

L->head = NULL;  //C并非面向对象,因此不能用构造函数初始化

for(int i = 0;i < 5;i++)

{

Insert(L,i);

}

View(L);

system("pause");

}

那么如何实现在O(1)的时间内对链表进行翻转呢?只需要在链表结构体中加一个记录尾结点 Node *tail的代码,然后exchange tail->np with head->np即可完成。

【原创】《算法导论》链表一章带星习题试解——附C语言实现的更多相关文章

  1. 《算法导论》第二章demo代码实现(Java版)

    <算法导论>第二章demo代码实现(Java版) 前言 表示晚上心里有些不宁静,所以就写一篇博客,来缓缓.囧 拜读<算法导论>这样的神作,当然要做一些练习啦.除了练习题与思考题 ...

  2. 算法导论 第六章 堆排序(python)

    6.1堆 卫星数据:一个带排序的的数通常是有一个称为记录的数据集组成的,每一个记录有一个关键字key,记录的其他数据称为卫星数据. 原地排序:在排序输入数组时,只有常数个元素被存放到数组以外的空间中去 ...

  3. 算法导论 第十三章 红黑树(python)-1插入

    红黑树是上一章二叉搜索树的改进,实现一种平衡 ,保证不会出现二叉树变链表的情况,基本动态集合操作的时间复杂度为O(lgn) 实际用途:c++stl中的set,map是用他实现的 红黑树的性质: 1.每 ...

  4. 算法导论 第三章 and 第四章

    第三章 渐进的基本O().... 常用函数 % 和  // 转换 斯特林近似公式 斐波那契数 第四章 分治策略:分解(递归)--解决(递归触底)--合并 求解递归式的3种方法: 1:代入法(替代法): ...

  5. 算法导论 第六章 思考题6-3 Young氏矩阵

    这题利用二叉堆维持堆性质的办法来维持Young氏矩阵的性质,题目提示中写得很清楚,不过确实容易转不过弯来. a,b两问很简单.直接看c小问: 按照Young氏矩阵的性质,最小值肯定在左上角取得,问题在 ...

  6. 算法导论 第六章 思考题 6-3 d叉堆

    d叉堆的实现相对于二叉堆变化不大,首先看它如何用数组表示. 考虑一个索引从1开始的数组,一个结点i最多可以有d个子结点,编号从id - (d - 2) 到 id + 1. 从而可以知道一个结点i的父结 ...

  7. 算法导论(第三版)习题Exercises4.3(第四章三节)算法导论的一个印刷错误

    本节系列证明都可见4.5节需要说明的有4.3-8,4.3-9两题 4.3-8(本题有误) T(n)=4T(n/2)+n2根据4.5理论,结果为Θ(n2lgn) 4.3-9 m = lgn T(2m) ...

  8. 算法导论 第七章 快速排序(python)

    用的最多的排序 平均性能:O(nlogn){随机化nlogn} 原地址排序 稳定性:不稳定 思想:分治 (切分左右) 学习方式:自己在纸上走一遍   def PARTITION(A,p,r): x = ...

  9. 算法导论 第六章 2 优先队列(python)

    优先队列:     物理结构: 顺序表(典型的是数组){python用到list}     逻辑结构:似完全二叉树 使用的特点是:动态的排序..排序的元素会增加,减少#和快速排序对比 快速一次排完 增 ...

随机推荐

  1. C++ 直方图匹配算法代码

    /*-------------------------------------------------------------------------*/ // 函数名称: histeq() // 传 ...

  2. Github干货系列:C++资源集合-

    Awesome CPP,这又是一个 Awesome XXX 系列的资源整理,由 fffaraz 发起和维护.内容包括:标准库.Web应用框架.人工智能.数据库.图片处理.机器学习.日志.代码分析等. ...

  3. [Android学习笔记]some tips

    集合合并去重: listA.removeAll(listB); listA.addAll(listB); android:singleLine="true"//单行显示 andro ...

  4. onkeypress事件.onkeydown事件.onkeyup事件

    onkeypress事件是在按键開始按的时候发生: onkeydown事件是在按键已经按下的时候发生: onkeyup事件是在按键松开(释放)的时候发生.

  5. python语言学习3 ——第一个python程序

    输入exit即退出,这样写的缺点是没有保存已经写的代码,下次需要重新写

  6. Hibernate实体对象继承策略

    Hibernate继承策略总共同拥有三种,一种是共用一张表:一种是每一个类一张表,表里面储存子类的信息和父类的信息:另一种是通过表连接的方式.每一个类都有一张表,可是子类相应的表仅仅保存自己的信息,父 ...

  7. 【C++】动态开辟二维数组

    二维数组在内存中的分配例如以下: C方式呈现: <span style="font-size:18px;"> #include <iostream> usi ...

  8. 服务器编程入门(10)TCP回射服务器实现 - 并发

    问题聚焦:     在前面我们大概浏览了一下服务器编程需要掌握的一些知识和技术,以及架构思想.        实践,才是检验真理的唯一标准..从这节起我们将在这些技术的基础上,一步步实现以及完善一个服 ...

  9. 将Eclipse包括第一3正方形jar包裹Project Export并产生能够执行jar

    于Project对,Export-Java-Runnable JAR file.需要注意的是一定要选择"Package required libraries into generated J ...

  10. Learning Cocos2d-x for WP8(3)——文字篇

    原文:Learning Cocos2d-x for WP8(3)--文字篇 C#兄弟篇Learning Cocos2d-x for XNA(3)——文字篇 文字,是人类文明的象征. 文字显示,可用字符 ...