java单链表常用操作

总结提高,与君共勉

概述、

数据结构与算法亘古不变的主题,链表也是面试常考的问题,特别是手写代码常常出现,将从以下方面做个小结

【链表个数】

【反转链表-循环】

【反转链表-递归】

【查找链表倒数第K个节点】

【查找链表中间节点】

【判断链表是否有环】

【从尾到头打印单链表-递归】

【从尾到头打印单链表-栈】

【由小到大合并有序单链表-循环】

【由小到大合并有序单链表-递归】

通常在java中这样定义单链表结构

<span style="font-family:Microsoft YaHei;font-size:14px;">class Node{
	int value;
	Node next;
	public Node(int n){
		this.value = n;
		this.next = null;
	}
}
</span>

1、链表个数

这个比较简单,不再赘述

<span style="font-family:Microsoft YaHei;font-size:14px;">	// 求单链表的结点个数
	public int getListLen(Node head) {
		int len = 0;
		while (head != null) {
			len++;
			head = head.next;
		}
		return len;
	}</span>

2、反转链表-循环

采用双指针,主要是4行代码,其中2,3俩行完成指针反转,1,4主要是保持head往下指

<span style="font-family:Microsoft YaHei;font-size:14px;">// 反转单链表(循环)
	public Node reverseList(Node head) {
		// 安全性检查
		if (head == null || head.next == null)
			return head;
		Node pre = null;
		Node temp = null;
		while (head != null) {
			// 以下1234均指以下四行代码
			temp = head.next;// 与第4行对应完成头结点移动
			head.next = pre;// 与第3行对应完成反转
			pre = head;// 与第2行对应完成反转
			head = temp;// 与第1行对应完成头结点移动
		}
		return pre;
	}</span>

3、反转链表-递归

<span style="font-family:Microsoft YaHei;font-size:14px;">// 将单链表反转,递归
	public static Node reverseListRec(Node head) {
		if (head == null || head.next == null)
			return head;
		Node reHead = reverseListRec(head.next);
		head.next.next = head;
		head.next = null;
		return reHead;
	}</span>

4、查找链表倒数第K个节点

双指针法,不多解释

<span style="font-family:Microsoft YaHei;font-size:14px;">// 查找单链表倒数第K个结点 双指针法
	public Node reKNode(Node head, int k) {
		if (head == null)
			return head;
		int len = getListLen(head);
		if (k > len)
			return null;
		Node targetK = head;
		Node nextK = head;
		// 先走到K个位置
		for (int i = 0; i < k; i++) {
			nextK = nextK.next;
		}
		// 再和头结点一起走,nextk走到结尾,此时targetk为倒数第K个节点
		while (nextK != null) {
			nextK = nextK.next;
			targetK = targetK.next;
		}
		return targetK;
	}</span>

5、查找链表中间节点

快慢指针,不多解释

<span style="font-family:Microsoft YaHei;font-size:14px;">public Node getMid(Node head) {
		// 类似的快慢指针法
		// 安全性检查
		if (head == null || head.next == null)
			return head;
		Node target = head;
		Node temp = head;
		while (temp != null && temp.next != null) {
			target = target.next;
			temp = temp.next.next;
		}
		return target;
	}</span>

6、判断链表是否有环

主要还是快慢指针,如果快的指针能够追上慢指针则有环

<span style="font-family:Microsoft YaHei;font-size:14px;">// 判断一个单链表中是否有环,快慢指针
	public boolean hasCycle(Node head) {
		boolean flag = false;
		Node p1 = head;
		Node p2 = head;
		while (p1 != null && p2 != null) {
			p1 = p1.next;
			p2 = p2.next.next;
			if (p2 == p1) {
				flag = true;
				break;
			}
		}
		return flag;
	}</span>

7、从尾到头打印单链表-递归

<span style="font-family:Microsoft YaHei;font-size:14px;">// 从尾到头打印单链表(递归)
	public void reList1(Node head) {
		// 安全性检查
		if (head == null)
			return;
		else {
			reList1(head.next);
			System.out.println(head.value);
		}

	}</span>

8、从尾到头打印单链表-栈

利用栈FILO的性质,先存储节点然后输出每个栈的节点值

<span style="font-family:Microsoft YaHei;font-size:14px;">// 从尾到头打印单链表(栈)
	public void reList2(Node head) {
		Stack<Node> s = new Stack<Node>();
		while (head != null) {
			s.push(head);
			head = head.next;
		}
		while (!s.isEmpty()) {
			System.out.println(s.pop().value);
		}
	}</span>

9、由小到大合并有序单链表-循环

<span style="font-family:Microsoft YaHei;font-size:14px;">	// 由小到大合并俩个有序的单链表(循环)
	public Node mergeSort1(Node head1, Node head2) {
		// 安全性检查
		if (head1 == null)
			return head2;
		if (head2 == null)
			return head1;
		// 新建合并节点
		Node target = null;
		// 确定第一个元素的节点
		if (head1.value > head2.value) {
			target = head2;
			head2 = head2.next;
		} else {
			target = head1;
			head1 = head1.next;
		}
		target.next = null;
		// 开始合并
		Node mergeHead = target;
		while (head1 != null && head2 != null) {
			// 当两个链表都不为空
			if (head1.value > head2.value) {
				target.next = head2;
				head2 = head2.next;
			} else {
				target.next = head1;
				head1 = head1.next;
			}
			target = target.next;
			target.next = null;
		}
		if (head1 == null)
			target.next = head2;
		else
			target.next = head1;
		return mergeHead;

	}</span>

