题目
这道题和之前的三数之和有着一样的思想,我也就不再过多介绍了,如果不理解的可以看看我之前讲三数之和的文章
这里就直接贴代码了
class Solution {
public List<List<Integer>> fourSum(int[] nums, int target) {
Set<List<Integer>> set=new HashSet<List<Integer>>();
List<List<Integer>> res=new ArrayList<List<Integer>>();
Arrays.sort(nums);
int n=nums.length;
int L=0,l=L+1,r=l+1,R=n-1;
int flag=0;
while(r<R){
int s=nums[l]+nums[L]+nums[R]+nums[r];
if(s==target){
set.add(Arrays.asList(nums[L],nums[l],nums[r],nums[R]));
r++;
R--;
}else if(s<target) r++;
else R--;
if(r>=R && flag==0){
l++;
r=l+1;
R=n-1;
}
if(l>=R-1){
L++;
l=L+1;
r=l+1;
R=n-1;
}
if(L>=R-2) break;
}
res=new ArrayList<>(set);
return res;
}
}
我这个代码虽然道理简单,可惜实际的运行效率却是十分低下,中间使用了set是为了直接去除重复项
在这里呢,我再给大家贴一个大佬的解决方案,效率比我的高许多
public List<List<Integer>> fourSum(int[] nums,int target){
/*定义一个返回值*/
List<List<Integer>> result=new ArrayList<>();
/*当数组为null或元素小于4个时,直接返回*/
if(nums==null||nums.length<4){
return result;
}
/*对数组进行从小到大排序*/
Arrays.sort(nums);
/*数组长度*/
int length=nums.length;
/*定义4个指针k,i,j,h k从0开始遍历,i从k+1开始遍历,留下j和h,j指向i+1,h指向数组最大值*/
for(int k=0;k<length-3;k++){
/*当k的值与前面的值相等时忽略*/
if(k>0&&nums[k]==nums[k-1]){
continue;
}
/*获取当前最小值,如果最小值比目标值大,说明后面越来越大的值根本没戏*/
int min1=nums[k]+nums[k+1]+nums[k+2]+nums[k+3];
if(min1>target){
break;
}
/*获取当前最大值,如果最大值比目标值小,说明后面越来越小的值根本没戏,忽略*/
int max1=nums[k]+nums[length-1]+nums[length-2]+nums[length-3];
if(max1<target){
continue;
}
/*第二层循环i,初始值指向k+1*/
for(int i=k+1;i<length-2;i++){
/*当i的值与前面的值相等时忽略*/
if(i>k+1&&nums[i]==nums[i-1]){
continue;
}
/*定义指针j指向i+1*/
int j=i+1;
/*定义指针h指向数组末尾*/
int h=length-1;
/*获取当前最小值,如果最小值比目标值大,说明后面越来越大的值根本没戏,忽略*/
int min=nums[k]+nums[i]+nums[j]+nums[j+1];
if(min>target){
continue;
}
/*获取当前最大值,如果最大值比目标值小,说明后面越来越小的值根本没戏,忽略*/
int max=nums[k]+nums[i]+nums[h]+nums[h-1];
if(max<target){
continue;
}
/*开始j指针和h指针的表演,计算当前和,如果等于目标值,j++并去重,h--并去重,当当前和大于目标值时h--,当当前和小于目标值时j++*/
while (j<h){
int curr=nums[k]+nums[i]+nums[j]+nums[h];
if(curr==target){
result.add(Arrays.asList(nums[k],nums[i],nums[j],nums[h]));
j++;
while(j<h&&nums[j]==nums[j-1]){
j++;
}
h--;
while(j<h&&i<h&&nums[h]==nums[h+1]){
h--;
}
}else if(curr>target){
h--;
}else {
j++;
}
}
}
}
return result;
}
作者:you-wei-wu
链接:https://leetcode-cn.com/problems/4sum/solution/ji-bai-9994de-yong-hu-you-dai-ma-you-zhu-shi-by-yo/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
大佬的代码注释也非常清楚,如果有什么不懂的地方,可以大家一起交流