【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会

这篇具有很好参考价值的文章主要介绍了【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

⭐⭐⭐⭐⭐⭐

🎊专栏【数据结构】

🍔喜欢的诗句:更喜岷山千里雪 三军过后尽开颜。

🎆音乐分享【勋章】

大一同学小吉,欢迎并且感谢大家指出我的问题🥰


【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会

⭐⭐⭐⭐⭐⭐ 

目录

⭐定义: 

⭐ 理解:

⭐存储方式 :

⭐顺序存储的优缺点:

优点:

缺点:

⭐链式存储的优缺点:

优点:

缺点:

⭐基本操作

✨顺序存储

🍔存储结构

🎈添加数字建立表

🎈输出线性表里面的数

 🎈插入数字

🎈删除数字

🎈取出数字

 🎈置空

😎完整代码1

🚌小细节:什么时候使用引用,什么时候不使用引用

😎完整代码2 

✨链式存储

🍔存储结构

⭐易错:首元结点VS头节点VS头指针

⭐链表增加头结点的好处

🎈便于首元结点的处理

🎈便于空表和非空表的统一处理

🎈初始化

🎈尾插法建立单链表

🎈头插法建立单链表

🎈插入

🎈删除

🎈取出元素

🎈输出链表

😎完整代码1

😎完整代码2 


 文章来源地址https://www.toymoban.com/news/detail-408858.html

⭐定义: 

线性表(List):零个或多个数据元素的有限序列

⭐ 理解:

线性表,顾名思义,就是具有像线一样性质的表,元素之间是有顺序的,若元素存在多个,那么第一个元素没有前驱元素最后一个元素没有后继元素其他元素既有前驱元素又有后继元素

⭐存储方式 :

线性存储

链式存储

⭐顺序存储的优缺点:

优点:

1.表中数据元素可以根据序号 随机存取

2. 存储密度大,存储密度为1(存储密度是指一个结点中数据元素所占的存储单元和整个结点所占的存储单元之比)

缺点:

1.做插入、删除操作时,要移动大量元素,因此对很长的线性表操作效率低,插入和删除操作不方便;

2.要预先分配存储空间,预先估计过大,会导致存储空间浪费,估计过小,会造成数据溢出。

⭐链式存储的优缺点:

优点:

1.做插入、删除操作时,不用移动大量元素,相比线性存储方式方便;

2.不用预先分配存储空间。 

缺点:

1.表中数据元素不可随机存取 

2.存储密度小,存储密度小于1

⭐基本操作

✨顺序存储

🍔存储结构

#define maxsize 100
typedef struct{
	ElemType *elem;//由于这里不知道线性表的具体长度,所以这里就用了指针
                   //可以是elem[maxsize]
	int length;
}SqList;

其中ElemType可以改为其他的数据类型 ,比如

typedef int ElemType

length表示当前线性表中数据元素的个数,注意数组里面的下标是从0开始的,那么

a1->elem[0]; //注意,用数组表示

a2->elem[1];

a3->elem[2] ;

an->elem[length-1];

🎈初始化

🚕分配空间,建立空表,使空表长度为0

使用引用初始化

int InitList(SqList &L)
{
	L.elem=new ElemType[maxsize];
	if(!L.elem) return -1;			//内存分配失败
	L.length=0;						//空表长度为0
	return 1;	
}

使用指针初始化 

int InitList(SqList* L)
{
	L->elem = new int[100];
	if (!L->elem) exit(overflow);
	L->length = 0;
	return OK;
}

🎈添加数字建立表

🚕向空表里面添加数字,从而建立线性表

使用引用

void InputList(SqList& L, int num)
{
	L.elem[k++] = num;
	L.length++;
}

使用指针

void InputList(SqList* L, int num)
{
	L->elem[k++] = num;
	L->length++;
}

🎈输出线性表里面的数

🚕输出线性表里面的数

使用引用

void OutputList(SqList& L)
{
	for (int i = 1; i <= L->length; i++)
	{
		cout << L.elem[i] << " ";
	}
}

使用指针 

void OutputList(SqList *L)
{
	for (int i = 1; i <= L.length; i++)
	{
		cout << L->elem[i] << " ";
	}
}

 🎈插入数字

🚕顺序存储插入数字,就是找到要插入的数字的位置,从这个位置开始的后面的所有的数字全都后移一位,这样子前面就回空出一个位置,就是要插入的数字的位置

使用引用

void ListInsert(SqList& L, int place, int num)
{
	L.length++;
	for (int i = L.length; i >= place; i--)
	{
		L.elem[i + 1] = L.elem[i];
	}
	L.elem[place] = num;

}

