剑指offer面试题13扩展:带头指针的单链表的操作

本文介绍了一种带有头指针的链表实现方法,并详细探讨了三种不同时间复杂度的链表节点删除算法,包括平均时间复杂度为O(1)的高效删除方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

代码:
//带有头指针的链表操作
//2014.7.6

#include<iostream>
#include<cstdlib>
using namespace std;
typedef struct node
{
	int data;
	struct node *next;
}Node;

void insert_node(Node **pphead,int v)
{
	if(*pphead==NULL)
	{
		*pphead=(Node *)malloc(sizeof(Node));
		(*pphead)->data=v;
		(*pphead)->next=NULL;
	}
	else
	{
		Node *t=*pphead;
		*pphead=(Node *)malloc(sizeof(Node));
		(*pphead)->data=v;
		(*pphead)->next=t;
	}
}
//--------------------------------------------------------------------------------------------------------平均时间复杂度为O(1)的节点删除算法
void delete_node1(Node **pphead,int v)//参考剑指offer ,面试题13,酷壳linus
{
	Node *pNode=*pphead;
	while(pNode!=NULL)
	{
		if(pNode->data==v)
			break;
		else
			pNode=pNode->next;
	}
	if(pNode==NULL)//遍历整个链表,没有找到对应节点
	{  
		cout<<"No node with v value exist"<<endl;
	}
    //要找的节点是尾节点(或者当链表中只有一个节点,且该节点就是要找的那个节点时)时,
	//由于无法再用“将要删除节点A的后继节点B的内容赋值给A,并将A的next指针指向B的后继节点,然后free(B)”,
	//只能用顺序查找的方法找到A的前驱节点C,让C指向A的后继节点
	else if(pNode->next==NULL)
	{
		Node *pre=*pphead;
		while(pre->next!=pNode)
		{
			pre=pre->next;
		}
		pre->next=NULL;
		free(pNode);
		pNode=NULL;
	}
	//当链表中的节点不是尾结点时,“将要删除节点A的后继节点B的内容赋值给A,并将A的next指针指向B的后继节点,然后free(B)”
	else
	{
		Node *pNext=pNode->next;
		pNode->next=pNext->next;
		pNode->data=pNext->data;
		free(pNext);
		pNext=NULL;
	}
}
//---------------------------------------------------------------------------------------------------------------------------------------

//------------------------------------------------------------------------------------------------------------------------时间复杂度为O(n),比较经典的的一种方法
void delete_node2(Node **pphead,int v)
{
	Node *cur=*pphead;
	Node *pre=NULL;
	if(cur==NULL)
	{
		cout<<"Now ,the link is empty!"<<endl;
		return;
	}
	for(  ;cur;pre=cur,cur=cur->next)
	{
		if(cur->data==v)
			break;
	}
	if(pre==NULL)//要删除的是第一个节点
	{
		//Node *pnode=*pphead;
		*pphead=cur->next;
		free(cur);
	}
	else
	{
		pre->next=cur->next;
		free(cur);
	}
}
//---------------------------------------------------------------------------------------------------------------------------------------

//----------------------------------------------------------------------------------------利用二级指针进行链表节点的删除,时间复杂度为O(1)
//---*********************************该方法是效率最高的一种删除链表节点的方法,利用二级指针进行链表的**************************************
void delete_node3(Node **pphead,int v)
{
	Node **cur=pphead;
	for(cur=pphead;*cur;)
	{
		Node *en=*cur;
		if(en->data==v)
		{
			*cur=en->next;
			free(en);
			break;
		}
		else
		{
			cur=&en->next;
		}
	}
}
void print_node(Node **pphead)
{
	for(Node *cur=*pphead;cur;cur=cur->next)
		cout<<cur->data<<" ";
	cout<<endl;
}
int main(void)
{
	Node *firNode=NULL;
	for(int i=1;i<=10;i++)
		insert_node(&firNode,i);
	print_node(&firNode);
	delete_node1(&firNode,10);
	delete_node2(&firNode,1);
	delete_node3(&firNode,3);
	delete_node3(&firNode,9);
	print_node(&firNode);
}

资源下载链接为: https://pan.quark.cn/s/22ca96b7bd39 在当今的软件开发领域,自动化构建与发布是提升开发效率和项目质量的关键环节。Jenkins Pipeline作为一种强大的自动化工具,能够有效助力Java项目的快速构建、测试及部署。本文将详细介绍如何利用Jenkins Pipeline实现Java项目的自动化构建与发布。 Jenkins Pipeline简介 Jenkins Pipeline是运行在Jenkins上的一套工作流框架,它将原本分散在单个或多个节点上独立运行的任务串联起来,实现复杂流程的编排与可视化。它是Jenkins 2.X的核心特性之一,推动了Jenkins从持续集成(CI)向持续交付(CD)及DevOps的转变。 创建Pipeline项目 要使用Jenkins Pipeline自动化构建发布Java项目,首先需要创建Pipeline项目。具体步骤如下: 登录Jenkins,点击“新建项”,选择“Pipeline”。 输入项目名称和描述,点击“确定”。 在Pipeline脚本中定义项目字典、发版脚本和预发布脚本。 编写Pipeline脚本 Pipeline脚本是Jenkins Pipeline的核心,用于定义自动化构建和发布的流程。以下是一个简单的Pipeline脚本示例: 在上述脚本中,定义了四个阶段:Checkout、Build、Push package和Deploy/Rollback。每个阶段都可以根据实际需求进行配置和调整。 通过Jenkins Pipeline自动化构建发布Java项目,可以显著提升开发效率和项目质量。借助Pipeline,我们能够轻松实现自动化构建、测试和部署,从而提高项目的整体质量和可靠性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值