线性表——单链表

  • 链表结点数据类型的定义
    在C++中,可以用结构类型来描述单链表的结点 ,由于结点的元素类型不确定,所以采用C++的模板机制。
template <typename T>
struct Node 
{ 
  T data; 
  Node<T> *next;      //此处<T>也可以省略
};
  • 单链表的实现
template <class T>
class LinkList { 
public: 
  LinkList ( ) {first=new Node<T>; first -> next= NULL ;}//无参构造函数,建立只有头结点的空链表
   LinkList ( T a[ ], int n ) ; //有参构造函数,建立有n个元素的单链表
  ~LinkList ( ) ; //析构函数
  int Length ( ) ;//求单链表的长度
  T Get ( int i ) ; //按位查找
  int Locate ( T x ) ;//按值查找
  void Insert ( int i, T x ) ;//插入操作
  T Delete ( int i ) ; //删除操作
  void PrintList ( ) ; //遍历操作
private: 
  	Node<T>  *first; // 单链表的头指针  , <T>可以省略
};
  • 无参构造函数

  • 有参构造函数
    有参构造函数生成单链表:头插法和尾插法。
    ①头插法:

template <class T>  
LinkList<T>:: LinkList(T a[ ], int n) {
    first=new Node<T>;   //生成头结点
   first->next=NULL;//尾节点指针域为空
   Node<T> *s;
   for (int i=0; i<n; i++){ 
          s=new Node<T>; 
          s->data=a[i];  //为每个数组元素建立一个结点
          s->next=first->next;
          first->next=s;
	}
}

②尾插法:

template <class T>  
LinkList<T>:: LinkList(T a[ ], int n) {
    Node<T> *r,*s;      //尾指针
    first=new Node<T>;   //生成头结点
	r=first;          
    for (int i=0; i<n; i++)	{ 
        s=new Node<T>; 
        s->data=a[i];  //为每个数组元素建立一个结点
        r->next=s; r=s;      //插入到终端结点之后
	}
    r->next=NULL;    //单链表建立完毕,将终端结点的指针域置空
 }
  • 析构函数
template <class T>
LinkList<T>:: ~LinkList()
{
   Node<T> *q;
   while (first)
   {
       q=first->next;
       delete first;
       first=q;
    }
}
  • 单链表的长度
template <class T>
int LinkList<T>::length()
{
	p=first->next;int count=0;
	while(p!=NULL)
	{
		p=p->next;
		count++;
	}
	return count;
}
  • 按位查找
    查找算法:
    1、工作指针P初始化,计数器初始化
    2、执行下列操作,直到p为空或指向第i个节点
    2.1 工作指针后移
    2.2 计数器增1
    3、若p为空,则第i个元素不存在,抛出位置异常;否则查找成功,返回节点p的数据元素
template <class T>
T LinkList<T>::Get(int i) {   
	  Node<T> *p; int count;
	  p=first->next; count=1;  //或p=first;  count=0;
	  while (p && count<i) {
    		p=p->next;       //工作指针p后移
			count++;
  	 }
	  if (!p) throw "位置";
	  else return p->data;
}
  • 按值查找
template <class T>
T LinkList<T>::Locate(T x) {   
	  Node<T> *p; int count;
	  p=first->next; count=1;  
	  while (p) {
	  	if(p->data==x)return count;
    		p=p->next;       //工作指针p后移
			count++;
  	 }
	 return 0;
}
  • 插入操作
    插入算法:
    1、工作指针p初始化,计数器初始化
    2、查找第i-1个节点,并使工作指针p指向该节点
    3、若查找不成功(P==NULL),说明位置错误,抛出位置异常,否则
    3.1 生成一个元素值为x的新节点s
    3.2 将s插入到p之后
template <class T>  
void LinkList<T>::Insert(int i, T x){  
   Node<T> *p; int count;
   p=first ; count=0;    //工作指针p初始化
   while (p && count<i-1)   {
     p=p->next;   //工作指针p后移
     count++;
   }
   if (!p) throw "位置";
    else { 
	  Node<T> *s;
      s=new Node<T>; 
	  s->data=x;  //向内存申请一个结点s,其数据域为x
      s->next=p->next;       //将结点s插入到结点p之后
      p->next=s;	
	}
 }
  • 删除操作
    删除算法:
    1、工作指针p初始化;累加器count初始化;
    2、查找第i-1个结点并使工作指针p指向该结点;
    3、若p不存在或p的后继结点不存在,则抛出位置异常;
    否则
    3.1 暂存被删结点和被删元素值
    3.2 摘链,将结点p的后继结点从链表上摘下
    3.3 释放被删结点
    3.4 返回被删元素值
template <class T>  
T LinkList<T>::Delete(int i){ 
  Node<T> *p; int j;
  p=first ; j=0;  //工作指针p初始化
  while (p && j<i-1) {  //查找第i-1个结点
    p=p->next; 
    j++;
  }
  if (!p || !p->next) throw "位置";  //结点p不存在或结点p的后继结点不存在
    else {
  	     Node<T> *q; T x;
          q=p->next; x=q->data;  //暂存被删结点
          p->next=q->next;  //摘链
          delete q; 
          return x;
	}
}
  • 遍历操作
template <class T>  
LinkList<T>:: PrintList()
{
    Node<T> *p;
	p=first->next;          
    while(p)  //等同于while(p!=NULL)
	{
		cout<<p->data;
        p=p->next;
	}
 }

不带头结点的操作
1.头插法:

template <class T>  
LinkList<T>:: LinkList(T a[ ], int n) {
    first=NULL;
   for (int i=0; i<n; i++){ 
          s=new Node<T>; 
          s->data=a[i];  //为每个数组元素建立一个结点
          s->next=first;
          first=s;
	}
}

2.尾插法:

template <class T>  
LinkList<T>:: LinkList(T a[ ], int n) 
{
 	node<T> *r;
    head=NULL;
    if(n<=0)return;
    s=new node<T>;
    s->data=a[0];
    s->next=head;
    head=s;   
    r=head;
	for(int i=1;i<n;i++)    
	{ 
         s=new node<T>;
         s->data=a[i];
         r->next=s;
         r=s;   
    }

3.插入操作

template <class T>  
void LinkList<T>::Insert(int i, T x){  
   Node<T> *p; int j;
   if(i<=0) throw “位置非法”;
   if (i==1 ){ s=new Node<T>;s->next=head;head=s;return}
   p=first ; j=1;    //工作指针p初始化
   while (p && j<i-1)   {
     p=p->next;   //工作指针p后移
     j++;
   }
   if (!p) throw "位置";
    else { 
	  Node<T> *s;
      s=new Node<T>; 
	  s->data=x;  //向内存申请一个结点s,其数据域为x
      s->next=p->next;       //将结点s插入到结点p之后
      p->next=s;	
	}
 }

在这里插入图片描述在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值