注意:这里是  i = L.length , 不是 i = L.length-1

你想一下,数组下标是从0开始的,所以整个表最后一个位置(L.elem[length])没有存数

 这样把数组后移,到最后刚好可以空出一个位置来存需要插入的数

使用指针

void ListInsert(SqList *L, int place, int num)
{
	L->length++;
	for (int i = L->length; i >= place; i--)
	{
		L->elem[i + 1] = L->elem[i];
	}
	L->elem[place] = num;

}

🎈删除数字

🚕与插入数字类似,删除数字采用的是向前覆盖数字

使用引用

void ListDelete(SqList& L, int place)
{

	for (int i = place; i <= L.length; i++)
	{
		L.elem[i - 1] = L.elem[i];
	}
	L.length--;
}

使用指针

void ListDelete(SqList *L, int place)
{

	for (int i = place; i <= L->length; i++)
	{
		L->elem[i - 1] = L->elem[i];
	}
	L->length--;
}

🎈取出数字

🚕就是给出这个数字的位置,就是想当于给出了这个数字在数组里面的位置,然后直接根据这个位置取值

int GetElem(SqList L, int i, int& e)//int *e
{
	if ((i < 1) || (i > L.length)) return -1;
	e = L.elem[i];                  //*e=L.elem[i];
	return 1;
}
int GetElem(SqList *L, int i, int& e)
{
	if ((i < 1) || (i > L->length)) return -1;
	e = L->elem[i];
	return 1;
}

 🎈置空

🚕直接L.length=0就可以了

(如果要返回线性表的长度,直接L.length即可)

😎完整代码1

#include<iostream>
using namespace std;
#define OK 1
#define overflow -1
int k=1;

typedef struct{
	int *elem;
	int length;
}SqList;

int InitList(SqList &L)
{
	L.elem=new int[100];
	if(!L.elem) exit(overflow);
	L.length=0;
	return OK;
}

//3
void InputList(SqList &L,int n)
{
	L.elem[k++]=n;
	L.length++;
}

//4
void OutputList(SqList &L)
{
	for(int i=1;i<=L.length;i++)
	{
		cout<<L.elem[i]<<" ";
	}
}	

//5
void ListInsert(SqList &L,int a,int b)
{
	L.length++;
	for(int i=L.length;i>=a;i--)
	{
		L.elem[i+1]=L.elem[i];
	}
	L.elem[a]=b;
	
}

//6 	
 void ListDelete(SqList &L,int x)
 {

	for(int i=x;i<=L.length;i++)
	{
		L.elem[i-1]=L.elem[i];
	 } 	
	L.length--;
 }
 
 //7
 int GetElem(SqList L,int i,int &e)
 { 
 	if((i<1)||(i>L.length)) return -1;
	 e=L.elem[i];
	 return OK;
 }
 
 //8
int ClearList(SqList L)
{
    L.length=0;
    return 1;
}
 
int main()
{
	SqList L;
	int s,e;
	InitList(L);
	if(InitList) cout<<"成功"<<endl;
	else cout<<"失败"<<endl;
	cout<<"0:退出程序"<<endl;
	cout<<"3:加入数"<<endl;
	cout<<"4:输出线性表里面的数"<<endl;
	cout<<"5:插入数"<<endl;
	cout<<"6:删除数"<<endl;
	cout<<"7:取出某个位置的数"<<endl;
	cout<<"8:置空"<<endl;
	cout<<endl;
	for(;;)
	{
		cout<<"请输入3~8之间的数,代表3~8题,0表示程序结束"<<endl;
		cin>>s;
		if(s==0) return 0;
		switch(s)
		{
		case 3:
			cout<<"请输入需要加入的数的个数:"<<endl;
			int n;
			cin>>n;
			cout<<"请输入需要加入的数:"<<endl;
			for(int i=1;i<=n;i++)
			{
				int t;
				cin>>t;
				InputList(L,t);	
			}
			break;
		case 4:
			OutputList(L);
			cout<<endl;
			break;
		case 5:
			for(int i=1;i<=2;i++)
			{
				cout<<"请输入插入的位置和插入的数是什么:"<<endl;
				int a,b;
				cin>>a>>b;
				ListInsert(L,a,b);
			}
			break;
		case 6:
			cout<<"请输入需要删除哪个位置的数:"<<endl;
			for(int i=1;i<=2;i++)
			{
				int aa;
				cin>>aa;
				ListDelete(L,aa);
			}
			break;	
		case 7:
			cout<<"请输入需要取出哪个位置的数:"<<endl;
			for(int i=1;i<=2;i++)
			{
				int a;
				cin>>a;
				GetElem(L,a,e);
				cout<<"第"<<a<<"个位置的数的值为:"<<e<<endl;
			}
			break;
		case 8:
			if (ClearList(L)) cout << "置空成功" << endl;
			else cout << "置空失败" << endl;
			break;
		}
	}
 } 

