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

前言

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

头文件

#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.
全部评论

相关推荐

行云流水1971:这份实习简历的优化建议: 结构清晰化:拆分 “校园经历”“实习经历” 板块(当前内容混杂),按 “实习→校园→技能” 逻辑排版,求职意向明确为具体岗位(如 “市场 / 运营实习生”)。 经历具象化:现有描述偏流程,需补充 “动作 + 数据”,比如校园活动 “负责宣传” 可加 “运营公众号发布 5 篇推文,阅读量超 2000+,带动 300 + 人参与”;实习内容补充 “协助完成 XX 任务,效率提升 X%”。 岗位匹配度:锚定目标岗位能力,比如申请运营岗,突出 “内容编辑、活动执行” 相关动作;申请市场岗,强化 “资源对接、数据统计” 细节。 信息精简:删减冗余表述(如重复的 “负责”),用短句分点,比如 “策划校园招聘会:联系 10 + 企业,组织 200 + 学生参与,到场率达 85%”。 技能落地:将 “Office、PS” 绑定经历,比如 “用 Excel 整理活动数据,输出 3 份分析表;用 PS 设计 2 张活动海报”,避免技能单独罗列。 优化后需强化 “经历 - 能力 - 岗位需求” 的关联,让实习 / 校园经历的价值更直观。 若需要进一步优化服务,私信
实习,投递多份简历没人回...
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务