采用模板类实现的好处是,不用拘泥于特定的数据类型。就像活字印刷术,制定好模板,就可以批量印刷,比手抄要强多少倍!

此处不具体介绍泛型编程,还是着重叙述链表的定义和相关操作。

 链表结构定义

定义单链表的结构可以有4方式。如代码所示。

本文采用的是第4种结构类型

/*************************************************************************
1、复合类:在Node类中定义友元的方式,使List类可以访问结点的私有成员
*************************************************************************/
class LinkNode
{
    friend class LinkList;
private:
    int data;
    LinkNode *next;
};

class LinkList
{
public:
    //单链表具体操作
private:
    LinkNode *head;
};

/*************************************************************************
2、嵌套类:在List内部定义Node类,但是Node的数据成员放在public部分,使List
和Node均可以直接访问Node的成员
*************************************************************************/
class LinkList
{
public:
    //单链表具体操作
private:
    class LinkNode
    {
    public:
        int data;
        LinkNode *next;
    };
    LinkNode *head;
};

/*************************************************************************
3、继承:在Node类中把成员定义为protected,然后让List继承Node类,这样就可以
访问Node类的成员了。
*************************************************************************/
class LinkNode
{
protected:
    int data;
    LinkNode *next;
};

class LinkList : public LinkNode
{
public:
    //单链表具体操作
private:
    LinkNode *head;
};

/*************************************************************************
4、直接用struct定义Node类,因为struct的成员默认为公有数据成员,所以可直接
访问(struct也可以指定保护类型)。
*************************************************************************/
struct LinkNode
{
    int data;
    LinkNode *next;
};

class LinkList
{
public:
    //单链表具体操作
private:
    LinkNode *head;
};

单链表的模板类定义

使用模板类需要注意的一点是template<class T>必须定义在同一个文件,否则编译器会无法识别。

如果在.h中声明类函数,但是在.cpp中定义函数具体实现, 会出错。所以,推荐的方式是直接在.h中定义。

/* 单链表的结点定义 */
template<class T>
struct LinkNode
{
    T data;
    LinkNode<T> *next;
    LinkNode(LinkNode<T> *ptr = NULL){next = ptr;}
    LinkNode(const T &item, LinkNode<T> *ptr = NULL)    
    //函数参数表中的形参允许有默认值,但是带默认值的参数需要放后面
    {
        next = ptr;
        data = item;
    }
};

/* 带头结点的单链表定义 */
template<class T>
class LinkList
{
public:
    //无参数的构造函数
    LinkList(){head = new LinkNode<T>;}
    //带参数的构造函数
    LinkList(const T &item){head = new LinkNode<T>(item);}
    //拷贝构造函数
    LinkList(LinkList<T> &List);
    //析构函数
    ~LinkList(){Clear();}
    //重载函数:赋值
    LinkList<T>& operator=(LinkList<T> &List);
    //链表清空
    void Clear();
    //获取链表长度
    int Length() const;
    //获取链表头结点
    LinkNode<T>* GetHead() const;
    //设置链表头结点
    void SetHead(LinkNode<T> *p);
    //查找数据的位置,返回第一个找到的满足该数值的结点指针
    LinkNode<T>* Find(T &item);
    //定位指定的位置,返回该位置上的结点指针
    LinkNode<T>* Locate(int pos);
    //在指定位置pos插入值为item的结点,失败返回false
    bool Insert(T &item, int pos);
    //删除指定位置pos上的结点,item就是该结点的值,失败返回false
    bool Remove(int pos, T &item);
    //获取指定位置pos的结点的值,失败返回false
    bool GetData(int pos, T &item);
    //设置指定位置pos的结点的值,失败返回false
    bool SetData(int pos, T &item);
    //判断链表是否为空
    bool IsEmpty() const;
    //打印链表
    void Print() const;
    //链表排序
    void Sort();
    //链表逆置
    void Reverse();
private:
    LinkNode<T> *head;
};

定位位置 

