zoukankan      html  css  js  c++  java
  • 链表:判断一个链表是否为回文结构

    题目:给定一个链表头节点 head,请判断链表是否为回文结构

    例如:

    1-->2-->1,返回 true;

    1-->2-->2-->1,返回 true;

    15-->6-->15,返回 true;

    1-->2-->3,返回 false;

    进阶:

    如果链表长度为 N,时间复杂度为 O(N),额外空间复杂度为 O(1);

    方法一:

    利用栈结构,将链表节点全部压入栈内,再重新从头遍历链表同时与栈内弹出的数据进行比较,如果该链表是一个回文结构则整个比较过程都为真,否则的话肯定有不相等的结点。

    时间复杂度O(N),空间复杂度O(N)

     1 class Node
     2 {
     3     public int data;
     4     public Node next;
     5 
     6     public Node(int data)
     7     {
     8         this.data = data;
     9     }
    10 }
    11 
    12 public boolean isPalindrome(Node head)
    13 {
    14     if(head == null || head.next == null)
    15         return true;
    16 
    17     Stack<Node> stack = new Stack<>();
    18     Node cur = head;
    19     while(cur != null)
    20     {
    21         stack.push(cur);
    22         cur = cur.next;
    23     }
    24 
    25     cur = head;
    26     while(!stack.isEmpty())
    27     {
    28         if(stack.pop().data != cur.data)
    29         return false;
    30         cur = cur.next;
    31     }
    32     return true;
    33 }

    方法二:

    方法二是方法一的优化版,虽然也是栈结构,但我们选择只将一半的节点压入即

     1 public boolean isPalindrome(Node head)
     2 {
     3     if(head == null || head.next == null)
     4         return true;
     5 
     6     Stack<Node> stack = new Stack<>();
     7     Node fast = head, slow = head.next;
     8     while(fast.next != null && fast.next.next != null)
     9     {
    10         slow = slow.next;
    11         fast = fast.next.next;
    12     }
    13 
    14     while(slow != null)
    15     {
    16         stack.push(slow);
    17         slow = slow.next;
    18     }
    19 
    20     while(!stack.isEmpty())
    21     {
    22         if(stack.pop().data != head.data)
    23             return false;
    24         head = head.next;
    25     }
    26     return true;
    27 }

    方法三:

    我们也可以在O(N)时间复杂度和O(1)空间复杂度内解决此题,具体分析如下:

    1. 改变链表右边区域的结构,使整个右半区反转,最后指向中间节点

    如图

    2. leftStart 和 rightStart 同时向中间遍历,每一步移动都比较当前节点的值,如果不一样则将标识量即为 false

    3. 不管是否为回文,都需要将原链表恢复原样

     1 public boolean isPalindrome(Node head)
     2 {
     3     if(head == null || head.next == null)
     4         return true;
     5 
     6     Node fast = head, slow = head;
     7     while(fast.next != null && fast.next.next != null)
     8     {
     9         slow = slow.next;
    10         fast = fast.next.next;
    11     }
    12 
    13     Node cur = slow.next;
    14     slow.next = null;
    15     Node pre = slow, next = null;
    16     while(cur != null)
    17     {
    18         next = cur.next;
    19         cur.next = pre;
    20         pre = cur;
    21         cur = next;
    22     }
    23 
    24     Node leftStart = head, rightStart = pre;
    25     boolean flag = true;
    26     while(leftStart != null && rightStart != null)
    27     {
    28         if(leftStart.data != rightStart.data)
    29         {
    30             flag = false;
    31             break;
    32         }
    33         leftStart = leftStart.next;
    34         rightStart = rightStart.next;
    35     }
    36     
    37     cur = pre.next;
    38     pre.next = null;
    39     while(cur != null)
    40     {
    41         next = cur.next;
    42         cur.next = pre;
    43         pre = cur;
    44         cur = next;
    45     }
    46 
    47     return false;
    48 }

     参考资料:程序员代码面试指南 IT名企算法与数据结构题目最优解,左程云

  • 相关阅读:
    GoldenGate 19.1实时文本文件加载攻略
    windows 10 excel 打开超连接提示 组织策略阻止...
    验证ogg同步数据库表无主键表且目标表包含隐藏字段
    配置ogg从Oracle到PostgreSQL的同步复制json数据
    pi
    GoldenGate 19.1 发布
    ogg同步DDL时,源和目标端表空间名称不同的解决思路
    总目录索引(开发精华总结)
    Spring Cloud Nacos分布式配置中心
    Spring Cloud Nacos&Feign负载均衡
  • 原文地址:https://www.cnblogs.com/2015110615L/p/6661512.html
Copyright © 2011-2022 走看看