ksum问题
2sum:
Given an array of integers, return indices of the two numbers such that they add up to a specific target.
You may assume that each input would have exactly one solution.
解答:
采用hashmap,从头遍历数组,如果hashmap不含有该元素,则将target减去这个元素的值和该元素的下标存入hashmap;如果hashmap含有该元素,则将hashmap中对应的下标和该元素的下标存入数组返回结果。
public class Solution {
    public int[] twoSum(int[] nums, int target) {
        Map<Integer, Integer> helper = new HashMap<Integer, Integer>();
        int[] result = {-1, -1};
        for (int i = 0; i < nums.length; i++) {
            if (helper.get(nums[i]) != null) {
                result[0] = helper.get(nums[i]);
                result[1] = i;
                return result;
            } else {
                helper.put(target - nums[i], i);
            }
        }
        return result;
    }
}
此题若不是返回下标,还可以采用将数组先进行排序,然后利用左右两个指针寻找相应元素。若左右两指针元素和小于target,则左指针右移,若大于则右指针左移,若相等则返回结果。
    public int[] twoSum_pointer(int[] numbers, int target) {
        if (numbers == null || numbers.length < 2) {
            return null;
        }
        Arrays.sort(numbers);
        int left = 0;
        int right = numbers.length - 1;
        int[] rst = new int[2];
        while ( left < right) {
            int sum = numbers[left] +  numbers[right];
            if( sum == target){
                rst[0] = left;
                rst[1] = right;
                break;
            } else if ( sum < target) {
                left++;
            } else {
                right--;
            }
        }
        return rst;
    }
3sum:
Given an array S of n integers, are there elements a, b, c in S such that a + b + c = 0? Find all unique triplets in the array which gives the sum of zero.
Note:
- Elements in a triplet (a,b,c) must be in non-descending order. (ie, a ≤ b ≤ c)
- The solution set must not contain duplicate triplets.
解答:
运用上题2sum的第二种方法,相当于把数组的每个元素当做target,然后往后寻找2sum为target相反数的两个元素。要注意的是避免重复结果。
public class Solution {
    public List<List<Integer>> threeSum(int[] nums) {
        List<List<Integer>> result = new LinkedList<List<Integer>>();
        if (nums == null || nums.length < 3) {
            return result;
        }
        Arrays.sort(nums);
        for (int i = 0; i < nums.length - 2; i++) {
            if (i > 0 && nums[i] == nums[i - 1]) {
                continue;
            }     //避免重复元素
            int left = i + 1;
            int right = nums.length - 1;
            while (left < right) {
                if (nums[i] + nums[left] + nums[right] == 0) {
                    List<Integer> temp = new LinkedList<Integer>();
                    temp.add(nums[i]);
                    temp.add(nums[left]);
                    temp.add(nums[right]);
                    result.add(temp);
                    left++;
                    right--;
                    while(left < right && nums[left] == nums[left - 1]) {
                        left++;
                    }
                    while(left < right && nums[right] == nums[right + 1]) {
                        right--;
                    }        //避免重复元素
                } else if (nums[i] + nums[left] + nums[right] > 0) {
                    right--;
                } else {
                    left++;
                }
            }
        }
        return result;
    }
}
3sum closest:
Given an array S of n integers, find three integers in S such that the sum is closest to a given number, target. Return the sum of the three integers. You may assume that each input would have exactly one solution.
解答:同样是2sum的扩展,先将数组排序,然后遍历数组,对每个元素向后寻找两个元素并求和,和等于target直接返回target,大于则左指针右移,小于则右指针左移,同时更新result值。注意,result初始值设为Integer.MAX_VALUE的一半,而不是Integer.MAX_VALUE,否则结果可能溢出而产生错误。比如:

正确代码如下:
public class Solution {
    public int threeSumClosest(int[] nums, int target) {
        if (nums == null || nums.length < 3) {
            return -1;
        }
        Arrays.sort(nums);
        int result = Integer.MAX_VALUE / 2;
        for (int i = 0; i < nums.length - 2; i++) {
            int left = i + 1;
            int right = nums.length - 1;
            while (left < right) {
                int temp = nums[i] + nums[left] + nums[right];
                if (temp == target) {
                    return temp;
                } else if (temp < target) {
                    left++;
                } else {
                    right--;
                }
                if (Math.abs(temp - target) < Math.abs(result - target)) {
                    result = temp;
                }
            }
        }
        return result;
    }
}
4sum:
Given an array S of n integers, are there elements a, b, c, and d in S such that a + b + c + d = target? Find all unique quadruplets in the array which gives the sum of target.
Note:
- Elements in a quadruplet (a,b,c,d) must be in non-descending order. (ie, a ≤ b ≤ c ≤ d)
- The solution set must not contain duplicate quadruplets.
解答:
先将数组进行排序,然后两个for循环嵌套,固定两个数组元素,再用左右两指针扫另外两个元素,根据sum与target大小关系确定指针的移动。
public class Solution {
    public List<List<Integer>> fourSum(int[] nums, int target) {
        List<List<Integer>> result = new LinkedList<List<Integer>>();
        if (nums == null || nums.length < 4) {
            return result;
        }
        Arrays.sort(nums);
        for (int i = 0; i < nums.length - 3; i++) {
            if (i > 0 && nums[i] == nums[i - 1]) {
                continue;
            }
            for (int j = i + 1; j < nums.length - 2; j++) {
                if (j > i + 1 && nums[j] == nums[j - 1]) {
                    continue;
                }
                int left = j + 1;
                int right = nums.length - 1;
                while (left < right) {
                    int temp = nums[i] + nums[j] + nums[left] + nums[right];
                    if (temp == target) {
                        List<Integer> ele = new LinkedList<Integer>();
                        ele.add(nums[i]);
                        ele.add(nums[j]);
                        ele.add(nums[left]);
                        ele.add(nums[right]);
                        result.add(ele);
                        left++;
                        right--;
                        while (left < right && nums[left] == nums[left - 1]) {
                            left++;
                        }
                        while (left < right && nums[right] == nums[right + 1]) {
                            right--;
                        }
                    } else if (temp < target) {
                        left++;
                    } else {
                        right--;
                    }
                }
            }
        }
        return result;
    }
}
值得注意的是,本题中的数据结构采用的是LinkedList而不是ArrayList,原因是插入操作比较多,用LinkedList效率更高。前者运行时间击败63%,后者54%。
ksum问题的更多相关文章
- 2016 一中培训 day 5 ksum
		又是一天的爆零!!!!! 原本第一题 很容易做 竟然优化过度 丢了答案 1693: ksum Time Limit 1000 ms Memory Limit 524288 KBytes Judge S ... 
