输入一个单向链表,输出该链表中倒数第K个结点

    输入一个单向链表,输出该链表中倒数第K个结点,具体实现如下:

#include <iostream>
using namespace std;
struct LinkNode
{
public:
	LinkNode(int value = 0) : nValue(value){ pNext = NULL; }
	~LinkNode() { pNext = NULL; }
	//private:
	friend class LinkList;
	int nValue;
	LinkNode *pNext;
};

//带头结点的链表
class LinkList
{
public:
	LinkList();
	~LinkList();
	void Insert(int nData); //插入元素
	void Delete(int nData); //删除元素
	void Sort();    //排序
	bool IsEmpty(); //判断是否为空
	void Reverse(); //反转
	void Destroy(); //销毁整个链表
	int Length();   //链表长度
	LinkNode* Find(int nData);  //查找元素
	bool IsLoop();  //判断是否为环
	void Print();   //打印链表
	LinkNode *FindByIndex(int n);   //返回链表中第i个元素
	LinkNode *FindLastNum(int n);   //使用两个指针 高效操作
private:
	LinkNode *pHead;
};

LinkList::LinkList()
{
	pHead = new LinkNode();
}
LinkList::~LinkList()
{
	Destroy();
}
//从大到小插入
void LinkList::Insert(int nData)
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return;
	}
	LinkNode *pCur = pHead;
	while (pCur->pNext != NULL)
	{
		if (pCur->pNext->nValue < nData)
		{
			break;
		}
		pCur = pCur->pNext;
	}
	LinkNode *pTmp = new LinkNode(nData);
	pTmp->pNext = pCur->pNext;
	pCur->pNext = pTmp;
}

void LinkList::Delete(int nData)
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return;
	}
	if (pHead->pNext == NULL)
	{
		cout << "链表为空!" << endl;
		return;
	}

	LinkNode *pCur = pHead;
	while (pCur->pNext)
	{
		if (pCur->pNext->nValue == nData)
		{
			LinkNode *pDel = pCur->pNext;
			pCur->pNext = pCur->pNext->pNext;

			pDel->pNext = NULL;
			delete (pDel);
		}
		else
		{
			pCur = pCur->pNext;
		}
	}
}

void LinkList::Sort()
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return;
	}
	if (pHead->pNext == NULL)
	{
		cout << "链表为空!" << endl;
	}

	LinkNode *pCur = pHead->pNext;
	LinkNode *pPre = pHead;

	LinkNode *pNewHead = new LinkNode();
	while (pCur)
	{
		LinkNode *pTmp = pCur;
		pCur = pCur->pNext;

		//将pTmp结点插入到pNewHead指向的新链表中
		LinkNode *p = pNewHead;
		while (p->pNext)
		{
			if (p->pNext->nValue > pTmp->nValue)
			{
				break;
			}
			p = p->pNext;
		}
		pTmp->pNext = p->pNext;
		p->pNext = pTmp;
	}
	delete pHead;
	pHead = pNewHead;
}
bool LinkList::IsEmpty()
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return false;
	}

	return pHead->pNext == NULL;
}

int LinkList::Length()
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return 0;
	}
	int nLength = 0;
	LinkNode *pCur = pHead->pNext;
	while (pCur)
	{
		nLength++;
		pCur = pCur->pNext;
	}

	return nLength;
}
void LinkList::Reverse()
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
	}
	if (pHead->pNext == NULL)
	{
		cout << "链表为空!" << endl;
	}
	else
	{
		LinkNode *pCur = pHead->pNext->pNext;
		LinkNode *pPre = pHead->pNext;
		LinkNode *pnext = NULL;

		while (pCur)
		{
			pnext = pCur->pNext;
			pCur->pNext = pPre;

			pPre = pCur;
			pCur = pnext;
		}

		pHead->pNext->pNext = NULL;
		pHead->pNext = pPre;
	}
}
void LinkList::Destroy()
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return;
	}

	while (pHead->pNext)
	{
		LinkNode *pDel = pHead->pNext;
		pHead->pNext = pDel->pNext;
		delete pDel;
	}
	delete pHead;
	pHead = NULL;
}
LinkNode* LinkList::Find(int nData)
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return NULL;
	}
	if (pHead->pNext == NULL)
	{
		cout << "链表为空!" << endl;
		return NULL;
	}

	LinkNode *pCur = pHead->pNext;
	while (pCur != NULL)
	{
		if (pCur->nValue == nData)
		{
			return pCur;
		}
	}

	return NULL;
}
void LinkList::Print()
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return;
	}
	if (pHead->pNext == NULL)
	{
		cout << "链表为空!" << endl;
		return;
	}
	else
	{
		LinkNode *pCur = pHead->pNext;
		while (pCur)
		{
			cout << pCur->nValue << " ";
			pCur = pCur->pNext;
		}
		cout << endl;
	}
}
bool LinkList::IsLoop()
{
	if (pHead == NULL)
	{
		cout << "链表未创建!" << endl;
		return false;
	}
	if (pHead->pNext == NULL)
	{
		cout << "链表为空!" << endl;
		return false;
	}

	LinkNode *pFast = pHead->pNext;
	LinkNode *pLow = pHead->pNext;

	while (pFast != NULL && pLow != NULL  && pFast->pNext != NULL)
	{
		pFast = pFast->pNext->pNext;
		pLow = pLow->pNext;
		if (pFast == pLow)
		{
			return true;
		}
	}
	return false;
}

