带头结点
#include <iostream>
#include <stdlib.h>
//定义单链表结点类型
typedef struct LNode
{
int data; //每个节点存放一个数据元素
struct LNode* next; //指针指向下一个结点
}LNode, * LinkList;
//初始化一个单链表
bool InitList(LinkList &L)
{
L=(LNode*)malloc(sizeof(LNode));
if(L==NULL)
return false;
L->next=NULL;
return true;
}
//判断单链表是否为空
bool Empty(LinkList L)
{
return L->next==NULL;
}
//判断结点p是否为循环单链表的表尾结点
bool isTail(LinkList L, LNode* p)
{
return p->next == NULL;
}
//正向建立单链表,尾插法
LinkList List_TailInsert(LinkList& L)
{
int x;
L = (LinkList)malloc(sizeof(LNode));//建立头结点
if (L)
{
LNode* s, * r = L; //r为表尾指针
std::cin >> x; //输入结点的值
while (x != -1) //输入-1表示结束
{
s = (LNode*)malloc(sizeof(LNode));
if (s)
{
s->data = x;
r->next = s; //在r结点之后插入元素x
r = s; //r指向新的表尾结点,永远指向最后一个结点
std::cin >> x;
}
}
r->next = NULL; //尾结点指针置空
}
return L;
}
//逆向建立单链表,头插法;重要应用: 链表的逆置
LinkList List_HeadInsert(LinkList& L)
{
LNode* s;
int x;
L = (LNode*)malloc(sizeof(LNode));
if (L)
{
L->next = NULL;
std::cin >> x;
while (x != -1)
{
s = (LNode*)malloc(sizeof(LNode));
if (s)
{
s->data = x;
s->next = L->next;
L->next = s;
std::cin >> x;
}
}
}
return L;
}
//指定结点的后插操作
bool InsertNextNode(LNode* p, int e)
{
if (p == NULL)
return false;
LNode* s = (LNode*)malloc(sizeof(LNode));
if (s == NULL) //内存分配失败
return false;
s->data = e; //用结点 s 保存数据元素 e
s->next = p->next;
p->next = s; //将结点 s 连接到 p 之后
return true;
}
//按位查找,返回第i个元素(带头结点)
LNode* GetElem(LinkList L, int i)
{
if (i < 0)
return NULL;
LNode* p; //指针p指向当前扫描到的结点
int j = 0; //当前p指向的是第几个结点
p = L; //L指向头结点,头结点是第0个结点(不存数据)
while (p != NULL && j < i)//循环找到
{
p = p->next;
j++;
}
return p;
}
//前插操作: 在 p 结点之前插入元素 e
bool InsertPriorNode(LNode* p, int e)
{
if (p == NULL)
return false;
LNode* s = (LNode*)malloc(sizeof(LNode));
if (s == NULL)
return false;
s->next = p->next;
p->next = s; //新结点 s 连到 p 之后
s->data = p->data; //将 p 中元素复制到 s 中
p->data = e; //p 中元素覆盖为 e
return true;
}
//在第 i 个位置上插入元素 e
bool ListInsert(LinkList& L, int i, int e)
{
if (i < 1)
return false;
LNode* p = GetElem(L, i - 1);//找到第i-1个结点
return InsertNextNode(p, e);
}
//在第i个位置上删除当前节点,并返回被删除的值
bool ListDelete(LinkList& L, int i, int& e)
{
if (i < 1)
return false;
LNode* p = GetElem(L, i - 1);//找到i-1个结点
if (p == NULL)
return false;
if (p->next == NULL)//第i-1结点之后已无其他结点
return false;
LNode* q = p->next; //令q指向被删除结点
e = q->data; //用e换回元素的值
p->next = q->next; //将*q结点从链中"断开"
free(q); //释放结点的存储空间
return true;
}
//删除指定结点 p,如果p是最后一个结点的话,只能从表头开始依次寻找p的前驱.
bool DeleteNode(LNode* p)
{
if (p == NULL&&p->next==NULL)
return false;
LNode* q = p->next; //令q指向*p的后继结点
p->data = p->next->data; //和后继结点交换数据域
p->next = q->next; //将*q结点从链中"断开"
free(q); //释放后继结点的存储空间
return true;
}
//按值查找,找到数据域==e的结点
LNode* LocateElem(LinkList L, int e)
{
LNode* p = L->next;
//从第1个结点开始查找数据域为e的结点
while (p != NULL && p->data != e)
p = p->next;
return p; //找到后返回该结点指针,否则返回NULL
}
//求表的长度
int Length(LinkList L)
{
int len = 0; //统计表长
LNode* p = L;
while (p->next != NULL)
{
p = p->next;
len++;
}
return len;
}
int main()
{
LinkList L; //声明一个指向单链表的指针
return 0;
}
不带头结点
#include <iostream>
#include <stdlib.h>
using namespace std;
//定义单链表结点类型
typedef struct LNode
{
int data; //每个节点存放一个数据元素
struct LNode* next; //指针指向下一个结点
}LNode, * LinkList;
//初始化一个单链表
bool InitList(LinkList& L)
{
L = NULL; //空表,暂时还没有任何结点
return true;
}
//判断单链表是否为空
bool Empty(LinkList L)
{
return L == NULL;
}
//求表的长度
int Length(LinkList L)
{
int len = 0; //统计表长
LNode* p = L;
while (p != NULL)
{
p = p->next;
len++;
}
return len;
}
//按位查找,返回第i个元素
LNode* GetElem(LinkList L, int i)
{
if (i < 0)
return NULL;
LNode* p; //指针p指向当前扫描到的结点
int j = 1; //当前p指向的是第几个结点
p = L; //p指向第1个结点
while (p != NULL && j < i)//循环找到第i个结点
{
p = p->next;
j++;
}
return p;
}
//按值查找
LNode* LocateElem(LinkList L, int e)
{
LNode* p = L;
while (p != NULL && p->data != e)
p = p->next;
return p; //找到后返回该结点指针,否则返回NULL
}
//指定结点的后插操作
bool InsertNextNode(LNode* p, int e)
{
if (p == NULL)
return false;
LNode* s = (LNode*)malloc(sizeof(LNode));
if (s == NULL) //内存分配失败
return false;
s->data = e; //用结点s保存数据元素e
s->next = p->next;
p->next = s; //将结点s连到p之后
return true;
}
//指定结点的前插操作
bool InsertPriorNode(LNode* p, int e)
{
if (p == NULL)
return false;
LNode* s = (LNode*)malloc(sizeof(LNode));
if (s == NULL) //内存分配失败
return false;
s->next = p->next;
p->next = s; //新结点s连到p之后
s->data = p->data; //将p中元素复制到s中
p->data = e; //p中元素覆盖为e
return true;
}
//按位插入
bool ListInsert(LinkList& L, int i, int e)
{
if (i < 1)
return false;
if (i == 1) //插入第1个结点的操作与其他结点操作不同
{
LNode* s = (LNode*)malloc(sizeof(LNode));
if (s == NULL)
return false;
s->data = e;
s->next = L;
L = s; //头指针指向新结点
return true;
}
return InsertNextNode(GetElem(L, i - 1), e);
}
//按位序删除
bool ListDelete(LinkList& L, int i, int& e)
{
if (i < 1)
return false;
if (i == 1) //i=1,特殊处理
{
if (L == NULL) //空表
return false;
LNode* q = L; //指向第1个结点
e = q->data; //返回删除的值
L = q->next; //L等于q指向的结点的next值,也就是指向下一个结点
free(q); //释放结点的存储空间
return true;
}
LNode* p; //指针p指向当前扫描到的结点
int j = 1; //当前p指向的是第几个结点
p = L; //p指向第1个结点
p = GetElem(L, i - 1);
if (p == NULL) //i值不合法
return false;
if (p->next == NULL)//第i-1个结点之后已无其他结点
return false;
LNode* q = p->next; //令q指向被删除结点
e = q->data; //用e返回元素的值
p->next = q->next; //将*q结点从链中“断开”
free(q); //释放结点的空间
return true;
}
//指定结点的删除
bool DeleteNode(LNode* p)
{
if (p == NULL)
return false;
LNode* q = p->next; //令q指向*p的后继结点
p->data = p->next->data;//和后继结点交换数据域
p->next = q->next; //将*q结点从链中“断开”
free(q); //释放后继结点的存储空间
return true;
}
//正向建立单链表
LinkList List_TailInsert(LinkList& L)
{
int x;
L = NULL; //空表
std::cin >> x; //输入结点的值
if (x != -1) //读入-1则输入停止
{
LNode* s, * r = L; //r为表尾指针
s = (LNode*)malloc(sizeof(LNode));//先插入第一个结点
if (s == NULL)
return L;
s->data = x;//第一个结点插入元素x
L = s; //L指向第一个结点
r = s; //r指向第一个结点
cin >> x;
while (x != -1)
{
s = (LNode*)malloc(sizeof(LNode));
if (s == NULL)
return L;
s->data = x;
r->next = s; //在r结点之后插入元素x
r = s; //永远保持r指向最后一个结点
cin >> x;
}
r->next = NULL; //尾结点指针置空
}
return L;
}
int main()
{
LinkList L;
InitList(L); //测试
//ListInsert(L, 1, 6);
//ListInsert(L, 2, 7);
int a = 0;
//cout << ListDelete(L, 2, a) << " " << a << endl;//由于不带头结点,测试是否能正确删除.
//cout << GetElem(L, 2) << endl;
//cout<<LocateElem(L, 7)<< endl;
//cout << Length(L) << endl;
//List_TailInsert(L);
return 0;
}