- 2Sum,3Sum,4Sum,kSum,3Sum Closest系列
		1).2sum 1.题意:找出数组中和为target的所有数对 2.思路:排序数组,然后用两个指针i.j,一前一后,计算两个指针所指内容的和与target的关系,如果小于target,i右移,如果大于 ... 
- kSum问题总结
		1.2Sum 题目: 方法一:两次迭代 public class TwoSum { public static int[] twoSum(int[] nums, int target) { int[] ... 
- 2sum,3sum,4sum,ksum
		1. 2sum 题目:给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标.你可以假设每种输入只会对应一个答案.但是,你不能重复利 ... 
- 【数组】kSum问题
		一.2Sum 思路1: 首先对数组排序.不过由于最后返回两个数字的索引,所以需要事先对数据进行备份.然后采用2个指针l和r,分别从左端和右端向中间运动:当l和r位置的两个数字之和小于目标数字targe ... 
- k-sum 问题
		问题描述 给定一个数组及数字 k ,从数组中找出所有相加结果为 k 的组合. 示例: 给定数组 [1,1,1] 令 k=2,输出: [[1,1]] 给定数组 [10, 1, 2, 7, 6, 1, 5 ... 
- [算法]K-SUM problem
		一.Two Sum Given an array of integers, find two numbers such that they add up to a specific target nu ... 
- 【JZOJ4815】【NOIP2016提高A组五校联考4】ksum
		题目描述 输入 输出 样例输入 3 4 1 3 4 样例输出 8 7 4 4 数据范围 样例解释 解法 二分做法 考虑到可以二分第k大的值mid,如果比mid大的区间和数小于或等于mid,那么mid就 ... 
- k-sum问题
		给定一个数组,里面的是任意整数,可能有重复,再给定一个目标T,从数组中找出所有和为T的K个数,要求结果中没有重复. Note: Elements in a quadruplet (a,b,c,d) m ... 
随机推荐
- java 四则混合运算 计算器
			public class Counter { /**用递归算法,把括号内的公式算出然后递归 * @param args */ public static void calculator (St ... 
- Tomcat8 + Redis实现session集中管理
			环境准备: 部署两台 tomcat 8.0 安装 redis 服务器 下载工具库( commons-pool2-2.3.jar.jedis-2.7.2.jar .改良版的 tomcat ... 
- SpringMVC4+MyBatis3+SQLServer 2014 整合(包括增删改查分页)
			前言 说起整合自然离开ssm,我本身并不太喜欢ORM,尤其是MyBatis,把SQL语句写在xml里,尤其是大SQL,可读性不高,出错也不容易排查. 开发环境 idea2016.SpringMVC4. ... 
- 谈JavaScript的继承
			最近在忙前端的工作,因为之前做.net和php的开发比较多,前端开发喜欢把库拿来就用,几次事实证明,不懂原理,连改代码也改不好,所以还是下定决心研究下JavaScript的几个技术难点. 0x1.Ja ... 
- JNI之C初探
			JNI是Java Native Interface的缩写,从Java1.1开始,JNI标准成为java平台的一部分,它允许Java代码和其他语言写的代码进行交互.JNI一开始是为了本地已编译语言,尤其 ... 
- 通过udev创建ASM共享磁盘(RAC)
			OS:RedHat EL6.0 Oracle: Oracle 11gR2 在Oracle 11gR2,构建RAC时可以通过ASM创建asm disk,但是需要安装asmlib相关软件:对于RedH ... 
- J2ee技术难点
			J2ee技术难点 session/cookie区别联系 jsp/servlet区别联系 filter执行流程 openSessionInView原理 clone与servilizable区别联系 eq ... 
- 第二章 mac上运行第一个appium实例
			一.打开appium客户端工具 1 检查环境是否正常运行: 点击左边第三个图标 这是测试你环境是否都配置成功了 2 执行的过程中,遇到Could not detect Mac OS ... 
- Sort List leetcode
			这个题一开始本想用快速排序的,但是想了20分钟都没有头绪,难点在于快速排序的随机访问无法用链表实现,不过如果可以实现快速排序partition函数就可以了,但是这可能比较复杂,于是改用其他排序方法,上 ... 
- 1640: [Usaco2007 Nov]Best Cow Line 队列变换
			1640: [Usaco2007 Nov]Best Cow Line 队列变换 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 543 Solved: 2 ... 