🚌小细节:什么时候使用引用,什么时候不使用引用

观察代码我们会发现,

有的是SqList L(没有使用引用) 

有的是SqList &L(使用了引用)

这是为什么呢

原因:我们发现,使用引用的代码段里面都会有分配空间的操作,而没有使用引用的代码段里面就没有分配空间的操作

分配一段空间,使整个程序改变了,所以要使用引用,而如果不分配空间,那么就直接使用就行了,不需要引用

😎完整代码2 

#include<iostream>
using namespace std;
#define OK 1
#define overflow -1
int k = 1;

typedef struct {
	int* elem;
	int length;
}SqList;

int InitList(SqList* L)
{
	L->elem = new int[100];
	if (!L->elem) exit(overflow);
	L->length = 0;
	return OK;
}

//3
void InputList(SqList* L, int n)
{
	L->elem[k++] = n;
	L->length++;
}

//4
void OutputList(SqList* L)
{
	for (int i = 1; i <= L->length; i++)
	{
		cout << L->elem[i] << " ";
	}
}

//5
void ListInsert(SqList *L, int place, int num)
{
	L->length++;
	for (int i = L->length; i >= place; i--)
	{
		L->elem[i + 1] = L->elem[i];
	}
	L->elem[place] = num;

}

//6 	
void ListDelete(SqList *L, int place)
{

	for (int i = place; i <= L->length; i++)
	{
		L->elem[i - 1] = L->elem[i];
	}
	L->length--;
}

//7
int GetElem(SqList L, int i, int& e)
{
	if ((i < 1) || (i > L.length)) return -1;
	e = L.elem[i];
	return OK;
}

//8
int ClearList(SqList *L)
{

	L->length=0;
    return 1;
}

int main()
{
	SqList L;
	int s, e;
	InitList(&L);
	if (InitList) cout << "成功" << endl;
	else cout << "失败" << endl;
	cout << "0:退出程序" << endl;
	cout << "3:加入数" << endl;
	cout << "4:输出线性表里面的数" << endl;
	cout << "5:插入数" << endl;
	cout << "6:删除数" << endl;
	cout << "7:取出某个位置的数" << endl;
	cout << "8:置空" << endl;
	cout << endl;
	for (;;)
	{
		cout << "请输入3~8之间的数,代表3~8题,0表示程序结束" << endl;
		cin >> s;
		if (s == 0) return 0;
		switch (s)
		{
		case 3:
			cout << "请输入需要加入的数的个数:" << endl;
			int n;
			cin >> n;
			cout << "请输入需要加入的数:" << endl;
			for (int i = 1; i <= n; i++)
			{
				int t;
				cin >> t;
				InputList(&L, t);
			}
			break;
		case 4:
			OutputList(&L);
			cout << endl;
			break;
		case 5:
			for (int i = 1; i <= 2; i++)
			{
				cout << "请输入插入的位置和插入的数是什么:" << endl;
				int a, b;
				cin >> a >> b;
				ListInsert(&L, a, b);
			}
			break;
		case 6:
			cout << "请输入需要删除哪个位置的数:" << endl;
			for (int i = 1; i <= 2; i++)
			{
				int aa;
				cin >> aa;
				ListDelete(&L, aa);
			}
			break;
		case 7:
			cout << "请输入需要取出哪个位置的数:" << endl;
			for (int i = 1; i <= 2; i++)
			{
				int a;
				cin >> a;
				GetElem(L, a, e);
				cout << "第" << a << "个位置的数的值为:" << e << endl;
			}
			break;
		case 8:
			if (ClearList(&L)) cout << "置空成功" << endl;
			else cout << "置空失败" << endl;
			break;
		}
	}
}

✨链式存储

🎆一定别忘记生成新结点

🍔存储结构

typedef struct LNode{
	int data;			//数据域
	struct LNode *next; //指针域
}LNode,*LinkList;		

LinkList为指向结构体LNode的指针类型 

🚥🚥🚥🚥🚥🚥

⭐习惯上用LinkList定义单链表,强调的是某个单链表的头指针,用LNode*定义指向单链表中任意结点的指针变量

例如:

