单链表的简单操作

单链表是一种最简单的线性表的链式存储结构,单链表又称线性链表,每个数据元素用一个结点来存储,结点分为存放数据元素的data和存放指向链表下一个结点的指针next。

链表实现:(LinkList.h)

//单链表
#ifndef LINKLIST_H_
#define LINKLIST_H_
#include <iostream>
//using namespace std;
template <typename T>
struct Node
{
	//数据成员
	T data;
	Node<T> *next;
	//构造函数
	Node();
	Node(T item, Node<T> *link=NULL);
};
//结点类实现部分
template <typename T>
Node<T>::Node()
{//构造
	next=NULL;
}
template <typename T>
Node<T>::Node(T item, Node<T> *link)
{//构造一个数据域为item和指针域为link的结点
	data=item;
	next=link;
}

//简单线性链表类
template <typename T>
class SimpleLinkList
{
protected:
	Node<T> *head;//头结点指针
	//
	//Node<T> *GetElemPtr(int position) const;
	void Init();//初始化线性表
public:
	Node<T> *GetElemPtr(int position) const;
	SimpleLinkList();//无参数构造函数
	virtual ~SimpleLinkList();//析构函数
	int Length() const;
	bool IsEmpty() const;
	void Clear();
	void Traverse(void(* Visit)(T &));
	int GetElem(int position, T &e) const;
	SimpleLinkList<T> &SetElem(int position, const T &e);
	SimpleLinkList<T> &Delete(int position, T &e);
	SimpleLinkList<T> &Insert(int position, const T &e);
	SimpleLinkList(const SimpleLinkList<T> &copy);
	SimpleLinkList<T> &operator =(const SimpleLinkList<T> &copy);
};
//函数实现
template<typename T>
Node<T> *SimpleLinkList<T>::GetElemPtr(int position) const
{
	Node<T> *tempPtr=head;
	int curPosition=0;
	while(tempPtr!=NULL && curPosition<position)
	{
		tempPtr=tempPtr->next;
		++curPosition;
	}
	if(tempPtr!=NULL && curPosition==position)
	{//查找成功
		return tempPtr;
	}
	else
	{//查找失败
		return NULL;
	}
}

