zoukankan      html  css  js  c++  java
  • LintCode 16. 带重复元素的排列

    写在前面:这题和全排列不含重复元素的那题几乎一样,我比较垃圾,就用HashSet去掉了重复的元素但是看了九章算法的答案也没看懂,他写的很有感觉。

    用了hash,本来想着怎么写hashcode()和equal()方法的,哪知道都帮我写好了,Integer类型的元素存储在List中的hashcode()和equal()的方法可以直接使用
    下面是源代码和我的测试代码:

    //这是ArrayList父类public abstract class AbstractList<E>中的源码
    public int hashCode() {
            int hashCode = 1;
            for (E e : this)
                hashCode = 31*hashCode + (e==null ? 0 : e.hashCode());
            return hashCode;
        }
    
    public boolean equals(Object o) {
            if (o == this)
                return true;
            if (!(o instanceof List))
                return false;
    
            ListIterator<E> e1 = listIterator();
            ListIterator<?> e2 = ((List<?>) o).listIterator();
            while (e1.hasNext() && e2.hasNext()) {
                E o1 = e1.next();
                Object o2 = e2.next();
                if (!(o1==null ? o2==null : o1.equals(o2)))
                    return false;
            }
            return !(e1.hasNext() || e2.hasNext());
        }
    @Test
        public void test() {
            List<Integer> list1 = new ArrayList<>();
            list1.add(1);
            List<Integer> list2 = new ArrayList<>();
            list2.add(2);
            List<Integer> list3 = new ArrayList<>();
            list3.add(1);
    
            System.out.println(list1.hashCode());
            System.out.println(list2.hashCode());
            System.out.println(list3.hashCode());
            System.out.println(list1==list2);
            System.out.println(list1==list3);
            System.out.println(list2==list3);
            System.out.println(list1.equals(list2));
            System.out.println(list1.equals(list3));
            System.out.println(list2.equals(list3));
            }
    //运行结果
    //        32
    //        33
    //        32
    //        false
    //        false
    //        false
    //        false
    //        true
    //        false
    

    下面是我写的代码,只是使用HashSet去重

    import org.junit.Test;
    
    import java.util.ArrayList;
    import java.util.HashSet;
    import java.util.List;
    
    public class permuteUnique {
        /**
         * @param : A list of integers
         * @return: A list of unique permutations
         * <p>
         * 16. 带重复元素的排列
         * 给出一个具有重复数字的列表,找出列表所有不同的排列。
         * <p>
         * 样例
         * 给出列表 [1,2,2],不同的排列有:
         * <p>
         * [
         * [1,2,2],
         * [2,1,2],
         * [2,2,1]
         * ]
         * 挑战
         * 使用递归和非递归分别完成该题。
         * <p>
         * 写过不含重复元素的全排列,这提是含重复元素的,应该相差不大
         */
        public List<List<Integer>> permuteUnique(int[] nums) {
            // write your code here
            HashSet<List<Integer>> hashResult = new HashSet<>();
            List<List<Integer>> result = new ArrayList<>();
            List<Integer> list = new ArrayList<>();
            if (nums == null) {
                return result;
            }
            if (nums.length == 0) {
                result.add(list);
                return result;
            }
            boolean[] color = new boolean[nums.length];
            for (int i = 0; i < nums.length; i++) {
                color[i] = false;
            }
            dfs(hashResult, list, nums, color);
            result = new ArrayList<>(hashResult);
            return result;
        }
    
    
        public void dfs(HashSet<List<Integer>> hashResult, List<Integer> list, int[] nums, boolean[] color) {
            if (list.size() == nums.length) {
                hashResult.add(new ArrayList<>(list));
                return;
            }
    
            for (int i = 0; i < nums.length; i++) {
                if (color[i] == false) {
                    list.add(nums[i]);
                    color[i] = true;
                    dfs(hashResult, list, nums, color);
                    color[i] = false;
                    list.remove(list.size() - 1);
                }
            }
        }
    
        @Test
        public void testPermuteUnique() {
    //        List<List<Integer>> result = permuteUnique(new int[]{1, 2, 2});
    //        for (int i = 0; i < result.size(); i++) {
    //            System.out.println(result.get(i).toString());
    //        }
    
        }
    }
    

    下面是九章的答案,写的很难懂,狠巧妙
    和没有重复元素的 Permutation 一题相比,只加了两句话:

    Arrays.sort(nums) // 排序这样所有重复的数
    if (i > 0 && nums[i] == nums[i - 1] && !visited[i - 1]) { continue; } // 跳过会造成重复的情况

        public List<List<Integer>> permuteUnique(int[] nums) {
            List<List<Integer>> results = new ArrayList<>();
            if (nums == null) {
                return results;
            }
    
            Arrays.sort(nums);
            dfs(nums, new boolean[nums.length], new ArrayList<Integer>(), results);
    
            return results;
        }
    
    
          private void dfs(int[] nums,
                         boolean[] visited,
                         List<Integer> permutation,
                         List<List<Integer>> results) {
            if (nums.length == permutation.size()) {
                results.add(new ArrayList<Integer>(permutation));
                return;
            }
    
            for (int i = 0; i < nums.length; i++) {
                if (visited[i]) {
                    continue;
                }
                if (i > 0 && nums[i] == nums[i - 1] && !visited[i - 1]) {
                    continue;
                }
    
                permutation.add(nums[i]);
                visited[i] = true;
                dfs(nums, visited, permutation, results);
                visited[i] = false;
                permutation.remove(permutation.size() - 1);
            }
        } 
  • 相关阅读:
    Jmeter设计压力测试场景&请求元件之并发场景设置&Jmeter查看压力测试结果&压力测试结果分析(二十九)
    Jmeter录制APP脚本(二十八)
    Jmeter优化web脚本&Jmeter回放web脚本和联调&Jmeter WEB脚本参数化(二十七)
    Jmeter结合badboy录制脚本(二十六)
    Jmeter录制WEB的原理(二十五)
    Jmeter实践:一粒云项目—Jmeter获取文件列表与下载接口串联测试及上传文件与下载接口之间的串联测试(二十四)
    Jmeter实践:一粒云项目—Jmeter完成文件的列表获取及JSON Extractor获取数组类型的数据及正则表达式获取数组类型的数据(二十三)
    Jmeter实践:一粒云项目—Jmeter完成文件下载及批量完成文件下载(二十二)
    Jmeter实践:一粒云项目—Jmeter完成文件上传及批量完成文件上传(二十一)
    Jmeter实践——新的项目介绍(二十)
  • 原文地址:https://www.cnblogs.com/wei1/p/9582055.html
Copyright © 2011-2022 走看看