定义LinkList L,那么L为单链表的头指针

定义LNode *p,那么p为指向单链表某个结点的指针,用*p代表该节点

注意区分指针变量和结点变量两个不同的概念

例如

若定义LinkList p或LNode *p

p表示指向某个结点的指针变量,表示该结点的地址

*p表示对应的结点变量,表示该结点的名称 

这两种定义方式是等价的

🚥🚥🚥🚥🚥🚥

⭐易错:首元结点VS头节点VS头指针

头节点首元结点之前附设的结点

头指针是指向链表第一个结点的指针

(如果有头节点,那么头指针指向的结点为头节点)

(如果没有头节点,那么头指针指向的结点为首元结点)

【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会

🚥🚥🚥🚥🚥🚥 

⭐链表增加头结点的好处

🎈便于首元结点的处理

增加头结点后,首元结点的地址

🎈便于空表和非空表的统一处理

没有头结点,设L为单链表的头指针,L应该指向首元结点,那么当单链表为长度为0的空表时 ,L指针为空(L==NULL)

有头结点,无论链表是否为空,头指针都是指向头节点的非空指针,那么当头结点的指针域为空,(L->next==NULL)如下图所示 

【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会

 🚥🚥🚥🚥🚥🚥

顺序表中,由于相邻的两个元素在物理位置上相邻,那么每个元素的存储位置都可以从线性表的起始位置计算得到

单链表里面,各个元素的存储位置都是任意的,都是每个元素的存储位置都包含着其直接前驱结点,因为p->next=ai , p->next->next=ai+1 , 那么想要取得第i个数据元素必须从头指针出发顺链寻找

🚥🚥🚥🚥🚥🚥

🎈初始化

生成新结点作为头结点,用头指针 L 指向头结点

头结点的指针域置空

int InitList(LinkList &L)
{
	L=new LNode;
	L->next=NULL;
	return 1;
}
int InitList(LinkList *L)
{
	*L=new LNode;
	(*L)->next=NULL;//必须加上括号
	return 1;
}

为什么必须加上():优先级:->大于*

🎈尾插法建立单链表

1.创建一个只有头结点的空链表

2.尾指针 r 初始化,指向头结点

3.根据创建链表包括的元素个数n,循环n次执行以下操作

~生成一个新结点*p

~输入元素并把值赋给新结点*p的数据域

~将新结点*p插入尾结点*r后面

~尾指针r指向新的尾结点*p

(在前面说过,*X表示结点的名称)

【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会

 

int Create_back(LinkList &L,int num)
{
	L=new LNode;
	LNode *p,*r;		//先建立一个带有头节点的空链表 
	L->next=NULL;		//尾指针r指向头结点
	r=L;
	for(int i=0;i<num;i++)
	{
		p=new LNode;	//生成新结点
		cin>>p->data;
		p->next=NULL;
		r->next=p;		//新结点*p插入尾结点*r后(*X表示结点的名称)
		r=p;			//r指向新的尾结点*p
	}	
	return 1;
	
}

代码里面的 LNode *p,*r;还可以写为LinkList p,r;具体原因在上面说过了(就是在上面的⭐注意区分指针变量和结点变量两个不同的概念

int Create_back(LinkList *L,int num)
{
	*L=new LNode;
	LNode *p,*r;		//先建立一个带有头节点的空链表 
	(*L)->next=NULL;		//尾指针r指向头结点
	r=*L;
	for(int i=0;i<num;i++)
	{
		p=new LNode;	//生成新结点
		cin>>p->data;
		p->next=NULL;
		r->next=p;		//新结点*p插入尾结点*r后(*X表示结点的名称)
		r=p;			//r指向新的尾结点*p
	}	
	return 1;
	
}

🎈头插法建立单链表

1.创建一个只有头结点的空链表

2.根据创建链表包括的元素个数n,循环n次执行以下操作

~生成一个新结点*p

~输入元素并把值赋给新结点*p的数据域

~将新结点*p插入到头结点后面

【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会

 

int Create_front(LinkList &L,int num)
{
	L=new LNode;
	LNode *p;
	L->next=NULL;
	for(int i=0;i<num;i++)
	{
		p=new LNode;		//生成新结点*p
		cin>>p->data;
		p->next=L->next;
		L->next=p;			//将新结点*p插入到头结点后面
	}
	return OK;
}
int Create_front(LinkList *L,int num)
{
	*L=new LNode;
	LNode *p;
	(*L)->next=NULL;
	for(int i=0;i<num;i++)
	{
		p=new LNode;		//生成新结点*p
		cin>>p->data;
		p->next=(*L)->next;
		(*L)->next=p;			//将新结点*p插入到头结点后面
	}
	return OK;
}

🎈插入

注意:插入和前面的前插法尾插法不同,前插法尾插法是建立单链表,而插入是在已经建立好的单链表里面再加入额外的结点

具体步骤如下图所示

【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会

注意:插入操作必须要找到该位置的前驱结点 

这句话看上去理所应当,但是在某些时候是真的特别有用) 

