判断一个单链表是否带环,并求出环的长度,及环的入口点

本文介绍了一种使用快慢指针检测单链表中是否存在环的方法,并详细解释了如何确定环的长度及环的入口节点。通过数学推导,阐述了快慢指针相遇时环的具体特性。

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

单链表是否带环?

用快慢指针分别遍历这个单链表,如果快慢指针能相遇,就代表有环的存在。

环的长度?

当快慢指针相遇时,记录当前位置,然后走一圈的长度就是环的长度。

环的入口点?

设快指针F走两个结点,慢指针S走一个结点。

F=2*S;                 //快指针是慢指针走的两倍

S=L1+a;           //慢指针相遇时走的长度

F=L1+n*L2+a;    //快指针相遇时走的长度

通过前三个式子得:n*L2=L1+a        即    L1=(n-1)L2+(L2-a)

                                     环外的链表长度=环长度的整数倍+(环长度-快慢指针相遇点到入口点的距离)

 

 

 

int HasListCircle(pList* pHead)//有环?
{
	assert(pHead != NULL);
	pList phead = *pHead;
	while (phead->next != NULL)
	{
		phead = phead->next;
	}
	phead->next = *pHead;//建环
	pList fast = *pHead;
	pList slow = *pHead;
	while (fast&&slow)
	{
		fast = fast->next->next;
		slow = slow->next;
		if (slow == fast)//快慢指针相遇
			return 1;
	}
	return 0;
}
// 求环的长度 
int GetCircleLen(pList* pHead)
{
	assert(pHead != NULL);
	int count = 0;
	pList phead = *pHead;
	
	pList cur = NULL;
	while (phead->next != NULL)
	{
		phead = phead->next;
	}
	phead->next = *pHead;//建环
	pList fast = *pHead;
	pList slow = *pHead;
	while (fast&&slow)
	{
		fast = fast->next->next;
		slow = slow->next;
		if (slow == fast)
		{
			cur = slow;//快慢指针相遇,标记相遇点
			break;
		}
	}
	while (cur->next != slow)
	{
		cur = cur->next;
		count++;
	}
	return count + 1;
}

int CircleEnter(pList* pHead)//环的入口
{
	assert(pHead);
	int count = 0;
	pList pre = *pHead;
	pList cur = *pHead;
	pList cur1 = *pHead;
	pList fast = *pHead;
	pList slow = *pHead;
	while (cur->next != NULL)
	{
		cur = cur->next;
	}
	while (cur1->data != cur->data / 2)
	{
		cur1 = cur1->next;
	}
	cur->next = cur1;//创建一个带环的链表,并且环外也有链表,
	while (fast&&slow)
	{
		fast = fast->next->next;
		slow = slow->next;
		if (slow == fast)
		{
			cur = slow;//找到相遇点,并且标记
			break;
		}
	}
	while (pre != cur)
	{
		pre = pre->next;//从头开始走
		cur = cur->next;//从相遇点开始走
	}
	return cur->data;//当相遇得时候,就是入口点
}

    

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值