template <typename T>
void SimpleLinkList<T>::Init()
{
	head=new Node<T>;//构造头指针
}
//构造函数
template <typename T>
SimpleLinkList<T>::SimpleLinkList()
{
	Init();
}
//析构函数
template<typename T>
SimpleLinkList<T>::~SimpleLinkList()
{
	Clear();//清空线性表
	delete head;//释放头结点所指空间
}
//返回链表长度
template<typename T>
int SimpleLinkList<T>::Length() const
{
	int count=0;
	for(Node<T> *tempPtr=head->next; tempPtr!=NULL; tempPtr=tempPtr->next)
	{
		++count;
	}
	return count;
}
//判断是否为空
template<typename T>
bool SimpleLinkList<T>::IsEmpty() const
{
	if(head->next==NULL)
		return 1;
	else
		return 0;
	//return head->next==NULL;
}
//清空链表
template <typename T>
void SimpleLinkList<T>::Clear()
{
	T temp;
	while(Length()>0)
	{
		Delete(0,temp);
	}
}
//遍历链表
template<typename T>
void SimpleLinkList<T>::Traverse(void(* Visit)(T &))
{
	for(Node<T> *tempPtr=head->next; tempPtr!=NULL; tempPtr=tempPtr->next)
	{
		(*Visit)(tempPtr->data);//对线性表的每个元素调用函数(*Visit)
		//std::cout<<tempPtr->data<<std::endl;
	}
}
//获取结点值
template <typename T>
int SimpleLinkList<T>::GetElem(int position, T &e) const
{
	if(position<0 || position>=Length())
	{
		std::cout<<"您说访问的位置超出链表范围!"<<std::endl;
		return -1;
	}
	else
	{
		Node<T> *tempPtr;
		tempPtr=GetElemPtr(position);
		e=tempPtr->data;
		return 0;
	}
}
//设置结点值
template<typename T>
SimpleLinkList<T> &SimpleLinkList<T>::SetElem(int position, const T &e)//是不是要加一个引用
{
	if(position<0 || position>Length())
	{
		std::cout<<"超出范围!"<<std::endl;
		return *this;//这个this
	}
	else
	{
		Node<T> *tempPtr;
		tempPtr=GetElemPtr(position);
		tempPtr->data=e;
		return *this;
	}
}
//删除结点
template<typename T>
SimpleLinkList<T> &SimpleLinkList<T>::Delete(int position, T &e)
{
	if(position<0 || position>Length())
	{
		std::cout<<"超出范围!"<<std::endl;
		return *this;//这个this
	}
	else
	{
		Node<T> *tempPtr;
		tempPtr=GetElemPtr(position);
		Node<T> *nextPtr=tempPtr->next;
		tempPtr->next=nextPtr->next;
		e=nextPtr->data;
		delete nextPtr;
		return *this;
	}
}
//插入结点
template<typename T>
SimpleLinkList<T> &SimpleLinkList<T>::Insert(int position, const T &e)
{
	if(position<0 || position>Length())
	{
		std::cout<<"超出范围!"<<endl;
		return *this;
	}
	else
	{
		Node<T> *tempPtr;
		tempPtr=GetElemPtr(position);
		Node<T>* newPtr;
		newPtr=new Node<T>(e,tempPtr->next);
		tempPtr->next=newPtr;
		return *this;

	}
}
template <typename T>
SimpleLinkList<T>::SimpleLinkList(const SimpleLinkList<T> &copy)
{
	int copyLength=copy.Length();
	Init();
	Node<T> *tempPtr;
	for(int curPosition=0; curPosition<copyLength; curPosition++)
	{
		//Node<T> *tempPtr;
		tempPtr=copy.GetElemPtr(curPosition+1);
		Insert(curPosition,tempPtr->data);
	}
	//return *this;
}
template<typename T>
SimpleLinkList<T> &SimpleLinkList<T>::operator =(const SimpleLinkList<T> &copy)
{
	if(&copy!=this)
	{
		int copyLength=copy.Length();
		Node<T> *tempPtr;
		Init();
		for(int curPosition=0; curPosition<=copyLength; curPosition++)
		{
			//Node<T> *tempPtr;
			tempPtr=copy.GetElemPtr(curPosition);
			Insert(curPosition,tempPtr->data);
		}
	}
	//return this;
}
template<typename T>
std::ostream & operator<<(std::ostream &os, SimpleLinkList<T>& LinkList)
{
	std::cout<<"输出链表: ";
	for(int curPosition=1; curPosition<=LinkList.Length(); curPosition++)
	{
		std::cout<<((LinkList.GetElemPtr(curPosition))->data)<<" ";
	}
	return os;
}
#endif

主程序:(LinkList.cpp)

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

#include "stdafx.h"
#include "LinkList.h"
#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
	SimpleLinkList<int> LinkList;
	cout<<"链表的长度为:"<<LinkList.Length()<<endl;
	cout<<"链表是否为空? ";
	if(LinkList.IsEmpty()==1)
	{
		cout<<"Yes!"<<endl;
	}
	else
	{
		cout<<"No!"<<endl;
	}
	LinkList.Insert(0,1);
	LinkList.Insert(1,2);
	LinkList.Insert(2,3).Insert(3,4).Insert(4,5).Insert(5,6);
	cout<<LinkList<<endl;
	cout<<"链表的长度为:"<<LinkList.Length()<<endl;
	cout<<"链表是否为空? ";
	if(LinkList.IsEmpty()==1)
	{
		cout<<"Yes!"<<endl;
	}
	else
	{
		cout<<"No!"<<endl;
	}
	//复制链表
	//SimpleLinkList<int> copy(LinkList);
	SimpleLinkList<int> copy=LinkList;
	copy.Insert(2,0).Insert(3,0);
	cout<<"链表的长度为:"<<copy.Length()<<endl;
	cout<<"链表是否为空? ";
	if(copy.IsEmpty()==1)
	{
		cout<<"Yes!"<<endl;
	}
	else
	{
		cout<<"No!"<<endl;
	}
	cout<<copy<<endl;
	cout<<"设置值之后:"<<endl;
	copy.SetElem(1,100);
	cout<<copy<<endl;
	copy.Clear();
	cout<<"链表的长度为:"<<copy.Length()<<endl;
	cout<<"链表是否为空? ";
	if(copy.IsEmpty()==1)
	{
		cout<<"Yes!"<<endl;
	}
	else
	{
		cout<<"No!"<<endl;
	}
	system("pause");
	return 0;
}

输出结果:

时间: 2024-10-18 10:58:40

单链表的简单操作的相关文章

对带头结点的单链表的简单操作