int ListInsert(LinkList &L,int num,int place)
{
	LNode *p,*s;
	p=L;
	int j=0;
	while(p&&j<place-1)        //找到插入位置
	{
		p=p->next;
		j++;
	}
	if(!p||j>place-1) return ERROR;
	s=new LNode;			   //生成新结点
	s->data=num;			   //数据域赋值
							   
	s->next=p->next;		   //结点*s的指针域指向a2
	p->next=s;				   //结点*p的指针域指向*s
	return OK;
}
int ListInsert(LinkList *L,int num,int place)
{
	LNode *p,*s;
	p=*L;
	int j=0;
	while(p&&j<place-1)        //找到插入位置
	{
		p=p->next;
		j++;
	}
	if(!p||j>place-1) return ERROR;
	s=new LNode;			   //生成新结点
	s->data=num;			   //数据域赋值
							   
	s->next=p->next;		   //结点*s的指针域指向a2
	p->next=s;				   //结点*p的指针域指向*s
	return OK;
	
}

🎈删除

1.先找到待删除位置a的前驱结点

2.创建临时结点q,保存待删除结点的地址

3.将结点*p的指针域指向a的直接后继结点 

4.释放结点a的空间

【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会

int LinkDelete(LinkList &L,int place)
{
	LNode *p,*q;
	p=L;
	int j=0;
	while((p->next)&&(j<place-1))
	{
		p=p->next;
		++j;
	} 
	if(!(p->next)||(j>place-1) )return ERROR;
	q=p->next;				//创建临时结点q
	p->next=q->next;		//改变待删除结点的前驱结点的指针域
	delete q;				//释放被删除结点的空间
	return OK;
}
int LinkDelete(LinkList *L,int place)
{
	LNode *p,*q;
	p=*L;
	int j=0;
	while((p->next)&&(j<place-1))
	{
		p=p->next;
		++j;
	} 
	if(!(p->next)||(j>place-1) )return ERROR;
	q=p->next;				//创建临时结点q
	p->next=q->next;		//改变待删除结点的前驱结点的指针域
	delete q;				//释放被删除结点的空间
	return OK;
}

🎈取出元素

找到要取出的元素的位置,然后返回数据域即可

int ListPop(LinkList &L,int num)
{
	LNode *p;
	p=L;
	int j=0;
	while(p&&j<num)
	{
		p=p->next;
		j++;
	}
	if(p)
	return p->data;
	else
	return -1;
	
}
int ListPop(LinkList *L,int num)
{
	LNode *p;
	p=*L;
	int j=0;
	while(p&&j<num)
	{
		p=p->next;
		j++;
	}
	if(p)
	return p->data;
	else
	return -1;
	
}

 

🎈输出链表

void OutputList(LinkList &L)
{
	LNode *p;
	p=L->next;
	while(p)
	{
		cout<<p->data<<' ';
		p=p->next;
	}
	cout<<endl;
}
void OutputList(LinkList *L)
{
	LNode *p;
	p=(*L)->next;
	while(p)
	{
		cout<<p->data<<' ';
		p=p->next;
	}
	cout<<endl;
}

😎完整代码1

#include<iostream>
using namespace std;
#define OK 1
#define ERROR -1
typedef struct LNode
{
	int data;
	struct LNode *next;
	
}LNode,*LinkList;


//1
int InitList(LinkList *L)
{
	*L=new LNode;
	(*L)->next=NULL;
	return OK;
}

int Create_front(LinkList *L,int num)
{
	*L=new LNode;
	LNode *p;
	(*L)->next=NULL;
	for(int i=0;i<num;i++)
	{
		p=new LNode;		//生成新结点*p
		cin>>p->data;
		p->next=(*L)->next;
		(*L)->next=p;			//将新结点*p插入到头结点后面
	}
	return OK;
}

//3
int ListInsert(LinkList *L,int num,int place)
{
	LNode *p,*s;
	p=*L;
	int j=0;
	while(p&&j<place-1)        //找到插入位置
	{
		p=p->next;
		j++;
	}
	if(!p||j>place-1) return ERROR;
	s=new LNode;			   //生成新结点
	s->data=num;			   //数据域赋值
							   
	s->next=p->next;		   //结点*s的指针域指向a2
	p->next=s;				   //结点*p的指针域指向*s
	return OK;
	
}

