1. 题目

2. 解答

2.1. 方法一

将链表每个节点向右移动 1 个位置,其实就是让链表最后一个结点指向第一个结点。

因此,向右移动 k 个位置就重复上述过程 k 次即可。

然后,我们注意到,若链表有 n 个结点,则移动 n 次后就还是原链表。

原始链表 1->2->3->NULL
向右旋转 1 步: 3->1->2->NULL
向右旋转 2 步: 2->3->1->NULL
向右旋转 3 步: 1->2->3->NULL

实际上,若链表有 n 个结点,我们只需要移动 n % k 次即可。

确定链表有多少个结点,我们则可以用快慢指针法。

  • 偶数个结点时,结点个数等于 i * 2。

  • 奇数个结点时,结点个数等于 i * 2 + 1。
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/
class Solution {
public:
ListNode* rotateRight(ListNode* head, int k) { if (head == NULL || head->next == NULL) return head; int node_num = 0; // 链表的结点个数
ListNode* slow = head;
ListNode* fast = head; // 利用快慢指针确定链表的结点总个数
while(fast && fast->next)
{
slow = slow->next;
fast = fast->next->next;
node_num++;
} if (fast) // 奇数个结点
{
node_num = node_num * 2 + 1;
}
else // 偶数个结点
{
node_num = node_num * 2;
} // 若链表有 K 个结点,向右移动 K 个位置就还是原链表
// 因此我们只需要移动 K % node_num 次即可
if (k % node_num == 0) return head;
else k = k % node_num; ListNode *temp = head;
ListNode *last_node = NULL; for (int i = 0; i < k; i++)
{
// 向右移动 1 个位置就是让最后一个结点指向第一个结点 // 先找到倒数第二个结点
while (temp->next->next)
{
temp = temp->next;
} // 最后一个结点指向第一个结点,倒数第二个节点指向 NULL
last_node = temp->next;
temp->next = NULL;
last_node->next = head; head = last_node;
temp = head;
} return head;
}
};

2.2. 方法二

上面的算法中每次将链表结点向右移动 1 个位置的时候,我们都要遍历一次链表,效率较低。

针对链表 1->2->3->4->5->NULL,如果要将链表每个结点向右移动 2 个位置,那倒数第 2 个结点就是旋转后新链表的起始位置;如果要将链表每个结点向右移动 3 个位置,那倒数第 3 个结点就是旋转后新链表的起始位置;而如果要将链表每个结点向右移动 33 个位置,那倒数第 3 个结点就是旋转后新链表的起始位置。

更一般的情况下,若链表长度为 n,移动次数为 k,旋转后链表的起始位置就是原链表倒数第 k % n 个结点

在上面的例子中,假设新链表的起始位置是结点 4,那链表就被分成了两部分。1->2->3 和 4->5->NULL,我们要做的就是把 1->2->3 拼接在 4->5->NULL 后即可。

实现思路是这样的:

我们先通过快慢指针确定链表长度 n,然后找到旋转后链表起始位置的上一个结点,记为 new_head_last,然后从起始位置 new_head 开始向后遍历,当到达 NULL 时,将 head 指向的头结点拼接在 new_head 后面,然后再将新链表的最后一个结点也即 new_head_last 后面置为 NULL 即可。

/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/
class Solution {
public:
ListNode* rotateRight(ListNode* head, int k) { if (head == NULL || head->next == NULL) return head; int node_num = 0; // 链表的结点个数
int mid_num = 0;
ListNode* slow = head;
ListNode* fast = head; // 利用快慢指针确定链表的结点总个数
while(fast && fast->next)
{
slow = slow->next;
fast = fast->next->next;
mid_num++;
} if (fast) // 奇数个结点
{
node_num = mid_num * 2 + 1;
}
else // 偶数个结点
{
node_num = mid_num * 2;
} // 若链表有 K 个结点,向右移动 K 个位置就还是原链表
// 因此我们只需要移动 K % node_num 次即可
if (k % node_num == 0) return head;
else k = k % node_num; int which_node_is_new = 0; // 旋转后的头结点是第几个节点
ListNode* new_head = NULL;
ListNode* new_head_last = NULL; // 查找旋转后头结点的上一个结点
// 此时 slow 指针指向中间结点,分为在 slow 前还是在 slow 后
which_node_is_new = node_num - k;
if (which_node_is_new > node_num / 2)
{
new_head_last = slow;
for (int i = 0; i < which_node_is_new - mid_num - 1; i++)
{
new_head_last = new_head_last->next;
}
}
else
{
new_head_last = head;
for (int i = 1; i < which_node_is_new; i++)
{
new_head_last = new_head_last->next;
}
} new_head = new_head_last->next;
ListNode* temp = new_head; while (temp->next)
{
temp = temp->next;
} temp->next = head;
new_head_last->next = NULL; return new_head;
}
};

获取更多精彩,请关注「seniusen」!

最新文章

  1. iptables过滤设置服务端口
  2. (Array)121. Best Time to Buy and Sell Stock
  3. ASP.NET使用ConfigurationSection在Web.Config创建自定义配置节集合
  4. 重新想象 Windows 8.1 Store Apps (76) - 新增控件: SearchBox
  5. Java中的继承与组合(转载)
  6. hasSet,TreeSet,ArrayList,LinkedList,Vector,HashMap,HashTable,TreeMap利用Iterator进行输出
  7. LeetCode Find Peak Element
  8. Linux C socket 编程之TCP
  9. APP主流UI框架结构
  10. P168 实战练习(权限修饰符)
  11. try catch 怎么写?
  12. 前端的数据库:IndexedDB 。 ps:入门
  13. C#进程启动实例
  14. [LeetCode] Interleaving String [30]
  15. Objective-c 访问控制
  16. 前端三大框架Angular &amp; React &amp; Vue
  17. laravel5.4+vue+element简单搭建(gulp+laravel Elixir)(转)
  18. 数据结构与算法之PHP排序算法(冒泡排序)
  19. Jquery中addClass方法不起作用的解决方案
  20. python--第十一天总结(paramiko 及数据库操作)

热门文章

  1. 使用transfor让图片旋转
  2. Oracle树形结构数据---常见处理情景
  3. JavaScript监控输入框字数变化,超出限制则禁止输入
  4. idea开启自动编译
  5. ABAP术语-IDOC
  6. jdbc学习笔记03
  7. go 下面定义嵌套结构
  8. python字符串的格式化输出
  9. bootstrap Table动态绑定数据并自定义字段显示值
  10. springmvc重定向请求。