前言

最近准备暑假回家回家修整一下,所以时间大部分用来完成项目上的工作,同时为了9月份的校招,晚上的时间我还在学习<cracking the coding intreview>,第二章链表有几个不错的题目,记录一下
 

单链表

题目: Implement an algorithm to find the nth to last element of a singly linked list.

译文: 实现一个算法从一个单链表中返回倒数第n个元素

思路

7个节点的示例链表图如下:




例如我们找倒数第3个节点5,有两种思路

(1)遍历一遍链表,获取链表的总数len,这样倒数第n个节点也就是正数第(len - n + 1)个节点,时间复杂度为O(2n)

(2)很tricky做法,使用两个指针p,q,p指向head节点,q先前进n个节点,然后两个指针依次指向下一个,到q为NULL时,p为倒数第n个节点,时间复杂度为O(n)

代码(c语言)

采用第二种做法,因此时间复杂度更低,而且感觉更牛逼一些

/**
* Implement an algorithm to find the nth to last element of a singly linked list
*/ #include <stdio.h>
#include <stdlib.h> typedef struct link {
int value;
struct link *next;
} link; /**
* 创建单链表
*
* T = O(n)
*
*/
void createLinklist(link **head, int data)
{
link *pre, *cur, *new;
cur = *head;
pre = NULL; while (cur != NULL) {
pre = cur;
cur = cur->next;
} new = (link *)malloc(sizeof(link));
new->value = data; if (pre == NULL)
*head = new;
else
pre->next = new;
} /**
* 打印单链表
*
* T = O(n)
*/
void printLinklist(link *head)
{
while (head->next != NULL) {
printf("%d ", head->value);
head = head->next;
}
printf("%d\n", head->value);
} /**
* 寻找单链表倒数第m个节点
*
* T = O(n)
*
*/
void findMthToLast(link *head, int m)
{
if (head == NULL) return;
link *s1, *s2;
int i;
// s1指向表头,s2指向从s1开始后m个元素的位置,然后s1与s2同时后移,到s2为NULL时停止,s1为mth to last
s1 = s2 = head; for (i = 0; i < m; i ++) {
s2 = s2->next;
} while (s2 != NULL) {
s1 = s1->next;
s2 = s2->next;
} printf("%d\n", s1->value);
} int main(void)
{
int i, n, m, data;
link *head; while (scanf("%d", &n) != EOF) {
for (i = 0, head = NULL; i < n; i ++) {
scanf("%d", &data);
createLinklist(&head, data);
} // 接收mth to last
scanf("%d", &m); if (m > n) {
printf("输入数据有误!\n");
} else {
printLinklist(head);
findMthToLast(head, m);
}
} return 0;
}

循环链表

题目:给定一个循环链表,实现一个算法返回这个环开始的结点

例子:

输入 : A->B->C->D->E->C[节点C在之前已经出现过]
输出:节点C

思路

先来个图示:





参考上面的做法,我们也设置两个指针,fast和slow,first一次走两个节点,slow一次走一个节点,从head出发,最后必然在循环内某个节点相遇,我们模拟一下:

1. s->B  f->D
2. s->C  f->C

(ps:不是每次都在起始点相遇哈,这里是巧合)

保持f不动,s继续移动,记录移动的次数,当再次到达f是,当前次数即为循环的长度len

然后定义两个节点p,q,p指向表头,q向前移动len步,然后一次走到p和q相遇,此节点极为链表的循环起始节点

代码(c语言)

/**
* Given a circular linked list, implement an algorithm which returns
* node at the begining of the loop
*/ #include <stdio.h>
#include <stdlib.h> typedef struct link {
int value;
struct link *next;
} link; /**
* 创建单链表
*
* T = O(n)
*
*/
void createLinklist(link **head, int data)
{
link *cur, *pre, *new; cur = *head;
pre = NULL; while (cur != NULL) {
pre = cur;
cur = cur->next;
} new = (link *)malloc(sizeof(link));
new->value = data;
new->next = cur; if (pre == NULL) {
*head = new;
} else {
pre->next = new;
}
} /**
* 从m个节点开始构建循环链表
*
* T = O(n)
*
*/
void initLoopList(link *head, int m)
{
link *cur, *pre, *target;
cur = head; while (-- m && cur != NULL) {
cur = cur->next;
}
target = cur; while (cur != NULL) {
pre = cur;
cur = cur->next;
}
pre->next = target;
} /**
* 寻找循环开始点的value
*
* T = O(n)
*
*/
void loopStart(link *head)
{
link *fast, *slow, *p, *q;
int i, len; for (slow = head, fast = slow->next->next; fast != slow;) {
slow = slow->next;
fast = fast->next->next;
} for (len = 1, slow = slow->next; slow != fast; slow = slow->next) {
len += 1;
} p = q = head;
for (i = 0; i < len; i ++) {
q = q->next;
} while (p != q) {
p = p->next;
q = q->next;
}
printf("%d\n", q->value);
} int main(void)
{
link *head;
int i, n, m, data; while (scanf("%d", &n) != EOF) {
// 创建单链表
for (i = 0, head = NULL; i < n; i ++) {
scanf("%d", &data);
createLinklist(&head, data);
} // 第m个点开始循环
scanf("%d", &m);
if (m > n) continue;
initLoopList(head, m); // 查找循环起始节点(只提供头节点)
loopStart(head);
} return 0;
}