#pragma once #include<stdio.h> #include<stdlib.h> #include<assert.h> #include<memory.h> #define DataType int           //int 可以改写为其它数据类型 typedef struct Node { DataType data; struct Node *next; }Node,*pNode;          //定义结点结构体      

c# 单链表实现 简单示例(可复制直接运行)

最近学习数据结构,发现c# 其实和c 的链表的实现差不多的 下面是一段可直接运行的代码 1 using System; 2 using System.Collections.Generic; 3 using System.Text; 4 using System.Threading; 5 6 namespace SingleLinkedList 7 { 8 class Program 9 { 10 static void Main(string[] args) 11 { 12 13 //实例调用

单链表的基础操作

单链表中节点的查找.插入.删除.求单链表长度等操作. 按序号查找结点值 在单链表中从第一个结点出发,顺指针next域逐个往下搜索,直到找到第i个结点为止,否则返回最后一个结点指针域NULL. 按序号查找结点值的算法如下: LNode GetElem(LinkList L,int i){ //本算法取出单链表L(带头结点)中第i个位置的结点指针 int j=1; //计数,初始为1 LNode *p = L->next; //头结点指针赋给p if(i==0) return L; //若i等于0,

数据结构关于单链表的一些操作的源代码

单链表的可以有许多问题,这是我特意整理一下的有关他的相关操作,给出代码,有需要的可以自己调试,重要的就是关于环的一些操作: #include <iostream>#include <cstdio>#include <cstdlib>#include <ctime>using namespace std;typedef int Elemtype;typedef struct Node{ Elemtype data; struct Node *next;}Nod

算法数据结构 单链表的实现+操作 以及和顺序表的对比

顺序表和单链表的优缺点对比: 顺序表的优点,无需为表示表中元素之间的逻辑关系而增加额外的存储空间: 可以快速的存取表中的任意位置的元素. 顺序表的缺点,插入后删除操作需要移动大量元素: 当线性表长度不稳定时,存储空间难确定,容易造成存储空间碎片. 对于单链表 链式存储即元素存储的内存单元可以是不连续,分散的.对于元素间如何来维护他们的关系(即逻辑结构,每个元素的前驱和后继.) 即用到一个指针域来存储他和前驱或是后继直接的关系. 如上面的是一个单链表的指针结构,即每个元素中存储了他的后继元素的内存

详谈单链表的有关操作集锦~

1.单链表    在 Java 中没有显式的指针类型,然而实际上对象的访问就是使用指针来实现的,即在Java 中是使用对象的引用来替代指针的.因此在使用 Java 实现该结点结构时,一个结点本身就是一个对象.结点的数据域 data 可以使用一个 Object 类型的对象来实现,用于存储任何类型的数据元素,并通过对象的引用指向该元素:而指针域 next 可以通过节点对象的引 用来实现.      单链表结点结构是结点的一种最简单的形式,除此之外还有其他不同的结点结构,但是这些结点结构都有一个数据域

带头节点的单链表的插入操作

1.偶然看到了十字链表的应用,想到之前在<数据结构与算法分析>的链表一章中,需要用多重表实现一个简单的查询功能.功能需求如下: “已知 学生 和 学校课程 总数 分别为 40000 和 2500,现在需要得到两份报告,一份显示每门课成注册的所有学生信息, 一份显示每个学生注册了哪些课程.” 显然可以用一个 40000 * 2500 个元素的二维数组来解决,但是每个学生选课数目很少,因此会浪费很多空间.因此选择十字链表来实现. 既然是链表,那么肯定要有插入操作,于是便有了本文.算是对功能实现前的

单链表的相关操作

#ifndef _SLIST_H #define _SLIST_H #ifdef __cplusplus extern "C" { #endif /*******1. 不带头结点的单链表*****/ /***** *@链表结点结构定义 *@ m_data:数据 *@m_pNext:指向下一结点的指针 ***/ struct listNode { int m_data; listNode* m_pNext; }; /******* *@ 用数组array初始化链表,数组元素个数为n *@

链表-单链表的各种操作

单链表的结构体的定义 typedef struct LNode { ElemType data; struct LNode *next; }LinkList; 基本的单链表的操作 /* 功能:构建一个空的带头节点的单链表*/ Status InitList (struct LNode **L) { (*L) = (struct LNode *)malloc(sizeof(struct LNode)); //产生头节点 if(!*L) exit(OVERFLOW); (*L)->next = NU