/* 返回链表中第pos个元素的地址,如果pos<0或pos超出链表最大个数返回NULL */
template<class T>
LinkNode<T>* LinkList<T>::Locate(int pos)
{int i = 0;LinkNode<T> *p = head;if (pos < 0)return NULL;while (NULL != p && i < pos){p = p->next;i++;}return p;
}

插入结点

单链表插入结点的处理如图

图:单链表插入操作

要在p结点后插入一个新结点node,(1)要让node的next指针指向p的next结点;(2)再让p的next指向node结点(即断开图中的黑色实线,改成红色虚线指向node)

接下来:node->next = p->next; p->next = node;

template<class T>
bool LinkList<T>::Insert(T &item, int pos)
{LinkNode<T> *p = Locate(pos);if (NULL == p)return false;LinkNode<T> *node = new LinkNode<T>(item);if (NULL == node){cerr << "分配内存失败!" << endl;exit(1);}node->next = p->next;p->next = node;return true;
}

删除结点

删除结点的处理如图:

图:单链表删除

删除pos位置的结点,如果这个位置不存在结点,则返回false;

如果找到对应结点,则通过实参item输出要删除的结点的数值, 然后删除结点并返回true。

template<class T>
bool LinkList<T>::Remove(int pos, T &item)
{LinkNode<T> *p = Locate(pos);if (NULL == p || NULL == p->next)return false;LinkNode<T> *del = p->next;p->next = del->next;item = del->data;delete del;return true;
}

清空链表 

遍历整个链表,每次head结点的next指针指向的结点,直到next指针为空。

最后保留head结点。

template<class T>
void LinkList<T>::Clear()
{LinkNode<T> *p = NULL;//遍历链表,每次都删除头结点的next结点,最后保留头结点while (NULL != head->next){p = head->next;head->next = p->next;   //每次都删除头结点的next结点
        delete p;}
}

求链表长度和打印链表

着两个功能的实现非常相近,都是遍历链表结点,不赘述。