《cracking the coding intreview》——链表的更多相关文章

  1. 链表<新>

    class Node: ''' 节点类 链表节点结构 data next data: 节点保存的数据 _next: 保存下一个节点对象 ''' def __init__(self, data, pne ...

  2. c++ 链表删除重复的数据

    //List.h #include <iostream> typedef int dataType; struct Node{ Node():data(),pNextNode(NULL){ ...

  3. 链表的基本操作(Basic Operations on a Linked List)

    链表可以进行如下操作: 创建新链表 增加新元素 遍历链表 打印链表 下面定义了对应以上操作的基本函数. 创建新链表 新链表创建之后里面并没有任何元素,我们要为数据在内存中分配节点,再将节点插入链表.由 ...

  4. linux内存源码分析 - 内存回收(lru链表)

    本文为原创,转载请注明:http://www.cnblogs.com/tolimit/ 概述 对于整个内存回收来说,lru链表是关键中的关键,实际上整个内存回收,做的事情就是处理lru链表的收缩,所以 ...

  5. PAT 1025 反转链表

    PAT (Basic Level) Practise 1025 Github链接:https://github.com/H-BING/object-oriented/tree/master/PAT10 ...

  6. (转)linux内存源码分析 - 内存回收(lru链表)

    原文:http://www.cnblogs.com/tolimit/p/5447448.html 概述 对于整个内存回收来说,lru链表是关键中的关键,实际上整个内存回收,做的事情就是处理lru链表的 ...

  7. OptimalSolution(3)--链表问题(1)简单

    单链表Node节点类 public class Node { public int val; public Node next; public Node(int val) { this.val = v ...

  8. java 8 jdk1.8 新特性

    1Lambda表达式 2函数式接口 函数式接口(Functional Interface)就是一个有且仅有一个抽象方法,但是可以有多个非抽象方法的接口. java 8为函数式接口引入了一个新注解@Fu ...

  9. 【数据结构和算法】001 单链表 LinkedList

    一.单链表(LinkedList)介绍和内存布局 链表是有序的列表,它在内存中的实际存储结构如下: 看上去虽然无序,但他是靠灭个链表节点元素的地址和next域来分清首尾相连的顺序,如下图所示,由头指针 ...

  10. 详细分析链表的数据结构的实现过程(Java 实现)

    目录 链表的数据结构的实现过程(Java 实现) 前言 基本概念 链表的基本结构 链表的基本操作的实现 在链表中添加元素 在链表头添加元素 在链表指定位置处添加元素 链表的虚拟头节点 链表的查询和修改 ...

随机推荐

  1. zoj 2402 - Lenny&#39;s Lucky Lotto Lists

    称号:序列,在前面的每个元件的至少两倍,最大值至n.问:长l船舶有许多这样的. 分析:dp,LIS类别似事. 状态:f(i,j)结束数字为j且长度为i的序列的个数.有转移方程: F[ i ][ j ] ...

  2. Android启动第三方应用程序

    主要是开始通过包名的第三方应用程序,获取的方法的包名是非常在线.不是说. 两种方式启动: 第一: Intent intent = new Intent(); intent.setClassName(& ...

  3. JavaEE Tutorials (1) - 概述

    1.1 Java EE 7平台新增特性3 1.2 Java EE应用模型3 1.3 分布式多层应用4 1.3.1 安全4 1.3.2 Java EE组件5 1.3.3 Java EE客户端6 1.3. ...

  4. crm操作安全角色

    using System;     using Microsoft.Xrm.Sdk;     using Microsoft.Crm.Sdk.Messages;     using System.Co ...

  5. android 如何加入第一3正方形lib图书馆kernel于

    注意:只能lib图书馆kernel编译到位.例如下列: alps/kernel/ alps/mediatek/custom/common/kernel/ alps/mediatek/custom/$p ...

  6. Hibernat之关系的处理多对多

    第一步:编写两个pojo,比如一个学生表一个课程表  这里使用注解. 需要 课程表: package com.qcf.pox; import java.util.HashSet; import jav ...

  7. c语言发挥帕斯卡三角

    我们已经确定了帕斯卡三角的规则,下面是我的代码,非常实用哦! !! #include<stdio.h>  void main()  {      int i,j,n,k;      sca ...

  8. appium之java API(2)

    TouchAction AppiumDriver的辅助类,主要针对手势操作,比方滑动.长按.拖动等.TouchAction的原理是讲一系列的动作放在一个链条中,然后将该链条传递给server.serv ...

  9. nginx 查看并发连接数

    这里仅仅说一下用命令查看(也可配置页面) 通过查tcp连接数 1.netstat -n | awk '/^tcp/ {++S[$NF]} END {for(a in S) print a, S[a]} ...

  10. 基于NHibernate二级缓存的MongoDB组件

    设计一套基于NHibernate二级缓存的MongoDB组件(上)   摘要:NHibernate Contrib 支持很多第三方的二级缓存,如SysCache,MemCache,Prevalence ...