首页 > 编程语言 > 详细

链表-删除排序链表中的重复元素

时间:2018-09-23 22:04:44      阅读:162      评论:0      收藏:0      [点我收藏+]

leetcode(使用的是中文网站:领扣):83 

给定一个排序链表,删除所有含有重复数字的节点,只保留原始链表中 没有重复出现 的数字。

示例 1:

输入: 1->2->3->3->4->4->5
输出: 1->2->5

示例 2:

输入: 1->1->1->2->3
输出: 2->3

 
一开始没有看到排序链表这个条件,所以解法有一点麻烦
解法一:
思路:再创建一个链表B,然后遍历原来的链表A,将A中的元素添加到B的链表尾部。
每次向B中添加元素时都检查一下B中是否已经存在这个元素,若存在,则不添加。代码如下:
package leetcode.LinkedList83;

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
class Solution {
    public ListNode deleteDuplicates(ListNode head) {
        if(head == null) {
            return null;
        }
        //原链表的当前节点
        ListNode cur = head.next;

        //创建一个新的链表
        ListNode newHead = new ListNode(head.val);
        ListNode newList_cur = newHead;

        while(cur!=null){
            //如果新的链表中没有这个节点,那么就添加进去
            if(!contains(newHead,cur.val)){
                newList_cur.next = new ListNode(cur.val);
                newList_cur = newList_cur.next;
            }
            cur = cur.next;
        }
        return newHead;
    }

    //检测链表中是否包含某个元素
    boolean contains(ListNode head,int val){
        while(head != null){
            if(head.val == val){
                return true;
            } else {
                head = head.next;
            }
        }
        return false;
    }
    public static void main(String[] args) {
        int arr[] = {1,2,3,4,3,5,6};
        ListNode head = new ListNode(arr);
        System.out.println(head);

        ListNode ret = (new Solution()).deleteDuplicates(head);
        System.out.println("最终结果:"+ret);
    }
}

 

这里为了测试代码,我们这里可以写一个将数组转化成链表的函数

本地测试用代码(后面的代码测试也是用的这个):

package leetcode.LinkedList83;

/**
 * 用于本地测试
 */
class ListNode {
  int val;
  ListNode next;
  ListNode(int x) { val = x; }


  // 链表节点的构造函数
  // 使用arr为参数,创建一个链表,当前的ListNode为链表头结点
  ListNode(int[] arr){
    if(arr == null || arr.length == 0){
      throw new IllegalArgumentException("arr can not be empty");
    }

    this.val = arr[0];
    ListNode cur = this;
    for (int i = 1; i < arr.length; i++) {
      cur.next = new ListNode(arr[i]);
      cur = cur.next;
    }
  }

  @Override
  public String toString(){
    StringBuilder sb = new StringBuilder();
    ListNode cur = this;
    while(cur != null){
      sb.append(cur.val);
      sb.append("->");
      cur = cur.next;
    }
    sb.append("NULL");
    return sb.toString();
  }
}

 

解法二(这里才用到题目的已经排序的条件):

思路:使用递归,在递推之后回归的过程中我们可以比较当前链表和子链的值是否相等,相等的话就直接返回子链,否则就返回当前链表。

链表的递归,可以想象成不断的把问题交给子链去解决,最后返回的时候也是不断的返回子链的结果。

package leetcode.LinkedList83;
public class Solution2 { public ListNode deleteDuplicates(ListNode head) { if(head == null || head.next == null) { return head; } else { // ListNode res = deleteDuplicates(head.next); // if(head.val == head.next.val){ // return res; // } else { // head.next = res; // return head; // } head.next = deleteDuplicates(head.next); return head.val == head.next.val ? head.next:head; } } public static void main(String[] args) { int arr[] = {1,2,3,3,5,6}; ListNode head = new ListNode(arr); System.out.println(head); ListNode ret = (new Solution2()).deleteDuplicates(head); System.out.println("最终结果:"+ret); } }

简单的画个图表示一下(用res更好理解一点):

技术分享图片

 

 最后是官方给出的使用迭代的方法:

解法三:

思路:每一次都比较当前节点和下一个节点的值,如果不同,就将当前节点后移,如果相同,就不改变当前节点,直至下一个节点的值与当前节点的值不相等。

public ListNode deleteDuplicates(ListNode head) {
    ListNode current = head;
    while (current != null && current.next != null) {
        if (current.next.val == current.val) {
            current.next = current.next.next;
        } else {
            current = current.next;
        }
    }
    return head;
}

 

刚刚开始接触leetcode,许多解法都还没怎么分析复杂度,所以自己的解法也不一定是什么好的方法,大家简单看下思路就好。
错漏之处请在评论区之处。
 

链表-删除排序链表中的重复元素

原文:https://www.cnblogs.com/bax-life/p/9693635.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!