1074 Reversing Linked List
题意:
每k个元素反转链表,不足k个就不反转。如原链表为1→2→3→4→5→6,k=3,则反转后的链表为3→2→1→6→5→4;若k=4,则反转后的链表为4→3→2→1→5→6。
思路:
这题会比较烦,写代码前一定要现在纸上理清思路,写出关键代码,不然出了错再改来改去真的很浪费时间,要是考试的话估计心态就蹦了。本题是比较经典的“反转链表(指反转整条链表)”的升级版,但做法是一样的。我们可以把“反转”这一动作单独抽象成一个函数,然后遍历链表,每遍历k个结点(假设是pi...pj),就把pi...pj这个子链表进行反转,易错点在于每k个子链表之间该如何衔接而确保不会断链,具体看代码,我已经注释的很详细了。
注:本题是关于“链表”操作非常经典的题目,应当熟练掌握,因为这是非常非常基础的问题。另外,个人觉得有必要说一下的是,网上很多的解题报告,以及《算法笔记》里的题解,都不是真正的“反转”操作,虽然也能AC,但不利于真正掌握链表、指针的操作,有些投机取巧。要是面试写白板代码的时候写成那种样子,是要被鄙视的。
代码:
#include <cstdio>
;
struct Node{
int data;
int curr,next;
}LinkList[N];
//反转操作,记录新链表的头结点和尾结点
//传入时,tail==head,记得加“&”
void reverseLinkList(int& head,int& tail)
{
//如果链表为空,或者只有一个结点,则直接返回
|| LinkList[head].next==-) return;
,p=head;//p为工作指针
tail=LinkList[head].curr;
){
int next=LinkList[p].next;
)
head=p;//如果next为空,则当前结点为最后一个结点,令其为新链表的头结点
LinkList[p].next=pre;
pre=p;
p=next;
}
}
int main()
{
//freopen("pat.txt","r",stdin);
int n,k,head;
scanf("%d%d%d",&head,&n,&k);
int curr,data,next;
;i<n;i++){
scanf("%d%d%d",&curr,&data,&next);
LinkList[curr].data=data;
LinkList[curr].curr=curr;
LinkList[curr].next=next;
}
int p=head;
,tail=-;
){
;
&& cnt<k){
p=LinkList[p].next;
cnt++;
}
){
int tmp=p;
p=LinkList[p].next;//先更新,再截断
LinkList[tmp].next=-;//把子链表截断
//翻转当前含有k个结点的子链表,并分别记录其头结点和尾结点
int tmptail=tmphead;
reverseLinkList(tmphead,tmptail);
) newhead=tmphead;
else LinkList[tail].next=tmphead;
tail=tmptail;//更新尾结点
}else{
LinkList[tail].next=tmphead;//剩余不足k个结点,就把剩余部分直接链在tail后面
p=-;//别忘了
}
}
p=newhead;
){
printf("%05d %d ",LinkList[p].curr,LinkList[p].data);
) printf("%05d\n",LinkList[p].next);
else printf("-1\n");
p=LinkList[p].next;
}
;
}
【第一次限时AC的时候是这么做的,但这不是纯正“翻转”操作】
#include <cstdio>
#include <vector>
#include <algorithm>
using namespace std;
;
struct Node{
int data;
int curr,next;
}LinkList1[N],LinkList2[N];
int main()
{
//freopen("pat.txt","r",stdin);
int n,k,head;
scanf("%d%d%d",&head,&n,&k);
int curr,data,next;
;i<n;i++){
scanf("%d%d%d",&curr,&data,&next);
LinkList1[curr].data=data;
LinkList1[curr].curr=curr;
LinkList1[curr].next=next;
}
;
,pre=-;
vector<Node> temp;
){
temp.push_back(LinkList1[p]);
cnt++;
if(cnt==k){
reverse(temp.begin(),temp.end());
){
newhead=temp[].curr;
pre=newhead;
}else{
LinkList2[pre].next=temp[].curr;
pre=temp[].curr;
}
;
<temp.size();i++){
LinkList2[pre].data=temp[i].data;
LinkList2[pre].curr=temp[i].curr;
LinkList2[pre].next=temp[i+].curr;
pre=temp[i+].curr;
}
LinkList2[pre].data=temp[i].data;
LinkList2[pre].curr=temp[i].curr;
LinkList2[pre].next=-;
//注意别忘了重置
cnt=;
temp.clear();
}
p=LinkList1[p].next;
}
){
LinkList2[pre].next=temp[].curr;
pre=temp[].curr;
;
<temp.size();i++){
LinkList2[pre].data=temp[i].data;
LinkList2[pre].curr=temp[i].curr;
LinkList2[pre].next=temp[i+].curr;
pre=temp[i+].curr;
}
LinkList2[pre].data=temp[i].data;
LinkList2[pre].curr=temp[i].curr;
LinkList2[pre].next=-;
}
p=newhead;
){
printf("%05d %d ",LinkList2[p].curr,LinkList2[p].data);
) printf("%05d\n",LinkList2[p].next);
else printf("-1\n");
p=LinkList2[p].next;
}
;
}
1074 Reversing Linked List的更多相关文章
- PAT 1074 Reversing Linked List[链表][一般]
1074 Reversing Linked List (25)(25 分) Given a constant K and a singly linked list L, you are suppose ...
- PAT 甲级 1074 Reversing Linked List (25 分)(链表部分逆置,结合使用双端队列和栈,其实使用vector更简单呐)
1074 Reversing Linked List (25 分) Given a constant K and a singly linked list L, you are supposed ...
- PAT 1074. Reversing Linked List (25)
Given a constant K and a singly linked list L, you are supposed to reverse the links of every K elem ...
- PAT Advanced 1074 Reversing Linked List (25) [链表]
题目 Given a constant K and a singly linked list L, you are supposed to reverse the links of every K e ...
- 1074. Reversing Linked List (25)
模拟题,注意当k == 1 与 k == n时情况 #include <stdio.h> #include <string.h> #include <iostream&g ...
- PAT (Advanced Level) 1074. Reversing Linked List (25)
简单题. #include<cstdio> #include<cstring> #include<cmath> #include<vector> #in ...
- PAT甲题题解-1074. Reversing Linked List (25)-求反向链表
题意说的很清楚了,这种题的话,做的时候最好就是在纸上自己亲手模拟一下,清楚一下各个指针的情况, 这样写的时候就很清楚各个指针变量保存的是什么值. PS:一次AC哈哈,所以说自己动手在纸上画画还是很有好 ...
- PAT 1074. Reversing Linked List
#include <cstdio> #include <cstdlib> #include <iostream> #include <unordered_ma ...
- 【PAT甲级】1074 Reversing Linked List (25 分)
题意: 输入链表头结点的地址(五位的字符串)和两个正整数N和K(N<=100000,K<=N),接着输入N行数据,每行包括结点的地址,结点的数据和下一个结点的地址.输出每K个结点局部反转的 ...
随机推荐
- iOS CoreData版本升级和数据库迁移
app中使用了CoreData,并且在下一个版本中有实体变动,比如实体新增字段.修改字段等改动, 那么app在覆盖安装时就要进行数据库迁移, 否则app就会crash. 那如何实现数据库迁移呢?大概需 ...
- 腾讯开源的Android UI框架——QMUI Android
各位同学,早上好,我是你们的老朋友D_clock爱吃葱花,前些天忙着发版本,最近也在看各种各样的新知识,有好多东西想写啊啊啊啊啊.嗯,先冷静捋一下,卖个关子.扯回正题,今天继续为大家推荐一个Githu ...
- discuz! 设置私密论坛版块的方法
Discuz!的强大功能不用细说, 话说对于有一部分需要设置具有一定访问权限的用户才能浏览的版块内容的话. 可能很多朋友不太清楚, 为了解决这个问题, 第一步以管理员的身份登陆, 然后 论坛-> ...
- Java(Android)线程池妙用
介绍new Thread的弊端及Java四种线程池的使用,对Android同样适用.本文是基础篇,后面会分享下线程池一些高级功能. 1.new Thread的弊端执行一个异步任务你还只是如下new T ...
- AS3中ASCII码和字符互转函数
AS3中ASCII码和字符互转函数 字符转成ASCII码: 格式:字符串变量.charCodeAt(字符位置); var str:String = “A”; trace(str.charCodeAt( ...
- Android 进阶13:几种进程通信方式的对比总结
不花时间打基础,你将会花更多时间解决那些不必要的问题. 读完本文你将了解: RPC 是什么 IDL 是什么 IPC 是什么 Android 几种进程通信方式 如何选择这几种通信方式 Thanks RP ...
- Sphinx 匹配模式
所谓匹配模式就是用户如何根据关键字在索引库中查找相关的记录. SPH_MATCH_ALL, 匹配所有查询分词(默认模式); 如“手机配件”,不匹配 “我有一部手机”,但可以匹配 “手机坏了,需要找配件 ...
- 完全卸载session 所需要的函数
session_unset() 删除内存当中的session数据:必须放在session_destroy的前边.因为应用session_destory后session_id();就会消失. 删除se ...
- HihoCoder 1053 : 居民迁移 二分+贪心+双指针(好题)
居民迁移 时间限制:3000ms 单点时限:1000ms 内存限制:256MB 描述 公元2411年,人类开始在地球以外的行星建立居住点.在第1326号殖民星上,N个居住点分布在一条直线上.为了方便描 ...
- 剑指Offer面试题:9.打印1到最大的n位数
一 题目:打印1到最大的n位数 题目:输入数字n,按顺序打印从1到最大的n位十进制.比如输入3,则打印出1.2.3一直到最大的3位数即999. 二 不考虑大数解法 // 打印从1到最大的n位数 voi ...