//4
int LinkDelete(LinkList *L,int place)
{
	LNode *p,*q;
	p=*L;
	int j=0;
	while((p->next)&&(j<place-1))
	{
		p=p->next;
		++j;
	} 
	if(!(p->next)||(j>place-1) )return ERROR;
	q=p->next;				//创建临时结点q
	p->next=q->next;		//改变待删除结点的前驱结点的指针域
	delete q;				//释放被删除结点的空间
	return OK;
}

//5
int ListPop(LinkList *L,int num)
{
	LNode *p;
	p=*L;
	int j=0;
	while(p&&j<num)
	{
		p=p->next;
		j++;
	}
	if(p)
	return p->data;
	else
	return -1;
	
}

void OutputList(LinkList *L)
{
	LNode *p;
	p=(*L)->next;
	while(p)
	{
		cout<<p->data<<' ';
		p=p->next;
	}
	cout<<endl;
}

int main()
{
	LinkList L;
	int n,e;
	cout<<"请输入你的操作"<<endl;
	printf("0:结束程序\n1:建立空表\n2:尾插法建立空表\n3:插入\n4:删除\n5:取出元素\n");
	for(;;)
	{
		cin>>n;
		switch(n)
		{
			case 0:
				return 0;
				break;
			case 1:
				cout<<"建立空表"<<endl;
				if(InitList(&L))
				{
					cout<<"建立成功"<<endl;
				}
				else
				{
					cout<<"建立失败"<<endl;
				}
				break;
			case 2:
				cout<<"输入21 18 30 75 42 56,使用尾插法建立链表"<<endl;
				cout<<"请输入要加入的数字的数量"<<endl;
				int num;
				cin>>num;
				if(Create_front(&L,num)!=0)
				{
					 cout<<"请进行下一步操作"<<endl;
					 OutputList(&L); 
					 
				}
				else cout<<"失败了"<<endl;
				break;
			case 3:
				cout<<"在第3个位置插入67,在第9个位置插入10"<<endl;
				cout<<"请输入需要插入的数字的个数"<<endl;
				int qqq;
				cin>>qqq;
				for(int i=0;i<qqq;i++)
				{
					cout<<"需要插入的数据和位置"<<endl;
					int nn,mm;
					cin>>nn>>mm;
					if(ListInsert(&L,nn,mm))
					{
						cout<<"插入成功"<<endl;
						OutputList(&L);
					
					}
					else
					{
						cout<<"插入失败"<<endl;
					}
				}
				break;
			case 4:
				cout<<"删除第6个元素和第8个元素"<<endl;
				cout<<"输入需要删除的元素是哪一个"<<endl;
				int qq;
				cin>>qq;
				if(LinkDelete(&L,qq))
				{
					cout<<"删除成功"<<endl;
					OutputList(&L);
				}	
				else
				{
					cout<<"删除失败"<<endl;
				}
				break;
			case 5:
				cout<<"取出第5个元素和第7个元素"<<endl;
				cout<<"需要取出多少个数"<<endl;
				int cnt;
				cin>>cnt;
				for(int i=0;i<cnt;i++)
				{
					cout<<"需要取出哪个位置的元素"<<endl;
					int _;
					cin>>_;
					if(ListPop(&L,_)==-1) cout<<"取出元素失败"<<endl;
					else  cout<<ListPop(&L,_)<<endl;
				}
				break;
				
		}
	}
}

😎完整代码2 

#include<iostream>
using namespace std;
#define OK 1
#define ERROR -1
typedef struct LNode
{
	int data;
	struct LNode *next;
	
}LNode,*LinkList;


//1
int InitList(LinkList &L)
{
	L=new LNode;
	L->next=NULL;
	return OK;
}

//2
int Create_back(LinkList &L,int num)
{
	L=new LNode;//头节点 
	LNode *p,*r;
	L->next=NULL;
	r=L;
	for(int i=0;i<num;i++)
	{
		p=new LNode;
		cin>>p->data;
		p->next=NULL;
		r->next=p;
		r=p;	
	}	
	return OK;
	
}

//3
int ListInsert(LinkList &L,int num,int place)
{
	LNode *p,*s;
	p=L;
	int j=0;
	while(p&&j<place-1)
	{
		p=p->next;
		j++;
	}
	if(!p||j>place-1) return ERROR;
	s=new LNode;
	s->data=num;
	s->next=p->next;
	p->next=s;
	return OK;
	
}

