Let me define the problem clearly. Given an array of size n and a number X, you are supposed to find a pair of numbers that sums to X. Only one pair would be enough.
Lets see how we can find the solution for this in O(n) using a HashSet. Yes, HashSet is a costly data structure to use, but lets consider this primitive just due to the linear order it provides.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package dsa.arrays; | |
import java.util.HashSet; | |
import java.util.Set; | |
public class FindSumHashSet { | |
public static void main(String a[]){ | |
FindSumHashSet sumFinder = new FindSumHashSet(); | |
sumFinder.begin(); | |
} | |
public void begin(){ | |
int[] sampleArray = getRandomArray(20); | |
int randomSum = sampleArray[15] + sampleArray[10]; | |
System.out.print("ARRAY : "); | |
for(int i:sampleArray){ | |
System.out.print(i+" "); | |
} | |
System.out.println(); | |
findPair(sampleArray,randomSum); | |
} | |
public void findPair(int[] sampleArray, int randomSum) { | |
Set<Integer> sampleArraySet = new HashSet<Integer>(); | |
for(int i=0;i<sampleArray.length;i++){ | |
sampleArraySet.add(sampleArray[i]); | |
int valueToFind = randomSum - sampleArray[i]; | |
if(sampleArraySet.contains(valueToFind)){ | |
System.out.println("SUM : "+randomSum); | |
System.out.println("PAIR : "+valueToFind+","+sampleArray[i]); | |
break; | |
} | |
} | |
} | |
private int[] getRandomArray(int size) { | |
int[] randomArray = new int[size]; | |
for(int i=0;i<size;i++){ | |
randomArray[i] = (int)(Math.random()*10); | |
} | |
return randomArray; | |
} | |
} | |
//SAMPLE OUTPUTS | |
//ARRAY : 7 3 6 4 3 4 7 7 5 1 4 6 2 4 1 7 5 8 9 7 | |
//SUM : 11 | |
//PAIR : 7,4 | |
//ARRAY : 0 2 9 6 0 7 6 5 1 7 9 0 7 1 2 4 4 3 9 0 | |
//SUM : 13 | |
//PAIR : 6,7 |
We shall improvise on this problem in space domain using fancy sorts and those we shall see in the coming posts.
Cheers,
Bragaadeesh
5 comments:
Considering that the members are not ordered, the best bet would be to use a hash based datastructure. I am not sure why you generalize HashSet as expensive. List or an array would do horribly when doing a linear search with this data (as you know, binary search is impossible here). I cant think of any other datastructure you could fit in. Also, the performance of HashSet depends on the efficient implementation of the hash. So, there could be a cost you would pay for larger datasets but i am sure that would be better than linear search any day.
@Arun: I completely accept your argument. The reason i want to solve the problem without using Hash is because that is what interviewers ask :) these days. They tell you not to solve using Hash. And they may say it should not be O(n*n) either.
You can easily tackle this question in O(n lg n) time by first sorting the array using in-place comparison sort like quick sort. After sorting, you can have two pointers pointed to the head and the tail, then find the sum.
it might be a good idea to use BST here. create a BST of n numbers (nlogn). then, for each number in an array (of size n), subtract it from TargetSum and then look for the other number in binary tree (logn) which should take (nlogn) total time ... the overall time would (nlogn + nlogn) which is nlogn... please correct me if I am wrong.
@vijay : yes that is a very good idea to solve the problem!!
Post a Comment