LinkNode* LinkList::FindByIndex(int n)
{
	if (n <= 0 || n > Length())
	{
		return NULL;
	}
	LinkNode *pCur = pHead->pNext;
	for (int i = 1; i < n; i++)
	{
		pCur = pCur->pNext;
	}

	return pCur;
}

LinkNode* LinkList::FindLastNum(int n)  //使用两个指针 高效操作
{
	if (pHead == NULL || pHead->pNext == NULL || n < 0)
	{
		return NULL;
	}
	LinkNode *pCur = pHead->pNext;
	LinkNode *pFirst = pHead->pNext;

	while (n > 0)
	{
		pFirst = pFirst->pNext;
		if (pFirst == NULL)
		{
			return NULL;
		}
		n--;
	}
	while (pFirst->pNext)
	{
		pFirst = pFirst->pNext;
		pCur = pCur->pNext;
	}

	return pCur;
}
int main()
{
	LinkList list;
	list.Insert(12);
	list.Insert(14);
	list.Insert(2);
	list.Insert(122);
	list.Insert(5);
	list.Insert(4);
	list.Insert(7);
	list.Print();
	int k;
	cout << "请输入要查询倒数第几个结点(计数从0开始):";
	cin >> k;
	LinkNode *p = list.FindByIndex(list.Length() - k);
	if (p)
	{
		cout << p->nValue << endl;
	}
	LinkNode *p2 = list.FindLastNum(k);
	if (p2)
	{
		cout << p2->nValue << endl;
	}
	system("pause");
	return 0;
}
运行结果如图1所示:

图1 运行效果

转载于:https://www.cnblogs.com/new0801/p/6176915.html

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 可以使用双指针法,定义两个指针p1和p2,先让p1指向链表的头结点,然后让p2指向第k个结点。接着,同时移动p1和p2指针,直到p2指向链表的末尾,此时p1所指向的结点就是倒数第k个结点。 具体实现如下: 1. 定义两个指针p1和p2,初始时都指向链表的头结点。 2. 让p2指针先向后移动k-1个结点,如果链表长度小于k,则直接返回null。 3. 同时移动p1和p2指针,直到p2指向链表的末尾。 4. 返回p1指向的结点。 代码实现如下: public ListNode findKthToTail(ListNode head, int k) { if (head == null || k <= ) { return null; } ListNode p1 = head; ListNode p2 = head; for (int i = ; i < k - 1; i++) { if (p2.next != null) { p2 = p2.next; } else { return null; } } while (p2.next != null) { p1 = p1.next; p2 = p2.next; } return p1; } ### 回答2: 首先需要了解单向链表的结构和特点。单向链表一个个节点组成,每个节点包括两部分:数据和指向下一个节点的指针。因此,单向链表每个节点只能访问其后面的节点,不能访问前面的节点。这就给查找倒数第k个节点带来了一定的难度。 假设单向链表的长度为n,则要查找倒数第k个节点,可以先让一个指针从链表的头节点开始向前移动k-1个位置,然后再让另一个指针从头节点开始一起向前移动,直到第一个指针到达链表末尾时,第二个指针(也就是倒数第k个节点)所指向的就是所求节点。 需要注意的是,在实际编程,要对链表的长度和k值进行一定的判断,防止出现链表长度不足k的情况。 下面是示例代码: ``` struct ListNode { int val; ListNode *next; ListNode(int x) : val(x), next(NULL) {} }; ListNode* findKthToTail(ListNode* head, int k) { if (head == NULL || k == 0) return NULL; ListNode* pAhead = head; ListNode* pBehind = head; for (int i = 1; i < k; i++) { if (pAhead->next != NULL) { pAhead = pAhead->next; } else { return NULL; } } while (pAhead->next != NULL) { pAhead = pAhead->next; pBehind = pBehind->next; } return pBehind; } ``` 该函数的参数为单向链表头指针和 k 值,返回倒数第k个节点的指针。 ### 回答3: 首先我们可以想到两个指针p1、p2,先让p1指向链表的头结点,然后让它向后移动k-1个节点,接着让p2指向头节点,接下来p1、p2两个指针同时向后遍历,直到p1指向链表的末尾时,此时p2所指向的节点就是倒数第k个节点了。 具体实现如下: 1. 判断链表是否为空,如果为空则返回null; 2. 初始化p1和p2指向头节点; 3. 让p1向后移动k-1个节点; 4. 判断p1是否为空,如果为空则说明k的值太大,超出了链表的长度,返回null; 5. 接下来p1、p2两个指针同时向后遍历,直到p1指向链表的末尾时,此时p2所指向的节点就是倒数第k个节点了。 代码实现如下: ``` public ListNode findKthToTail(ListNode head, int k) { if(head == null || k == 0) { return null; } ListNode p1 = head; ListNode p2 = head; for(int i = 0; i < k-1; i++) { if(p1.next != null) { p1 = p1.next; } else { return null; // k的值太大,超出了链表的长度 } } while(p1.next != null) { p1 = p1.next; p2 = p2.next; } return p2; } ``` 以上就是输出单向链表倒数第k个节点的具体实现方法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值