template<class T>
void LinkList<T>::Print() const
{int count = 0;LinkNode<T> *p = head;while (NULL != p->next){p = p->next;std::cout << p->data << " ";if (++count % 10 == 0)  //每隔十个元素,换行打印cout << std::endl;}
}template<class T>
int LinkList<T>::Length() const
{int count = 0;LinkNode<T> *p = head->next;while (NULL != p){p = p->next;++count;}return count;
} 

单链表倒置

单链表的倒置处理如图:

图:单链表倒置

(1)初始状态:prev = head->next; curr = prev->next;

(2)让链表的第一个结点的next指针指向空

(3)开始进入循环处理,让next指向curr结点的下一个结点;再让curr结点的next指针指向prev。即:next = curr->next; curr->next = prev;

(4)让prev、curr结点都继续向后移位。即:prev = curr; curr = next;

(5)重复(3)、(4)动作,直到curr指向空。这时循环结束,让haed指针指向prev,此时的prev是倒置后的第一个结点。即:head->next = prev;

template<class T>
void LinkList<T>::Reverse()
{LinkNode<T> *pre = head->next;LinkNode<T> *curr = pre->next;LinkNode<T> *next = NULL;head->next->next = NULL;while (curr){next = curr->next;curr->next = pre;pre = curr;curr = next;}head->next = pre;
}

转载于:https://www.cnblogs.com/jingmoxukong/p/3827011.html

单链表的C++实现(采用模板类)相关推荐

  1. 单链表删除所有值为x的元素_线性表之单链表

    单链表 一种以链接方式存储的线性表,适用于频繁增删操作,存储空间不定的情形. 单链表的一个存储结点包含两个域,数据域和指针域.数据域用于存储线性表的一个数据元素,指针域用于指示下一个结点开始的存储地址 ...

  2. 数据结构学习之单链表

    节点类 #ifndef Node_H #define Node_H template <class Type> class Node //单链节点类 { public: Type data ...

  3. 实现单链表--Python

    单链表的结构 单链表结点的组成: 元素 链接域(保存下一结点的地址) 在单链表里,表里的n 个结点通过链接形成一条结点链.从表里的任一个结点都可以找到保存下一个元素的结点. 链表中的一个重要部分就是头 ...

  4. 自己动手写一个单链表

    文章有不当之处,欢迎指正,如果喜欢微信阅读,你也可以关注我的微信公众号:好好学java,获取优质学习资源. 一.概述 单向链表(单链表)是链表的一种,其特点是链表的链接方向是单向的,对链表的访问要通过 ...

  5. php数据结构课程---2、链表(php中 是如何实现单链表的(也就是php中如何实现对象引用的))...

    php数据结构课程---2.链表(php中 是如何实现单链表的(也就是php中如何实现对象引用的)) 一.总结 一句话总结: php是弱类型语言,变量即可表示数值,也可表示对象:链表节点的数据域的值就 ...

  6. java 链表反转_剑指BAT:如何最优雅着反转单链表?

    前言 以专题的形式更新刷题贴,欢迎跟我一起学习刷题,相信我,你的坚持,绝对会有意想不到的收获.每道题会提供简单的解答,如果你有更优雅的做法,欢迎提供指点,谢谢 [题目描述] 反转单链表.例如链表为: ...

  7. 面试题----单链表实现栈

    编程实现下面的栈顶操作: class MyData {void push(data);void pop(&data);bool isEmpty(); }; 解析:显然这里需要实现栈的3种基本操 ...

  8. 小白算法积累——单链表6#带头结点单链表+递增有序

    题目:有一个带头结点的单链表L,设计一个算法使其元素递增有序. 关键字:带头结点单链表+递增有序 思路 采用直接插入排序算法的思想:就是先分理出头结点+第一个结点组成原始新链表,然后依次将后续结点摘下 ...

  9. java实现单链表NodeList

    目录 一.单链表的概念 二.单链表的实现 1.定义节点类 2.定义单链表 1.属性 2.方法 三.完整实现 一.单链表的概念 单链表的概念:单链表是一种链式存取的数据结构,链表中的数据是以结点来表示的 ...

最新文章

  1. Android学习书籍参考
  2. 节能无线信标Ver0:功率测试
  3. ZigZag Conversion
  4. es dsl java api_求帮助将dsl翻译为java的API
  5. 【Auto.js】QQ自动回赞_简易版
  6. Java DataInputStream readUnsignedByte()方法(带示例)
  7. LeetCode(1108)——IP 地址无效化(JavaScript)
  8. 专访 Unity 技术总监 Mark Schoennagel、杨栋:基于 Unity 制作影视动画的几点建议!
  9. JavaScript函数的arguments(2)
  10. Java基础总结04-数组
  11. 华为机顶盒系统时间同步服务器,华为悦盒主时间同步服务器地址
  12. IC前端设计使用的EDA软件
  13. 群晖DS218+部署GitLab
  14. 采集51job职位数量画图后发邮件
  15. 分体式水晶头_六类水晶头的接线方法(分体式安装图解)
  16. 传统的6d位姿估计fangfa1_6D目标姿态估计,李飞飞夫妇等提出DenseFusion
  17. 与奥运会有关的常用英语术语及句子
  18. 【标题】视频标注软件DARKLABEL V2.4 主页中英对照图
  19. “您希望继续执行吗? [Y/n] y 中止。”
  20. java开发工程师英文_java工程师英文简历范文

热门文章

  1. java冻结行列,poi冻结行和列 - osc_0k23td2u的个人空间 - OSCHINA - 中文开源技术交流社区...
  2. matlab引擎函数,Matlab引擎库函数
  3. html5相对父元素定位,layer弹出层设置相对父级元素定位
  4. php中qq第三方登录演示,实现腾讯qq第三方登录
  5. 计算机usb共享网络泄密,杜绝USB泄密 MyUSBOnly
  6. linux终端帮助,Linux下的帮助命令
  7. python传送带和斐波那契
  8. Java程序员的工资为什么一直那么高?
  9. Docker的运行机制
  10. Spring入门 IOC