//4
int LinkDelete(LinkList &L,int place)
{
	LNode *p,*q;
	p=L;
	int j=0;
	while((p->next)&&(j<place-1))
	{
		p=p->next;
		++j;
	} 
	if(!(p->next)||(j>place-1) )return ERROR;
	q=p->next;
	p->next=q->next;
	//e=p->data;
	delete q;
	return OK;
}

//5
int ListPop(LinkList &L,int num)
{
	LNode *p;
	p=L;
	int j=0;
	while(p&&j<num)
	{
		p=p->next;
		j++;
	}
	if(p)
	return p->data;
	else
	return -1;
	
}

void OutputList(LinkList &L)
{
	LNode *p;
	p=L->next;
	while(p)
	{
		cout<<p->data<<' ';
		p=p->next;
	}
	cout<<endl;
}

int main()
{
	LinkList L;
	int n,e;
	cout<<"请输入你的操作"<<endl;
	printf("0:结束程序\n1:建立空表\n2:尾插法建立空表\n3:插入\n4:删除\n5:取出元素\n");
	for(;;)
	{
		cin>>n;
		switch(n)
		{
			case 0:
				return 0;
				break;
			case 1:
				cout<<"建立空表"<<endl;
				if(InitList(L))
				{
					cout<<"建立成功"<<endl;
				}
				else
				{
					cout<<"建立失败"<<endl;
				}
				break;
			case 2:
				cout<<"输入21 18 30 75 42 56,使用尾插法建立链表"<<endl;
				cout<<"请输入要加入的数字的数量"<<endl;
				int num;
				cin>>num;
				if(Create_back(L,num)!=0)
				{
					 cout<<"请进行下一步操作"<<endl;
					 OutputList(L); 
					 
				}
				else cout<<"失败了"<<endl;
				break;
			case 3:
				cout<<"在第3个位置插入67,在第9个位置插入10"<<endl;
				cout<<"请输入需要插入的数字的个数"<<endl;
				int qqq;
				cin>>qqq;
				for(int i=0;i<qqq;i++)
				{
					cout<<"需要插入的数据和位置"<<endl;
					int nn,mm;
					cin>>nn>>mm;
					if(ListInsert(L,nn,mm))
					{
						cout<<"插入成功"<<endl;
						OutputList(L);
					
					}
					else
					{
						cout<<"插入失败"<<endl;
					}
				}
				break;
			case 4:
				cout<<"删除第6个元素和第8个元素"<<endl;
				cout<<"输入需要删除的元素是哪一个"<<endl;
				int qq;
				cin>>qq;
				if(LinkDelete(L,qq))
				{
					cout<<"删除成功"<<endl;
					OutputList(L);
				
				}	
				else
				{
					cout<<"删除失败"<<endl;
				}
				break;
			case 5:
				cout<<"取出第5个元素和第7个元素"<<endl;
				cout<<"需要取出多少个数"<<endl;
				int cnt;
				cin>>cnt;
				for(int i=0;i<cnt;i++)
				{
					cout<<"需要取出哪个位置的元素"<<endl;
					int _;
					cin>>_;
					if(ListPop(L,_)==-1) cout<<"取出元素失败"<<endl;
					else  cout<<ListPop(L,_)<<endl;
				}
				break;
				
		}
	}
} 

 🥰如果大家有不明白的地方,或者文章有问题,欢迎大家在评论区讨论,指正🥰

Code over!

 