10、由小到大合并有序单链表-递归

<span style="font-family:Microsoft YaHei;font-size:14px;">// 由小到大合并俩个有序的单链表(递归)
	public Node mergeSort2(Node head1, Node head2) {
		if (head1 == null)
			return head2;
		if (head2 == null)
			return head1;
		if (head1.value > head2.value) {
			head2.next = mergeSort2(head2.next, head1);
			return head2;
		} else {
			head1.next = mergeSort2(head1.next, head2);
			return head1;
		}
	}</span>

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-08 09:45:20

java单链表常用操作的相关文章

java数据结构:单链表常见操作代码实现

一.概述: 本文主要总结单链表常见操作的实现,包括链表结点添加.删除:链表正向遍历和反向遍历.链表排序.判断链表是否有环.是否相交.获取某一结点等. 二.概念: 链表: 一种重要的数据结构,HashMap等集合的底层结构都是链表结构.链表以结点作为存储单元,这些存储单元可以是不连续的.每个结点由两部分组成:存储的数值+前序结点和后序结点的指针.即有前序结点的指针又有后序结点的指针的链表称为双向链表,只包含后续指针的链表为单链表,本文总结的均为单链表的操作. 单链表结构: Java中单链表采用No

Java单链表、双端链表、有序链表实现

Java单链表.双端链表.有序链表实现 原创 2014年03月31日 23:45:35 标签: Java / 单链表 / 双端链表 / 有序链表 65040 单链表: insertFirst:在表头插入一个新的链接点,时间复杂度为O(1) deleteFirst:删除表头的链接点,时间复杂度为O(1) 有了这两个方法,就可以用单链表来实现一个栈了,见http://blog.csdn.net/a19881029/article/details/22579759 find:查找包含指定关键字的链接点

java 单链表的实现

package liaobiao;//链表测试public class Node { private int value; private Node next; //存放下一个节点的指针 //构造方法,进行传参 public Node(int value){ this.value = value; } public Node(){ } //像链表的尾部加入元素,需要先找到最后一个节点 public void addNode(Node n ){ //需要先找到最后一个节点 Node node =t

表单的常用操作

知识点: 表单是前后端数据交互的一种重要方式,使用js操作表单也是十分常见的.不过好像每次到表单操作我都要去查API,所以本次想对表单的常用操作做个小结,以备后面随时查看. 首先,我们要知道如下的一些知识: 1. 表单字段在向后台提交数据时,使用的是表单控件的name属性的value,与id无关.(大家可以在百度搜细说表单找到Fish Li大神的这篇文章看看) 2. 表单向服务端传数据时会经过编码.目前基本上只会只使用二种编码规则:application/x-www-form-urlencode

JAVA单链表的实现-不带头结点但带有尾指针

1,本程序实现了线性表的链式存储结构.实现的链表带有两个指针,一个始终指向链表中的第一个结点,另一个指针始终指向链表中的最后一个结点. 之所以设置尾指针,是因为,在插入元素到链表中的末尾时,可以通过尾指针直接找到链表的最后一个元素,从而不需要遍历链表就可以完成插入操作. 2,具体实现链表的类名为LList2.java,它首先实现了线性表的接口ListInterface,该接口的定义见:http://www.cnblogs.com/hapjin/p/4549492.html LList2.java

java单链表

单链表 一.单链表的概念 链表是最基本的数据结构,其存储的你原理图如下图所示 上面展示的是一个单链表的存储原理图,简单易懂,head为头节点,他不存放任何的数据,只是充当一个指向链表中真正存放数据的第一个节点的作用,而每个节点中都有一个next引用,指向下一个节点,就这样一节一节往下面记录,直到最后一个节点,其中的next指向null. 链表有很多种,比如单链表,双链表等等.我们就对单链表进行学习,其他的懂了原理其实是一样的. 二.用java实现单链表 语言只是一种工具,数据结构真正体会的是那种

基于单链表的操作

单链表 功能设计 1从首元结点开始输出数据域即p->data直到p->next=NULL.typedef struct Node 定义一个链式存储的单链表Node *为结构体指针类型例如对于单链表Lp=L-〉next通过p->data 访问该元素的数据值. 2creatlist(linklist *H) 从空表开始每次读入数据生成新结点将读入的数据存放到新结点的数据域中然后将新结点插入到当前链表的表头结点之后直至读入结束位置为止. 3Leng(linklist *H ) 在单链表中整个链

java单链表代码实现

用惯了C++,java写起来果然不太爽...不废话了,上代码... package javaInnerclassDemo; class Link{ class Node{ private String name; private Node next; public Node(String name){ this.name=name; } public void setname(String name){ this .name = name; } public String getname(){

单链表的操作和指针函数基础

/* ============================================================================ Name : TestLinkedList.c Author : lf Version : Copyright : Your copyright notice Description : 1 单链表的增删改查等操作 2 指针函数的使用.如示例中的:void (*visit)(link) ==========================