1.链表的引入
由于在ArrayList任意位置插入或者删除元素时就需要将元素整体往前或者往后搬移 ,时间复杂度为O(n),效率比较低,因此ArrayList不适合在任意位置插入或者删除元素,因此引入了LinkedList,即链表结构。
2.链表的分类
2.1 单向或者双向
无头指每一个节点都可以存放数字
2.2 带头或者不带头
data:image/s3,"s3://crabby-images/59df3/59df3403298702e14ce669030295ca5beeec59d4" alt=""
2.3 循环或者非循环
data:image/s3,"s3://crabby-images/d620f/d620ff209325b4829759881f4d363f8861f5afba" alt=""
无头单向非循环结构简单,作为其他数据结构的子结构,在笔试面试中出现的比较多
同时,无头双向循环链表,为LinkedList底层实现的链表。
data:image/s3,"s3://crabby-images/894cf/894cf98d9759cf2590d5185f5b3e9dc8669d8286" alt=""
3.链表的实现(无头单项非循环)
import java.util.LinkedList;
public class MyLinkedList implements IList{
static class ListNode {
public int val;
public ListNode preV;
public ListNode next;
public ListNode(int val) {
this.val = val;
}
}
public ListNode head;//这里的head并不是指头而是方便取的名字
public ListNode last;
@Override
public void addFirst(int data) {
ListNode cur = head;
ListNode node = new ListNode(data);
if(head.next == null){
head = last = node;
}else{
node.next = cur;
cur.preV = node;
head = node;
}
}
@Override
public void addLast(int data) {
ListNode cur = head;
ListNode node = new ListNode(data);
if(head == null) {
head = last = node;
}else{
last.next = node;
node.preV = last;
last = node;
}
}
@Override
//在指定位置增删元素
public void addIndex(int index, int data) {
int len = size();
if(index < 0 || index > len ){
System.out.println("不符合条件");
return;
}if(index == 0){
addFirst(index);
return;
}if(index == len){
addLast(data);
return;
}else{
ListNode cur = findOfKey(index);
ListNode node = new ListNode(data);
node.next = cur;
node.preV = cur.preV;
cur.preV.next = node;
cur.preV = node;
}
}
private ListNode findOfKey(int index){
ListNode cur = head;
while(index != 0){
cur = cur.next;
index--;
}
return cur;
}
@Override
public boolean contains(int key) {
ListNode cur = head;
while(cur != null){
if(cur.val == key)
{
return true;
}
cur = cur.next;
}
return false;
}
@Override
public void remove(int key) {
ListNode cur = head;
while(cur != null) {
if (cur.val == key) {
if (cur == head) {
head = head.next;
if (head != null) {
head.preV = null;
}
} else {
cur.preV.next = cur.next;
if ( cur.next == null) {
last = last.preV;
}else {
cur.next.preV = cur.preV;
}
}
}
cur =cur.next;
}
}
@Override
public void removeAllKey(int key) {
ListNode cur = head;
while(cur != null) {
if (cur.val == key) {
if (cur == head) {
head = head.next;
if (head != null) {
head.preV = null;
}
}
else {
cur.preV.next = cur.next;
if (cur.next == null) {
last = last.preV;
}
cur.next.preV = cur.preV;
}
}
cur = cur.next;
}
}
@Override
public int size() {
ListNode cur = head;
int len = 0;
while(cur != null){
len++;
cur = cur.next;
}
return len;
}
@Override
public void clear() {
ListNode cur = head;
if(cur != null){
ListNode curN = cur.next;
cur.preV = null;
cur.next = null;
cur = curN;
}
head = null;
last = null;
}
@Override
public void display() {
ListNode cur = head;
while(cur != null){
System.out.print(cur.val+" ");
cur = cur.next;
}
System.out.println();
}
接口:
public interface IList {
public void addFirst(int data);
public void addLast(int data);
public void addIndex(int index,int data);
public boolean contains (int key);
public void remove(int key);
public void removeAllKey(int key);
public int size();
public void clear();
public void display();
}
4.链表面试题
4.1 移除链表元素
data:image/s3,"s3://crabby-images/e40ca/e40ca6e548da76eee2ab8e1d108af63d15679085" alt=""
public class Solution {
public ListNode removeElements(ListNode head, int val) {
if(head == null){
return head;
}
ListNode pre = head;
ListNode cur = head.next;
while(cur != null){
if(cur.val == val){
pre.next = cur.next;
cur = cur.next;
}else{
pre = cur;
cur = cur.next;
}
}
if(head.val == val){
head = head.next;
}
return head;
}
}
4.2 反转链表
data:image/s3,"s3://crabby-images/ee671/ee671f0feef354283bfc7d55f0585b51b99989eb" alt=""
class Solution {
public ListNode reverseList(ListNode head) {
if(head == null){
return head;
}
ListNode cur = head.next;
head.next = null;
while(cur != null){
ListNode curN = cur.next;
cur.next = head;
head = cur;
cur = curN;
}
return head;
}
}
4.3 链表的中间结点
data:image/s3,"s3://crabby-images/38725/38725944089b7b0e2e24fc09d47269614f9df5ff" alt=""
class Solution {
public ListNode middleNode(ListNode head) {
if(head == null){
return head;
}
ListNode fast = head;
ListNode slow = head;
while(fast != null && fast.next != null){
fast = fast.next.next;
slow = slow.next;
}
return slow;
}
}
4.4 返回倒数第k个节点
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode(int x) { val = x; }
* }
*/
class Solution {
public int kthToLast(ListNode head, int k) {
if(head == null){
return head.val;
}
ListNode fast = head;
ListNode slow = head;
int count = k - 1;
while(count != 0){
fast = fast.next;
count--;
}
while(fast.next != null){
fast = fast.next;
slow = slow.next;
}
return slow.val;
}
}
4.5 合并两个有序链表
data:image/s3,"s3://crabby-images/5e87d/5e87ddcc7ada513f6cd96de969ec5354c3941d35" alt=""
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution {
public ListNode mergeTwoLists(ListNode list1, ListNode list2) {
ListNode newHead = new ListNode(-1);
ListNode tmp = newHead;
while(list1 != null && list2 != null){
if(list1.val <= list2.val){
tmp.next = list1;
list1 = list1.next;
tmp = tmp.next;
}else{
tmp.next = list2;
tmp = tmp.next;
list2 = list2.next;
}
}
if(list1 != null){
tmp.next = list1;
}
if(list2 != null){
tmp.next = list2;
}
return newHead.next;
}
}
4.6 链表分割
data:image/s3,"s3://crabby-images/b1225/b12259041f4ebab0ba3f415d14827555d1ea6816" alt=""
public class Partition {
public ListNode partition(ListNode pHead, int x) {
ListNode as = null;
ListNode ae = null;
ListNode bs = null;
ListNode be = null;
ListNode cur = pHead;
while(cur != null){
if(cur.val < x){
if(as == null){
as = ae = cur;
}else{
ae.next = cur;
ae = ae.next;
}
}else{
if(bs == null){
bs = be =cur;
}else{
be.next = cur;
be = be.next;
}
}
cur = cur.next;
}
if(as == null){
return bs;
}
ae.next = bs;
if(bs != null){
be.next = null;
}
return as;
}
}
4.7 链表回文
data:image/s3,"s3://crabby-images/4c5af/4c5aff66678e6ba6df9b9899fc2ee20508a6c115" alt=""
public class PalindromeList {
public boolean chkPalindrome(ListNode A) {
if(A == null){
return false;
}
// write code here
ListNode fast = A;
ListNode slow = A;
while(fast != null && fast.next != null){
fast = fast.next.next;
slow = slow.next;
}
ListNode cur = slow.next;
while(cur != null){
ListNode curN = cur.next;
cur.next = slow;
slow = cur;
cur = curN;
}
while(A != slow){
if(A.val != slow.val){
return false;
}
if(A.next == slow){
return true;
}
A= A.next;
slow = slow.next;
}
return true;
}
}
4.8 相交链表
data:image/s3,"s3://crabby-images/58956/5895677c289e358b253bcba812dcf718707a24a0" alt=""
public class Solution {
public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
ListNode pl = headA;
ListNode ps = headB;
int lenA = 0;
int lenB = 0;
//计算长度
while(pl != null){
lenA++;
pl = pl.next;
}
while(ps != null){
lenB++;
ps = ps.next;
}
pl = headA;
ps = headB;
//计算两个链表的差值
int len = lenA - lenB;
if(len < 0){
pl = headB;
ps = headA;
len = lenB - lenA;
}
while(len != 0){
pl = pl.next;
len--;
}
while(pl != ps){
pl = pl.next;
ps = ps.next;
}
if(pl == null){
return null;
}
return pl;
}
}
4.9 环形链表
data:image/s3,"s3://crabby-images/9c827/9c827089103b781ddf63511c22085ab665c7477a" alt=""
public class Solution {
public boolean hasCycle(ListNode head) {
ListNode fast = head;
ListNode slow = head;
while(fast != null && fast.next != null){
fast = fast.next.next;
slow = slow.next;
if(fast == slow ){
return true;
}
}
return false;
}
}
5.双向链表的基本实现(LinkedList)
public class MyLinkedList implements IList{
static class ListNode {
public int val;
public ListNode preV;
public ListNode next;
public ListNode(int val) {
this.val = val;
}
}
public ListNode head;
public ListNode last;
@Override
public void addFirst(int data) {
ListNode cur = head;
ListNode node = new ListNode(data);
if(head.next == null){
head = last = node;
}else{
node.next = cur;
cur.preV = node;
head = node;
}
}
@Override
public void addLast(int data) {
ListNode cur = head;
ListNode node = new ListNode(data);
if(head == null) {
head = last = node;
}else{
last.next = node;
node.preV = last;
last = node;
}
}
@Override
//在指点位置增删元素
public void addIndex(int index, int data) {
int len = size();
if(index < 0 || index > len ){
System.out.println("不符合条件");
return;
}if(index == 0){
addFirst(index);
return;
}if(index == len){
addLast(data);
return;
}else{
ListNode cur = findOfKey(index);
ListNode node = new ListNode(data);
node.next = cur;
node.preV = cur.preV;
cur.preV.next = node;
cur.preV = node;
}
}
private ListNode findOfKey(int index){
ListNode cur = head;
while(index != 0){
cur = cur.next;
index--;
}
return cur;
}
@Override
public boolean contains(int key) {
ListNode cur = head;
while(cur != null){
if(cur.val == key)
{
return true;
}
cur = cur.next;
}
return false;
}
@Override
public void remove(int key) {
ListNode cur = head;
while(cur != null) {
if (cur.val == key) {
if (cur == head) {
head = head.next;
if (head != null) {
head.preV = null;
}
} else {
cur.preV.next = cur.next;
if ( cur.next == null) {
last = last.preV;
}else {
cur.next.preV = cur.preV;
}
}
}
cur =cur.next;
}
}
@Override
public void removeAllKey(int key) {
ListNode cur = head;
while(cur != null) {
if (cur.val == key) {
if (cur == head) {
head = head.next;
if (head != null) {
head.preV = null;
}
}
else {
cur.preV.next = cur.next;
if (cur.next == null) {
last = last.preV;
}
cur.next.preV = cur.preV;
}
}
cur = cur.next;
}
}
@Override
public int size() {
ListNode cur = head;
int len = 0;
while(cur != null){
len++;
cur = cur.next;
}
return len;
}
@Override
public void clear() {
ListNode cur = head;
if(cur != null){
ListNode curN = cur.next;
cur.preV = null;
cur.next = null;
cur = curN;
}
head = null;
last = null;
}
@Override
public void display() {
ListNode cur = head;
while(cur != null){
System.out.print(cur.val+" ");
cur = cur.next;
}
System.out.println();
}
//在指定位置进行打印
public void display2(ListNode newHead) {
ListNode cur = head;
while (cur != null) {
System.out.print(cur.val + " ");
cur = cur.next;
}
System.out.println();
}
public ListNode reverseList() {
if(head == null){
return head;
}
ListNode cur = head.next;
head.next = null;
while(cur != null){
ListNode curN = cur.next;
cur.next = head;
head = cur;
cur = curN;
}
return head;
}
}
5.1LinkedList的一些性质
data:image/s3,"s3://crabby-images/3e292/3e292d9c19c010709dcb0dfd5606af908e3b0f55" alt=""
data:image/s3,"s3://crabby-images/74b67/74b67ce6667255e4b795161a754665af2e1ddc35" alt=""
5.2 LinkedList的使用
data:image/s3,"s3://crabby-images/2d8ea/2d8eaa112ca94b0e3e175daae436e9becd87a498" alt=""
2 其他方法的使用
data:image/s3,"s3://crabby-images/3738c/3738cf8975a2c4c9c7c4b22c397ec6d82c2a8669" alt=""
3 LinkedList的遍历
data:image/s3,"s3://crabby-images/0a255/0a2556e1caf2db6d512fc7fb77c58a66251981ab" alt=""
6.ArrayList 与 LinkedList的区别
data:image/s3,"s3://crabby-images/0d4a9/0d4a9e60609262a5e5302b8b4eb4fd124bd0ccdc" alt=""