到了这里,关于【数据结构】线性表(顺序存储和链式存储)两种方法,细节满满,保你学会的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • 数据结构-线性表的链式存储(包含代码实现)

    数据结构-线性表的链式存储(包含代码实现)

    链式存储结构 结点在存储器中的位置是任意的,即逻辑上相邻的数据元素在物理上不一定相邻 线性表的链式存储结构又称为 非顺序映像 或 链式映像。 用一组 物理位置任意的存储单元 来存放线性表的数据元素 这组存储单元既可以是连续的,也可以是不连续的,甚至是零散

    2024年02月06日
    浏览(11)
  • 数据结构:线性表顺序存储结构———顺序表

    目录 顺序表的定义 初始化线性表 销毁线性表 求线性表的长度 判断是否为空表 插入数据元素 逻辑序号与物理序号的区别 删除数据元素 输出线性表  按序号求线性表中的元素 按元素值查找 整体上创建顺序表 顺序表实验 线性表的顺序存储是把线性表中的所有元素按照其逻辑

    2024年02月03日
    浏览(11)
  • C/C++数据结构---顺序表---线性存储结构

    C/C++数据结构---顺序表---线性存储结构

    个人主页: 仍有未知等待探索_小项目,洛谷刷题,数据结构-CSDN博客 专题分栏---数据结构: 数据结构_仍有未知等待探索的博客-CSDN博客 目录 一、知识储备 二、引例  三、顺序表 第一步,先创建一个顺序表类型 第二步,定义和初始化顺序表    第三步,顺序表的基本操作

    2024年02月08日
    浏览(11)
  • 【数据结构】线性表的顺序存储结构及实现——C语言版

    【数据结构】线性表的顺序存储结构及实现——C语言版

    线性表的顺序存储结构称为 顺序表 ,其基本思想是 用一段地址连续的存储单元一次存储线性表的数据元素。 设顺序表的每个元素占用 c 个存储单元,则第 i 个元素的存储地址为: 所以, 只要确定了存储顺序表的起始地址(即基地址),计算任意一个元素的存储地址的时间

    2024年03月15日
    浏览(17)
  • 【数据结构与算法分析】使用C语言实现队列的两种(带头结点与不带头结点)链式存储,并且给出一种循环队列的设计思想

    【数据结构与算法分析】使用C语言实现队列的两种(带头结点与不带头结点)链式存储,并且给出一种循环队列的设计思想

      当我们编写程序时,经常需要处理各种数据结构。队列是一种常见的数据结构,它有着广泛的应用场景。队列的基本操作包括入队和出队,应用于模拟等待队列、消息队列、计算机缓存等场合。   在实际编程中,我们可以用不同的数据结构来实现队列。本文主要介绍了

    2024年02月08日
    浏览(205)
  • 【数据结构】(顺序表)C语言实现线性表顺序存储的创建、插入、删除、查找、输出等基本操作(附完整代码)

    要求:利用书本上的线性表的顺序存储结构定义 #define MAXSIZE 100 //顺序表可能达到的最大长度 typedef struct{ ElemType *elem; // 存储空间基址 int length; // 当前长度 int listsize; // 当前分配的存储容量(以sizeof(ElemType)为单位) } SqList; 1)编写完成下列功能的函数: (1)初始化一个线性表

    2024年04月28日
    浏览(41)
  • 数据结构中: 一元多项式的运算(相加,相减,相乘)------用C语言 / C++来实现。 数据结构线性表的操作和应用(顺序存储)

    数据结构中: 一元多项式的运算(相加,相减,相乘)------用C语言 / C++来实现。 数据结构线性表的操作和应用(顺序存储)

    线性表的操作和应用(顺序存储)。用顺序存储实现一元多项式,并进行加、减、乘运算。 (1)一元多项式结构体创建  (2)初始化 (3)一元多项式赋值             (4)打印一元多项式 (5)加法运算                        (6)减法运算 (7)乘法运算    全部代

    2024年02月01日
    浏览(40)
  • 数据结构(顺序结构、链式结构、索引结构、散列结构)

    数据结构(顺序结构、链式结构、索引结构、散列结构)

    数据结构,就是一种程序设计优化的方法论,研究数据的逻辑结构和物理结构以及它们之间相互关系,并对这种结构定义相应的运算, 目的是加快程序的执行速度、减少内存占用的空间 。 数据的逻辑结构指反映数据元素之间的逻辑关系,而与数据的存储无关,是独立于计算

    2024年02月03日
    浏览(12)
  • 【脚踢数据结构】队列(顺序和链式)

    【脚踢数据结构】队列(顺序和链式)

    (꒪ꇴ꒪ ),Hello我是 祐言QAQ 我的博客主页:C/C++语言,Linux基础,ARM开发板,软件配置等领域博主🌍 快上🚘,一起学习,让我们成为一个强大的攻城狮! 送给自己和读者的一句鸡汤🤔: 集中起来的意志可以击穿顽石! 作者水平很有限,如果发现错误,可在评论区指正,感谢🙏

    2024年02月12日
    浏览(9)
  • 数据结构:线性表的链式储存

    数据结构:线性表的链式储存

     🌈个人主页:Rookie Maker 🔥 系列专栏:数据结构 🏆🏆关注博主,随时获取更多关于IT的优质内容!🏆🏆   😀欢迎来到我的代码世界~ 😁 喜欢的小伙伴记得一键三连哦 ૮(˶ᵔ ᵕ ᵔ˶)ა ​ 链表:线性表的链式储存方式,逻辑结构不一定连续,物理结构不一定连续 描述

    2024年04月27日
    浏览(12)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包