当前位置: 首页 > news >正文

链表中的回文判断

给一个链表,判断这个链表是否为回文链表。能否使用O(1)的空间复杂度解决问题?

思路1:使用辅助空间,我们这里给出了使用动态数组作为检查表,给出了两种实现方式,但是这种实现方式效率不高。

​ public class ListNode { public int val; public ListNode next; public ListNode(int x) { this.val = x; this.next = null; } public static ListNode createList(int[] nums) { if(null == nums || 0 == nums.length) return null; ListNode head = new ListNode(nums[0]); ListNode needle = head; for(int i = 1; i < nums.length;++i) { ListNode node = new ListNode(nums[i]); needle.next = node; needle = needle.next; needle.next = null; } return head; } }
import java.util.ArrayList; import java.util.List; class Solution { public boolean isPalindrome(ListNode head) { if (null == head || null == head.next) return true; List<Integer> all = new ArrayList<Integer>(); while (head != null) { all.add(head.val); head = head.next; } for (int i = 0; i < all.size() / 2; ++i) { if ((int) all.get(i) != (int) all.get(all.size() - 1 - i)) return false; } return true; } public static void main(String[] args) { int[] nums1 = { 1, 2, 4, 2, 1 }; ListNode l1 = ListNode.createList(nums1); boolean result = new Solution().isPalindrome(l1); System.out.print(result); } }

思路2:使用O(1)空间复杂度,即需要的临时空间较少,且跟链表长度没有关系,我们这里给出了两种实现方式,实现思路相同。使用快慢指针,找到中间结点位置,一种是反转链表的前半部分;一种是反转链表的后半部分,反转后半部分更容易实现,效率也要高。

class Solution { public boolean isPalindrome(ListNode head) { if (null == head || null == head.next) return true; // 找中间位置开始 ListNode fast = head; ListNode faster = head; while (faster != null && faster.next != null) { fast = fast.next; faster = faster.next.next; } // 找中间位置结束 // 反转fast之前的所有元素 // pre指向当前结点的前驱,反转后第一个结点的后继 ListNode pre = null; // 指向当前遍历的结点 ListNode cur = head; while (cur != fast) { // 记录当前结点的下一个结点,否则执行下一条一句就丢了后面没有反转的剩余结点 ListNode next = cur.next; // 真正的反转,指针变化方向,因为链表最后一个结点的next为空,这也是为什么pre的初始值为null cur.next = pre; // 向后继续遍历剩余未反转的结点,pre和cur均要向后移动一位 pre = cur; cur = next; } // 到此,cur指向fast,而pre指向了最后一个被反转的结点,也就是新链表的头 // 比较元素值是否相同开始 // 链表元素个数为奇数个的情况 if (null != faster && null == faster.next)// odd fast = fast.next; // 比较反转后的[pre,fast)与[fast,tail]到链表尾部 while (pre != null && fast != null) { if (pre.val != fast.val) return false; pre = pre.next; fast = fast.next; } // 比较元素值是否相同结束 return true; } public static void main(String[] args) { int[] nums1 = { 1, 2, 4, 2, 1 }; ListNode l1 = ListNode.createList(nums1); boolean result = new Solution().isPalindrome(l1); System.out.print(result); } }
http://www.rkmt.cn/news/94843.html

相关文章:

  • 系统基础服务
  • Python基础知识的总结(2)
  • Go程序的执行顺序
  • JavaWeb-Request应用与Cookie[特殊字符]️Session
  • 数据结构:二叉排序树,平衡二叉树,红黑树的介绍
  • 基于条件风险价值CVaR的微网动态定价与调度策略附Matlab代码
  • Go语言变量
  • 架构系统序化
  • OpenVSCode Server终极性能调优与资源管理完整指南
  • ASP毕业设计题目推荐:基于ASP+Access的校园二手交易平台设计与实现
  • 【Arduino Uno】数码管模拟值实验
  • 进程PCB
  • Springboot医药采购管理2mqc3(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
  • LeetCode 41. 缺失的第一个正数 | 原地哈希最优解全解析
  • 线性回归模型
  • 榛子矮砧密植:水肥一体化系统的铺设要点指南
  • java基础流程控制笔记
  • 校招 Java 面试必看:JVM 其实就考这 3 个点(我帮你讲透)
  • [从零构建操作系统]08 函数调用时栈的底层行为解析
  • MATLAB与FlightGear联合仿真教程:包含Simulink工程文件的PDF指南
  • Springboot医疗云胶片管理系统nem7x(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
  • Day 37 MLP神经网络的训练
  • 探索含光伏、火电与飞轮储能系统的奇妙调频之旅
  • 高效获取高质量外链:2026年必须掌握的10个核心策略
  • Flutter国际化(i18n)实现详解
  • YOLOv13涨点改进 | 独家创新首发、Conv卷积改进篇 | SCI一区 2025 | 引入MSConvStar多尺度卷积星形模块,有效增强捕捉多范围特征,助力目标检测、图像分割、图像分类高效涨点
  • LLC谐振变换器恒压恒流双竞争闭环Simulink仿真探索
  • Feign基本知识
  • YOLOv13涨点改进 | 全网独家创新、Neck特征融合改进篇 | TGRS 2025顶刊 | 引入ADSF自适应特征融合模块,自适应融合浅层特征与深层特征,适合红外小目标检测、图像分割等有效涨点
  • 常用软件工具的使用(1) ---- git 的安装和基础操作