删除链表的中间节点和a/b处的节点

题目

  给定链表表的头节点head,实现别删除链表的中间节点的函数
  例如:
  不删除任何节点
  1 → 2,删除节点1;
  1 → 2 → 3,删除节点2;
  1 → 2 → 3 → 4,删除节点2;
  1 → 2 → 3 → 4 → 5,删除节点3

  进阶: 给定链表的头节点head、整数a和b,实现删除位于a/b处节点的函数。
  例如:
  链表: 1 → 2 → 3 → 4 → 5,假设a/b的值为 r。
  如果 r 等于0,不删除任何节点;
  如果 r 在区间(0,1/5]上,删除节点1;
  如果 r 在区间(1/5,2/5]上,删除节点2;
  如果 r 在区间(2/5,3/5]上,删除节点3;
  如果 r 在区间(3/5,4/5]上,删除节点4;
  如果 r 在区间(4/5,1]上,删除节点5;
  如果 r 大于1,不除任何节点。

思路

  先来分析原问题,如果链表为空或者长度为1,不需要调整,则直接返回;如果链表的长度为2,将头节点除即可;当链表长度到达3,应该删除第2个节点;当链表长度为4,应该除第2个节点;当链表长度为5,应该删除第3个节点……也就是链表长度每增加2 (3,5,7……)。要删除的节点就后移一个节点。删除节点的问题在之前的题目中我们已经讨论过,如果要删除个节点,则需要找到待删除节点的前一个节点。
  具体过程请参看如下代码中的 removemidnode方法

  接下来时论进阶问题,首先需要解决的问题是,如何根链表的长度n,以及a与b的值决定该删除的节点是那一个节点呢? 根据如下方法: 先计算 double r
= ((double)(a * n)) / ((double)b) 的值,然后 r 向上取整之后的整数值,代表该删除的节点是第几个节点。
  下面几个例子来验证一下。
  如果表长度为7,a= 5,b = 7,
   r = (7 * 5) / 7= 5.0,向上取整后为5,所以应该删除第5个节点。
  如果链表长度为7,a = 5,b= 6。
  r = (7 * 5) / 6=5.8333…,向上取后为6,所以应该除第6个节点。
  如果链表长度为7,a=1,b=6
  r=,(7 * 1) / 6 = 1.6666…,向上取整后为2,所以应该别第2个节点。
  知道该除第几个节点之后,接下来需要找到删除节点的前一个节点即可,具体过程参
  看如下代码中的 removebyratio方法

代码演示

package com.itz.zcy.listQuestion;

/**
 * 给定链表表的头节点head,实现别删除链表的中间节点的函数
 */
public class RemoveListCentreNodeAndabNode {

    /**
     * 节点
     */
    public static class Node {
        public int value;
        public Node next;

        public Node(int data) {
            this.value = data;
        }
    }

    /**
     * 给定链表表的头节点head,实现别删除链表的中间节点的函数
     *
     * @param head
     * @return
     */
    public static Node removeMidNode(Node head) {
//        异常判定
        if (head == null || head.next == null) {
            return head;
        }
//        土豪只有一个节点直接返回
        if (head.next == null) {
            return head.next;
        }
        Node pre = head;
        Node cur = head.next.next;
//        当下一个和下一个的下一个不为空时候,指向下一个和下一个的下一个
        while (cur.next != null && cur.next.next != null) {
            pre = pre.next;
            cur = cur.next.next;
        }
//        删除节点
        pre.next = pre.next.next;
        return head;
    }

    /**
     * 进阶: 给定链表的头节点head、整数a和b,实现删除位于a/b处节点的函数
     *
     * @param head
     * @param a
     * @param b
     * @return
     */
    public static Node removeByRatio(Node head, int a, int b) {
        if (a < 1 || a > b) {
            return head;
        }
        int n = 0;
        Node cur = head;
//        求出链表的长度
        while (cur != null) {
            n++;
            cur = cur.next;
        }
//        计算需要删除的节点
        n = (int) Math.ceil(((double) (a * n)) / (double) b);
//        等于1是直接删除第一个节点
        if (n == 1) {
            head = head.next;
        }
//        大于1的时候
        if (n > 1) {
            cur = head;
            while (--n != 1) {
                cur = cur.next;
            }
            cur.next = cur.next.next;
        }
        return head;
    }

    public static void main(String[] args) {
        Node node1 = new Node(1);
        node1.next = new Node(2);
        node1.next.next = new Node(3);
        node1.next.next.next = new Node(4);
        node1.next.next.next.next = new Node(5);
        node1.next.next.next.next.next = new Node(6);
        node1.next.next.next.next.next.next = new Node(7);
        Node head1 = removeMidNode(node1);
        while (true){
            if (head1!=null){
                if (head1.next == null){
                    System.out.print(head1.value);
                }else {
                    System.out.print(head1.value+" → ");
                }
            }else {
                break;
            }
            head1 = head1.next;
        }
        System.out.println();
        Node node2 = new Node(1);
        node2.next = new Node(2);
        node2.next.next = new Node(3);
        node2.next.next.next = new Node(4);
        node2.next.next.next.next = new Node(5);
        node2.next.next.next.next.next = new Node(6);
        node2.next.next.next.next.next.next = new Node(7);
        Node head2 = removeByRatio(node2,2,3);
        while (true){
            if (head2!=null){
                if (head2.next == null){
                    System.out.print(head2.value);
                }else {
                    System.out.print(head2.value+" → ");
                }
            }else {
                break;
            }
            head2 = head2.next;
        }
//        1 → 2 → 3 → 5 → 6 → 7
//        1 → 2 → 3 → 4 → 6 → 7
    }
}

总结

该题目比较简单;时间复杂度是O(n),空间复杂度O(1)。

文献:左程云著 《程序员代码面试指南IT名企算法与数据结构题目最优解》(第二版)
版权声明:此文版权归作者所有!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容