实验二 不带头结点的单链表

前言

熟练掌握动态链表结构及有关算法的设计方法

头文件

#include <stdio.h>
#include <stdlib.h>

/* 链表实现的头文件,文件名slnklist.h */
 typedef int datatype;
 typedef struct link_node{
   datatype info;
   struct link_node *next;
 }node;
typedef node *linklist;

/*头插法建立单链表*/

linklist creatbystack()
{  linklist  head,s;
    datatype x;
    head=NULL;
    printf("请输入若干整数序列:\n");
    scanf("%d",&x);
    while (x!=0)		/*以0结束输入*/
    {   s=(linklist)malloc(sizeof(node));  /*生成待插入结点*/
        s->info=x;
        s->next=head;			/*将新结点插入到链表最前面*/
        head=s;
        scanf("%d",&x);
    }
    return head;				/*返回建立的单链表*/
}

/*函数功能:尾插法建立单链表   */
linklist creatbyqueue()
{
    linklist head,r,s;
    datatype x;
    head=r=NULL;
    printf("请输入若干整数序列:\n");
    scanf("%d",&x);
    while (x!=0) /*以0结束输入*/
    {    s=(linklist)malloc(sizeof(node));
         s->info=x;
         if (head==NULL)		/*将新结点插入到链表最后面*/
            head=s;
         else
            r->next=s;
        r=s;
        scanf("%d",&x);
   }
    if (r)  r->next=NULL;
    return head;					/*返回建立的单链表*/
}


/*函数功能:输出不带头结点的单链表      */

void print(linklist head)
{   linklist p;
    int i=0;
    p=head;
    printf("List is:\n");
    while(p)
    {
        printf("%5d",p->info);
        p=p->next;
         i++;
		 if (i%10==0) printf("\n");
    }
    printf("\n");
}

/*释放不带头结点的单链表 */
void delList(linklist head)
{ linklist p=head;
  while (p)
  { head=p->next;
    free(p);
    p=head;
  }
}


1.删除不带头结点的单链表的第一个值为x的节点

#include"slnklist.h"
linklist delx(linklist head, datatype x)
{
	linklist p,pre;
	if (!head)
	{
		printf("链表为空!\n");
	}
	p = head;
	pre = NULL;
	while (p&&p->info != x)
	{
		pre = p;
		p = p->next;
	}
	if (!p)
	{
		printf("链表中没有%d这个值的节点!\n", x);
	}
	else {
		if (pre==NULL)
		{
			head = NULL;
			free(p);
		}
		else
		{
			pre->next = p->next;
			free(p);
		}
	}
	return head;
}
int main()
{
	datatype x;
	linklist head;
	head = creatbyqueue();
	print(head);
	printf("请输入要删除的值:");
	scanf("%d", &x);
	head = delx(head, x);
	print(head);
	delList(head);
	system("pause");
	return 0;
}

2.倒置单链表

#include"slnklist.h"
linklist reverse1(linklist head)
{
	//这里注意linklist就是node *
	/*法一,最容易想到的,也是最傻的,用两个for循环反复交换信息域的值,*/
	linklist p;
	p = head;
	int n = 0;
	int m;
	while (p) { p = p->next; n++; }//得出总结点个数
	m = n;
	p = head;//重新上链
	for (int i = 0; i < n - 1; i++)
	{
		for (int i = 0; i < m-1; i++)
		{
			datatype temp = p->info;
			p->info = p->next->info;
			p->next->info = temp;
			p = p->next;
		}
		m--;
		p = head;
	}
	return head;
}
void reverse2(linklist *head)
{
	/*之后想到的法2,通过改变head指向原来链表最后一个节点,然后只要更改每个节点的指针域即可*/
	linklist p, pre;
	p = *head;
	*head = NULL;//注意这里是为了将原来链表的指针域设为空指针
	while (p)
	{
		pre = p;
		p = p->next;
		pre->next = *head;
		*head = pre;
	}
}
int main()
{
	datatype x;
	linklist head;
	head = creatbystack();
	print(head);
	head = reverse1(head);
	printf("reverse1:\n");
	print(head);
	printf("reverse2:\n");
	reverse2(&head);
	print(head);
	delList(head);
	system("pause");
	return 0;
}

3.插入新节点后依旧保持原链表的有序性

#include"slnklist.h"
linklist insert(linklist head, datatype x)
{
	/*法一*/
	linklist q, p, pre;
	q = (linklist)malloc(sizeof(linklist));
	q->info = x;
	pre = NULL;
	p = head;
	while (p)
	{
		pre = p;
		p = p->next;
		if (p)
		{
			if (pre->info <= q->info)
			{
				if (q->info <= p->info)//插入的节点作为中间节点
				{
					q->next = p;
					pre->next = q;
					break;
				}
			}
			else//插入的节点作为头结点
			{
				q->next = pre;
				head = q;
				break;
			}

		}
		else//插入的节点作为尾节点
		{
			if(pre->info>=q->info)//只有一个节点,而插入的节点值小于它
			{
				q->next = pre;
				head = q;
				break;
			}
			pre->next = q;
			q->next = NULL;
		}
	}
	return head;
}
/*
法二
*/
linklist insert2(linklist head, datatype x)
{
	linklist pre, p, q;
	q = (linklist)malloc(sizeof(linklist));
	q->info = x;
	pre = NULL;
	p = head;
	while (p&&p->info < q->info)
	{
		pre = p;
		p = p->next;
	}
	if (pre == NULL)//插入的节点值做头结点
	{
		q->next = p;
		head = q;
	}
	else//插入的节点做中间节点或尾节点
	{
		pre->next = q;
		q->next = p;
	}
	return head;
}
int main()
{
	datatype x;
	linklist head;
	printf("输入一组升序排序的数:\n");
	head = creatbyqueue();
	print(head);
	printf("请输入要插入的值:\n");
	scanf("%d", &x);
	head = insert2(head, x);
	print(head);
	//delList(head);
	system("pause");
}

4.删除不带头节点中所有值为x的节点

#include"slnklist.h"
linklist delallx(linklist head, datatype x)
{
	//相对于删除第一个值为x的节点来说,加个while循环和p继续下滑即可
	linklist  pre, p;
	pre = NULL;
	p = head;
	while (p)
	{
		while (p &&p->info != x)            
		{
			pre = p;
			p = p->next;
		}
		if (p)                                       
		{
			if (pre == NULL)                 //删除的结点为第一个结点
			{
				head = p->next;
				free(p);
				p = head;
			}
			else                                    //删除的结点不是第一个结点
			{
				pre->next = p->next;
				free(p);
				p = pre->next;
			}
		}

	}
	return head;
}
int main()
{
	datatype x;
	linklist head;
	head = creatbyqueue();
	print(head);
	printf("请输入要删除的值:");
	scanf("%d", &x);
	head = delallx(head, x);
	print(head);
	delList(head);
	system("pause");
}

后记

Afraid to miss will never meet such a person.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

逝不等琴生

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值