Just aced my interview with Amazon today (got an email within minutes after the interview that I was selected to move on to the final panel interview). Thanks so much for these videos, def helped get me in the right mindset and made the coding portion of the interview a breeze.
Nick, you are the best! I’m a software developer over 7 years and I’d watched so many empty talks and crappy solutions from other people for data structure problems. You teach the technique to solve a problem rather than useless looping, mapping ext... Thanks man! Keep up the great work!
Yup Was the first idea I had as well. Allocate 2x input. Loop over the input and store the negative numbers in the first list and positive in the 2nd. Memory consumption is larger for this case. But end step is just merge the two lists. Original list could be used for this to maintain 3N memory complexity and 2N run time.
I love to watch your tutorials. It gives me a sense of relief each time I watch it. Not to mention that they are extremely helpful. I’m preparing to work at faang and every morning I come to this channel as a pilgrim
Yeah. I was so confused at the beginning because I couldnt think of a solution that would be better than O(N log n) because in every case you still have to sort the numbers. The problems got so much easier with the realization that the numbers are sorted...
If the input isn’t sorted there is no solution better than O(nlog(n)) , it is easy to prove: Assume that there is a solution in o(nlog(n))(way better than O(nlog(n))) so I can sort any array in this time complexity by first taking the sqrt of any element then running the solution for this problem , but we know that you can’t sort arrays in better than O(nlog(n)), contradiction. Q.E.D.
I am currently going through the software developer hiring process. I have lots of feelings of imposter syndrome and doubt. However, your solutions and thought process are fantastic and allow me to relate to you and help me get over my anxiety. This has been such a help to gain confidence. You rock I would love for you to bring these back wherever you are in life right now. Thanks man
Nice one. If you search for a Google interview question(from Google itself), this version is also shown there(find the two values in an array that add up to a given sum, in linear time). Note that this solution only works if the values in the input are sorted. If they are not then you cannot guarantee that the pointers are going to be pointing to the smallest/largest values.
Great lesson Nick. Another way to optimize this code even better is to loop through the array halfway. This will still give us O(n) for both time and space complexity. Below is my solution (Code in JavaScript btw): function sortedSquareArr(A) { const sortedArr = new Array(); let first = 0; let last = A.length - 1; let i = last; // O(n) - Time complexity // O(n) - Space complexity while(first
Yours is better but I'll say what my idea was. I was thinking binary search the first positive number and save the index. Then I would compare the negative values on the left vs the positive one on the right and same idea as yours but constructing the output from smallest to biggest. Once either one of the pointer reached an end of the input array just fill in the squares of the side that wasn't done yet. Searching: log(n) Construction: n Run time: log(n)+n = O(n) What I said, yours is truly linear but I just wanted to share my idea. Great type of videos, I like them. Keep the good work and explanation method, mentioning the trivial solution is always a good starting point, I like that about the way you solve these
I've literally approached that the same way. Binary search for the index of the first positive value or smallest negative (in case there were no positive ones), and maintain two pointers; right for the found index, and left for right-1. Then just simply comparing whether left or right should be pushed first into a newly created array.
@@GustavoCevalhos in the given approach you only need to perform a binary search once. Just to find a starting index. The comparison itself runs in the linear time. Hence it's O(n + logn) => O(n) overall
You could modify the space complexity by keeping track of the current number as a variable and modifying the input array if the solution required lowest possible space complexity as well.
We just need the merge from mergesort between the positive side of the array and the absolute value of the negative side in reverse, and square each element.
After hearing the question, I was like loop through and square, are you serious? You call that a coding interview, blah blah blah, then 2 minutes in... oh
I'm happy because I managed to figure it out, I once saw somewhat simmilar problem, where you're given two sorted arrays of n size, and you want to find a pair of numbers x and y, one from each array, such that x+y=m. You would put one pointer i on the start of first array and j on the end of second one, and if sum>m then decrease j, if sum
Actually u do one extra calculation which is abs() , couse u can just square both compare them put larger in output array and put smallest in var to compare with the next one from other side
Use two pointers ... One at the end of the negative half ( move towards right) One at the starting of the positive half and merge the two halves ( neglect the negative sign while merging moves to left ) Place the merged ones in output array Later square the merge array or do it while merging Return it back Voila O(n)
This optimized approach is only possible if input array starts with highest negative and end with highest positive number, ie, the input is sorted either in increasing or decreasing order.
2nd method will be faster, it's only taking 1 parse through the array, so N. In first case we were going through the array to square each element, so the time complexity is already N + sorting time
There's no need, he gave the time complexity. The O(n^2) solution can on occasion be faster than O(n), so execution time doesn't tell you much unless you run it on a sufficient number of inputs.
please don’t ever stop making these videos. These help so much with learning and you explain everything with such good detail, I can follow along. Thank you!!!
int numbers[] = {-3,-4,1,2,3,1}; algorithm not working for this input .Works only if the largest element is at the left and not somewhere else. I think you also have to check the last number with any new greater element and then swap them both.
Nick: Simple idea is to square the array then sort it, but it isn't efficient because sorting is n*log(n) at best, we are aiming for linear time complexity Every other python "expert" in the comments: lemme write one liner to show how smart I am using built in sort method
This is a nice Coding Interview Question. I had an interview at Facebook and I failed it because I didn't know O(n) algorithm for a much complex problem.
Nice. This wouldn't improve the complexity, but you could improve performance by simply draining the remaining numbers once you have hit the boundary rather than continuing to check each number.
My solution is finding the first positive value using binary search, splitting negative part from positive part, square both parts and merge them taking into account that the negative part is now "reversely sorted". Divide and conquer style!
My first thought was: Those negative values are the ones making this difficult, so let's get rid of them. And in the end, the worst-case complexity is still N (more precisely logn + 3N)
This algorithm will work if the input array is sorted. We can use modified counting sort algorithm to solve the problem for unsorted input array. It'll take linear time. Maybe I'm wrong.
If there's an O(n) solution for an unsorted array, we can input some positive integers, then apply this algorithm and sqrt each member, in output we'll get an original sorted array and it will work for O(n), but sorting is actually O(n*log n)
@@aertyty3900, count sort and redix sort take linear time (it seems linear) and in required output all numbers are positive. If we take square for each element in array, it'll be done in O(n) and then apply one of these sorting algorithms. It's my thought and I'm not professional programmer.
For loop of the array which iterated through the size of the array, find number closest to zero, and work backwards. Keep track of lowest, highest, and previously sorted number to make it quicker, then, once sorted, a while or for loop to square each element in the array.
Great solution. I would also add that if all the input numbers are positive, this approach can be simplified and done in-place, thus reducing the space complexity to O(1). That’s because when they are all positive, you know squaring them will not change their sorted order. Simply check the first index of your input array, and if its value is positive, square all the numbers in-place. You could also do something similar to save space if all the input numbers are negative, since you know their squares will be in reverse sorted order.
just looking at 6:55, I paused and opened pycharm. I tried once, didn't work. Debugged in two places and it worked!! thanks nick for making these videos.
You could just look for the 1 positive number and place a pointer on the positive side and one on the negative. Now just sq the values on either side see compare which is smaller and add that to the op array
There is a way around the n log n complexity of sorting: the problem explicitly bounds the range of inputs, so you can use a linear-time radio sort. It would be dumb, but you could do it.
If both left and right values are equal, we can increment left and decrement right i and can sort the values in result array ,except when they are equal
Small tweak of merge sort is best. Find the first negative and first positive. Pad output array, start stepping positive/negative side, insert the smallest. This method avoids using the Math.Abs() and will be faster. O(N).
I came up with this before watching your solution: def squared_sorted_array(l): l=[abs(x) for x in l] n=len(l) o=[0]*n for i in range(n): o[i]=min(l) del l[l.index(min(l))] l=[x**2 for x in o] return l But, initially I thought of the same process that you used, but I thought it might be messy so discarded that.
Even though i understand that your algorithm is clearer to read, but I must point out two things: - you are calling Math.abs each iteration, which is not ideal. you'll have to square it anyway, calling abs is 100% unnecessary. - you aren't really taking advantage of having a sorted array properly. After all the negative (or positive) numbers are "consumed" you know for sure that you can just consume all the leftover numbers in order. pros of mine: - takes less if()s - has no dependency - ifs are faster to evaluate cons of mine: - harder to reaad - took me a bit more time, which is obviously not ideal in an interview int[] sortedSquaredArray(int[] array) { int i = 0; int length = array.length; //square negative ones first for (; (i < length) && (array[i] < 0); i++) { //by naming the arrayaccess value, you don't need to access the array 2 times, only once int val = array[i]; array[i] = val * val; } int[] result = new int[length]; int l = 0; int left = i-1; int right = i; //i'm iterating from the inside, not the outside like you do. while (left >= 0 && right < length) { int leftValue = array[left]; //by naming the arrayaccess value, you don't need to access the array 2 times, only once int rightValue = array[right]; int rightSquaredValue = rightValue * rightValue; if (leftValue < rightSquaredValue) { result[l++] = leftValue; left--; } else { result[l++] = rightSquaredValue; right++; } } //when we get to the end of the minuses or the pluses, we might have a bunch of numbers left. //these numbers are all sorted they need 0 checking, they just need to be added to the result array. if(left >= 0){ for (; left >= 0; left--) { result[l++] = array[left]; } return result; } for (; right < length; right++) { int val = array[right]; result[l++] = val * val; } return result; } please notice me :D
Guys, don't you see that stuff like this gets more and more complicated and thus less maintainable? Would you like to hunt for a bug in code like this? (Do you have experience doing this, especially if this isn't your own code originally?) One of the main goals in modern software development isn't the fanciest most optimized version of an algorithm but to have *readable* code. Code which the next maintainer will understand as fast as possible. It's alright to bring an algorithms from N log N down to N if you really have large input. If this is going to save 20 seconds of runtime within the next four years, it is probably not a good idea to even optimize for this. But any optimization beyond this (as in the post above) needs an enormously good reasoning to be justified. »This is even a little faster« is not such a good reasoning ;-)
@@alfeberlin Then again, this is really easy to test, as you just have to compare its output to the sort(squared_arr) version on random arrays. So you could become pretty confident that the algorithm does what its supposed to do.
@@alfeberlin Honestly, if you want readable code you only do sorting by hand if it is absolutely a must for performance. I'd just .stream().map(...).sorted().collect it. if i really wanted to make it performant, i would do sth like this: Stream negatives = .stream().takeWhile(i < 0).map(...) then Stream positives = .stream().reversed().takeWhile(i >= 0).map(...) and use something like negatives.mergeSorted(positives).reversed() and there you go, 3 lines of code, and even newbies can understand it. Google's coding challenges are stupid, it's much harder to model problems properly than to come up with fancy algorithms, and still they ask you to come up with algorithms that you can argue a lot about whether it's good or not.
I think on 6th line of second optimised code there should be Math.abs [right] as well, because as you said array full of negative numbers is a valid input.
Just one minor add on to the comparison, we can replace the call to Math.abs by adding the two numbers together and return right on positive/left on negative.
I would argue in favor of the square everything then sort it method. It is O(n * log(n)), which is not slow as you said. Also our n is less than 10000, so when this is not called in a hot loop millions of times readability wins over shaving of one log(n) of the time complexity.
Niko Ursol If we are really nailing it to that detail, you’ll square each number twice by the way you describe. Better just compare the sum of two numbers with zero to get the optimal solution timewise.
@@MY-bc3qq why would you have to square it twice? It's only one time thing - before you comparing them, and than just merge stuff together as you go. You have to copy each int twice tho, but its cheap. (by copy twice i mean copy in buff to square them, and then emplace (copy again to resulting list) in the end even squaring whole list, grouping them together, reversing one half, merging them would be in O(n)
If you order the list using absolute values, yo have the right order, example: [-6, -4, 1, 2, 3, 4, 5] => [1, 2, 3, 4, 5, 6] => [1, 4, 9, 16, 25, 36] it takes o(n)
Is there ever a case where you would need to have math.abs for the right pointer as well. Such as, if there are more negative numbers than positive and the right pointer hits a negative.
I've tried it, I thought it would be good but then I found a worst case that is O(n^2).. The idea was identical to the one proposed by Nick but every time right value is less that abs(left value) you rotate left by one the sub array from left to right, put the left squared in the right place and keep going. The worst case is for instance: -20, -19, -18, 4, 8, 12 because you go through the whole array and you rotate, and I implemented left rotation by one in O(n), so you got (n^2).. Maybe we can make some assumption on average case?
just to be clear: + before the last step the indices are always equal (because the total number of processed elements is len(array) - 1 + hence it doesn't really matter for the last step which index is advanced; + if all elements are negative, the left pointer always advances because the absolute value of a negative is greater than any negative number (including the first) + if all elements are positive, the right pointer always advances because the left pointer points to the element with the smallest absolute value + the same reasoning applies when the negatives run out before the positives vs the positives run out before the negatives + if the order is negative, positive, negative, positive... or positive, negative, positive, negative -- then the two pointers meet in the middle, and which pointer advances last depends on the parity of the length and which element came first + elements with equal absolute values are treated as elements from the "right" side (they fall in the else clause)
@Nick White it may have been more intuitive (though that's debatable) to condition the loop on equality of the indices and then just take the final element. But this is OK too, because on each loop step each index is changed by one, so the sum of the distances traveled by each index equals the number of iterations.
watched this video, went to solve the question, proceeds to forget about the for loop and does it with while loop, now i have 2 solutions. So the algorithm explanation was well done.
What if you go through the list, if you see a negative int, square it and throw it in a max heap (or max priority queue, or I'm assuming a normal queue would work also since array is sorted so negative ints thrown into the queue will be smallest square at the back biggest square at the front), If you see a positive int then square it? check max heap and keep popping until max element is less than your positive int, then just keep going, throwing elements in whatever new list you initialized at the beginning Should be O(n) if I'm not missing anything
i thought of travelling to 1st positive number and then use 2 pointers on right and left list just like in video, but i guess that travelling is unnecessary now
@@Akash-Watched While there is no comparison-based sorting algorithm with time complexity less than O(nlogn), this is done on an array of integers. You can use counting sort on this. The problem is counting sort is inefficient on arrays with a large range of values.
What Nick is doing has time complexity O(n)/linear and he said it's the best way to do it, what if we square numbers of array and then apply MERGE SORT that has time complexity O(nlog(n)) in all the best to worst case time complexity scenarios (we would need left and right(start and end) values in array for merge sort). Which one would be good way to solve the problem please tell me, I'm confused? P.S. Nick's solution and mindset towards problem was absolutely amazing! :)
Just aced my interview with Amazon today (got an email within minutes after the interview that I was selected to move on to the final panel interview). Thanks so much for these videos, def helped get me in the right mindset and made the coding portion of the interview a breeze.
Did you get the job? If so congratulations!
bro party !!!!
Hi
youtube algorithm picks up another algo video
David Hahn sounds like like recursion right there lmao
@@tannerbarcelos6880 its a pointer
@@tannerbarcelos6880
yeah, it's like like like recursion there
yup
CanAlgolism
Nick, you are the best!
I’m a software developer over 7 years and I’d watched so many empty talks and crappy solutions from other people for data structure problems. You teach the technique to solve a problem rather than useless looping, mapping ext...
Thanks man! Keep up the great work!
And all of my friends accused my teachers, saying that we would never use Math as an adult. Because my computer will be my calculator...
This is essentially the final stages of a merge sort, with one list starting on the left, and the other starting on the right.
that's interesting
Yup
Was the first idea I had as well.
Allocate 2x input.
Loop over the input and store the negative numbers in the first list and positive in the 2nd. Memory consumption is larger for this case. But end step is just merge the two lists. Original list could be used for this to maintain 3N memory complexity and 2N run time.
I could binge 4 seasons of this
same
Haha! Facts
I love to watch your tutorials. It gives me a sense of relief each time I watch it.
Not to mention that they are extremely helpful.
I’m preparing to work at faang and every morning I come to this channel as a pilgrim
The whole time I thought the input wouldn't be sorted.
Yeah. I was so confused at the beginning because I couldnt think of a solution that would be better than O(N log n) because in every case you still have to sort the numbers. The problems got so much easier with the realization that the numbers are sorted...
Yeah
If the input isn’t sorted there is no solution better than O(nlog(n)) , it is easy to prove:
Assume that there is a solution in o(nlog(n))(way better than O(nlog(n))) so I can sort any array in this time complexity by first taking the sqrt of any element then running the solution for this problem , but we know that you can’t sort arrays in better than O(nlog(n)), contradiction. Q.E.D.
@@iradnuriel9087 Given that the range of input is between -10000 and 10000, you can use counting sort for linear time sort.
@@arnabmitra08 that is exactly what came to my mind
It's much simpler than this pointer stuff anyway.
I am currently going through the software developer hiring process. I have lots of feelings of imposter syndrome and doubt. However, your solutions and thought process are fantastic and allow me to relate to you and help me get over my anxiety. This has been such a help to gain confidence. You rock I would love for you to bring these back wherever you are in life right now. Thanks man
Nice one. If you search for a Google interview question(from Google itself), this version is also shown there(find the two values in an array that add up to a given sum, in linear time).
Note that this solution only works if the values in the input are sorted. If they are not then you cannot guarantee that the pointers are going to be pointing to the smallest/largest values.
hey bro...u r doing gud .... keep updating by these type of questions.......
Great lesson Nick. Another way to optimize this code even better is to loop through the array halfway. This will still give us O(n) for both time and space complexity. Below is my solution (Code in JavaScript btw):
function sortedSquareArr(A) {
const sortedArr = new Array();
let first = 0;
let last = A.length - 1;
let i = last;
// O(n) - Time complexity
// O(n) - Space complexity
while(first
Yours is better but I'll say what my idea was.
I was thinking binary search the first positive number and save the index. Then I would compare the negative values on the left vs the positive one on the right and same idea as yours but constructing the output from smallest to biggest. Once either one of the pointer reached an end of the input array just fill in the squares of the side that wasn't done yet.
Searching: log(n)
Construction: n
Run time: log(n)+n = O(n)
What I said, yours is truly linear but I just wanted to share my idea.
Great type of videos, I like them. Keep the good work and explanation method, mentioning the trivial solution is always a good starting point, I like that about the way you solve these
I've literally approached that the same way. Binary search for the index of the first positive value or smallest negative (in case there were no positive ones), and maintain two pointers; right for the found index, and left for right-1. Then just simply comparing whether left or right should be pushed first into a newly created array.
Basically we are doing one iteration of merge sort at the end:
1. split list into two sorted halves
2. merge them
-- I like this solution better.
Hi!
You would be searching N times in the input, which would lead to N*Lg(N) overall time complexity.
@@GustavoCevalhos in the given approach you only need to perform a binary search once. Just to find a starting index. The comparison itself runs in the linear time. Hence it's O(n + logn) => O(n) overall
@@brunokawka True! I misread the explanation, you are correct :)
You could modify the space complexity by keeping track of the current number as a variable and modifying the input array if the solution required lowest possible space complexity as well.
We just need the merge from mergesort between the positive side of the array and the absolute value of the negative side in reverse, and square each element.
After hearing the question, I was like loop through and square, are you serious? You call that a coding interview, blah blah blah, then 2 minutes in... oh
I was I can do this in less than 5 lines in python but this vid has almost 13 mins
@@dragon_warrior_ yes hes doing on java, phyton is to ez
Yep wtf
Atleast we can take the absolute value of the elements and sort it and then get the output...
Edit: I did not watch this video fully...
@@baka_geddy Sure that works, however the goal of the video is to bring it to linear runtime, i.e., no direct sorting.
I'm happy because I managed to figure it out, I once saw somewhat simmilar problem, where you're given two sorted arrays of n size, and you want to find a pair of numbers x and y, one from each array, such that x+y=m. You would put one pointer i on the start of first array and j on the end of second one, and if sum>m then decrease j, if sum
6:07 Hmm. 36 < 25
Actually u do one extra calculation which is abs() , couse u can just square both compare them put larger in output array and put smallest in var to compare with the next one from other side
Use two pointers ...
One at the end of the negative half ( move towards right)
One at the starting of the positive half and merge the two halves ( neglect the negative sign while merging moves to left )
Place the merged ones in output array
Later square the merge array or do it while merging
Return it back
Voila O(n)
This optimized approach is only possible if input array starts with highest negative and end with highest positive number, ie, the input is sorted either in increasing or decreasing order.
Can you tell which screen recorder you use?
Do a time execution from both code options to show the conclusion of being faster.
2nd method will be faster, it's only taking 1 parse through the array, so N.
In first case we were going through the array to square each element, so the time complexity is already N + sorting time
There's no need, he gave the time complexity.
The O(n^2) solution can on occasion be faster than O(n), so execution time doesn't tell you much unless you run it on a sufficient number of inputs.
Man! You are gold. Thankyou so much for making these videos
please don’t ever stop making these videos. These help so much with learning and you explain everything with such good detail, I can follow along. Thank you!!!
"...yyyoooOOOOOOOOOOOOO (slap) what is up uhh (forgot the word for "coders") coding people that, watch coding videos"
lmao I always love your intros
int numbers[] = {-3,-4,1,2,3,1}; algorithm not working for this input .Works only if the largest element is at the left and not somewhere else. I think you also have to check the last number with any new greater element and then swap them both.
Nick: Simple idea is to square the array then sort it, but it isn't efficient because sorting is n*log(n) at best, we are aiming for linear time complexity
Every other python "expert" in the comments: lemme write one liner to show how smart I am using built in sort method
This is a nice Coding Interview Question. I had an interview at Facebook and I failed it because I didn't know O(n) algorithm for a much complex problem.
Nice. This wouldn't improve the complexity, but you could improve performance by simply draining the remaining numbers once you have hit the boundary rather than continuing to check each number.
My solution is finding the first positive value using binary search, splitting negative part from positive part, square both parts and merge them taking into account that the negative part is now "reversely sorted".
Divide and conquer style!
My first thought was: Those negative values are the ones making this difficult, so let's get rid of them. And in the end, the worst-case complexity is still N (more precisely logn + 3N)
4:07 Arrays.sort(square_the_array);
Isn't this question too easy for something like facebook? idk
dude , are you dumb somehow?
The time complexity is nlogn....too high.Facebook wants less not the average soln
This algorithm will work if the input array is sorted. We can use modified counting sort algorithm to solve the problem for unsorted input array. It'll take linear time. Maybe I'm wrong.
If there's an O(n) solution for an unsorted array, we can input some positive integers, then apply this algorithm and sqrt each member, in output we'll get an original sorted array and it will work for O(n), but sorting is actually O(n*log n)
@@aertyty3900, count sort and redix sort take linear time (it seems linear) and in required output all numbers are positive. If we take square for each element in array, it'll be done in O(n) and then apply one of these sorting algorithms. It's my thought and I'm not professional programmer.
@@muhammadrizwan4999 I meant that for the given bounds both algorithms should work but if number length is unlimited it will run out of memory
@@aertyty3900 yes and we are sacrificing memory for speed. My bad I wasn't able to understand your previous reply.
For loop of the array which iterated through the size of the array, find number closest to zero, and work backwards. Keep track of lowest, highest, and previously sorted number to make it quicker, then, once sorted, a while or for loop to square each element in the array.
Great solution. I would also add that if all the input numbers are positive, this approach can be simplified and done in-place, thus reducing the space complexity to O(1). That’s because when they are all positive, you know squaring them will not change their sorted order. Simply check the first index of your input array, and if its value is positive, square all the numbers in-place. You could also do something similar to save space if all the input numbers are negative, since you know their squares will be in reverse sorted order.
just looking at 6:55, I paused and opened pycharm. I tried once, didn't work. Debugged in two places and it worked!! thanks nick for making these videos.
Your teaching style really clicks for me! Thank you!
last approach does not work for the input : { 4, -2, 0, 9, 3} -- output will be : 0,4,81,9,16
The problem says that the input array is already sorted. The input you gave isn't.
You could just look for the 1 positive number and place a pointer on the positive side and one on the negative. Now just sq the values on either side see compare which is smaller and add that to the op array
If you need a handy one-liner in java:
int[] squaredArray(int[] array) {
return Arrays.stream(array).map(i -> i * i).sorted().toArray();
}
Ikr
There is a way around the n log n complexity of sorting: the problem explicitly bounds the range of inputs, so you can use a linear-time radio sort. It would be dumb, but you could do it.
Second approach will not for input (1,-4,7,-2,5,3,-6). I mean, if elements(abs values) are not arranged in decreasing and increasing order.
I find this similar to merge sort while we try to merge 2 list into 1. nice video man.....
If both left and right values are equal, we can increment left and decrement right i and can sort the values in result array ,except when they are equal
Nick White - you really have an excellent thought process while dealing with problems.
Small tweak of merge sort is best. Find the first negative and first positive. Pad output array, start stepping positive/negative side, insert the smallest. This method avoids using the Math.Abs() and will be faster. O(N).
I came up with this before watching your solution:
def squared_sorted_array(l):
l=[abs(x) for x in l]
n=len(l)
o=[0]*n
for i in range(n):
o[i]=min(l)
del l[l.index(min(l))]
l=[x**2 for x in o]
return l
But, initially I thought of the same process that you used, but I thought it might be messy so discarded that.
Such a great start 0:00👌
Even though i understand that your algorithm is clearer to read, but I must point out two things:
- you are calling Math.abs each iteration, which is not ideal. you'll have to square it anyway, calling abs is 100% unnecessary.
- you aren't really taking advantage of having a sorted array properly. After all the negative (or positive) numbers are "consumed" you know for sure that you can just consume all the leftover numbers in order.
pros of mine:
- takes less if()s
- has no dependency
- ifs are faster to evaluate
cons of mine:
- harder to reaad
- took me a bit more time, which is obviously not ideal in an interview
int[] sortedSquaredArray(int[] array) {
int i = 0;
int length = array.length;
//square negative ones
first
for (; (i < length) && (array[i] < 0); i++) {
//by naming the arrayaccess value, you don't need to access the array 2 times, only once
int val = array[i];
array[i] = val * val;
}
int[] result = new int[length];
int l = 0;
int left = i-1;
int right = i;
//i'm iterating from the inside, not the outside like you do.
while (left >= 0 && right < length) {
int leftValue = array[left];
//by naming the arrayaccess value, you don't need to access the array 2 times, only once
int rightValue = array[right];
int rightSquaredValue = rightValue * rightValue;
if (leftValue < rightSquaredValue) {
result[l++] = leftValue;
left--;
} else {
result[l++] = rightSquaredValue;
right++;
}
}
//when we get to the end of the minuses or the pluses, we might have a bunch of numbers left.
//these numbers are all sorted they need 0 checking, they just need to be added to the result array.
if(left >= 0){
for (; left >= 0; left--) {
result[l++] = array[left];
}
return result;
}
for (; right < length; right++) {
int val = array[right];
result[l++] = val * val;
}
return result;
}
please notice me :D
Guys, don't you see that stuff like this gets more and more complicated and thus less maintainable? Would you like to hunt for a bug in code like this? (Do you have experience doing this, especially if this isn't your own code originally?)
One of the main goals in modern software development isn't the fanciest most optimized version of an algorithm but to have *readable* code. Code which the next maintainer will understand as fast as possible. It's alright to bring an algorithms from N log N down to N if you really have large input. If this is going to save 20 seconds of runtime within the next four years, it is probably not a good idea to even optimize for this. But any optimization beyond this (as in the post above) needs an enormously good reasoning to be justified. »This is even a little faster« is not such a good reasoning ;-)
@@alfeberlin Then again, this is really easy to test, as you just have to compare its output to the sort(squared_arr) version on random arrays. So you could become pretty confident that the algorithm does what its supposed to do.
@@lucast2212 Testability doesn't make the code clean ;-) You will find out the difference as soon as one of the tests fails ;-)
I'll take "what is compiler optimisation" for $0
@@alfeberlin Honestly, if you want readable code you only do sorting by hand if it is absolutely a must for performance. I'd just .stream().map(...).sorted().collect it.
if i really wanted to make it performant, i would do sth like this:
Stream negatives = .stream().takeWhile(i < 0).map(...)
then
Stream positives = .stream().reversed().takeWhile(i >= 0).map(...)
and use something like negatives.mergeSorted(positives).reversed()
and there you go, 3 lines of code, and even newbies can understand it. Google's coding challenges are stupid, it's much harder to model problems properly than to come up with fancy algorithms, and still they ask you to come up with algorithms that you can argue a lot about whether it's good or not.
What is the time complexity for your solution ?, Please tell me it's O (n) ? Or different
That is O(n). One loop through the output array which is n long.
@@TheBreezerFly ok thank you for your reply
you was 18k sub like a day ago ,I blinked and now you're 21.4K looks like somebody discovered the youtube algorithm secrete!!!!
gj keep it up:)
Time complexity always messing with my happiness.
LOL
Me too 😂
I literally failed 4 interviews due to time complexity
I think on 6th line of second optimised code there should be Math.abs [right] as well, because as you said array full of negative numbers is a valid input.
Your videos help me have the proper problem solving mindset. Thank you!
I know coding very little, but enjoy problem solving videos. thanks Nick.
Would max heap work here better?
ofcourse
Dude keep making these kind of videos, it helped so many people. God bless
Thanks for the amazing content!
Good work, using right and left pointer is the best way to go about it. May be only way!!
The solution seems so easy once you explain it in your style. Thanks a lot for coming of with such great contents.
This is the best explanation of this problem over the internet !!!
in 2nd approach if we use while loop instead for loop will improve time complexity @Nick
Thank you, still learning how to code but I was able to follow along with your explanation 👍
Just one minor add on to the comparison, we can replace the call to Math.abs by adding the two numbers together and return right on positive/left on negative.
That's interesting. Is using Math.abs () not recommended when you want to consider the most optimal solution?
Oh, I thought to start pointers from middle where sign changes. Great solution.
I would argue in favor of the square everything then sort it method. It is O(n * log(n)), which is not slow as you said. Also our n is less than 10000, so when this is not called in a hot loop millions of times readability wins over shaving of one log(n) of the time complexity.
Likely better to square them all up front. That avoids the absolute value and array operations are usually faster. Good video. Thanks.
Is it really faster though? You would need to loop through the array twice.
@@StraightCrossing you can do it as you compare them
Niko Ursol If we are really nailing it to that detail, you’ll square each number twice by the way you describe. Better just compare the sum of two numbers with zero to get the optimal solution timewise.
@@MY-bc3qq why would you have to square it twice? It's only one time thing - before you comparing them, and than just merge stuff together as you go.
You have to copy each int twice tho, but its cheap. (by copy twice i mean copy in buff to square them, and then emplace (copy again to resulting list)
in the end even squaring whole list, grouping them together, reversing one half, merging them would be in O(n)
Yeah. You are making potentially millions of calls to ABS(). That's expensive
If you order the list using absolute values, yo have the right order, example:
[-6, -4, 1, 2, 3, 4, 5] => [1, 2, 3, 4, 5, 6] => [1, 4, 9, 16, 25, 36]
it takes o(n)
Is there ever a case where you would need to have math.abs for the right pointer as well. Such as, if there are more negative numbers than positive and the right pointer hits a negative.
Well explained please dnt stop these tutorials ☺️
Loop from the number closest to zero, compare it to the neighbours with their absolute values, square the larger one and put it in the result array.
Your last method are basically kind of merge two sorted arrays were you handle the minus value as their abs value
You could save a scrap or two of memory through destructuring by doing "int i = right" in the loop.
First i think about using hashtable so the time O(n) but it is not optimal way for memory. Your solution is so great.
That was very helpful thanks nick🤗
this solution is far better than your previous solution of the same problem. I guess you did that for leetcode. thanks for your work.
what about looping through the input array and adding the square of each number to a treeset?
Elegant solution, keep it doing well
can it be done inplace in O(N) ?
I've tried it, I thought it would be good but then I found a worst case that is O(n^2).. The idea was identical to the one proposed by Nick but every time right value is less that abs(left value) you rotate left by one the sub array from left to right, put the left squared in the right place and keep going. The worst case is for instance: -20, -19, -18, 4, 8, 12 because you go through the whole array and you rotate, and I implemented left rotation by one in O(n), so you got (n^2).. Maybe we can make some assumption on average case?
just to be clear:
+ before the last step the indices are always equal (because the total number of processed elements is len(array) - 1
+ hence it doesn't really matter for the last step which index is advanced;
+ if all elements are negative, the left pointer always advances because the absolute value of a negative is greater than any negative number (including the first)
+ if all elements are positive, the right pointer always advances because the left pointer points to the element with the smallest absolute value
+ the same reasoning applies when the negatives run out before the positives vs the positives run out before the negatives
+ if the order is negative, positive, negative, positive... or positive, negative, positive, negative -- then the two pointers meet in the middle, and which pointer advances last depends on the parity of the length and which element came first
+ elements with equal absolute values are treated as elements from the "right" side (they fall in the else clause)
@Nick White it may have been more intuitive (though that's debatable) to condition the loop on equality of the indices and then just take the final element. But this is OK too, because on each loop step each index is changed by one, so the sum of the distances traveled by each index equals the number of iterations.
watched this video, went to solve the question, proceeds to forget about the for loop and does it with while loop, now i have 2 solutions. So the algorithm explanation was well done.
Would love to watch more of such solutions to interview questions
Man this guy is good.
Me watching every video of Nick :"Damn Maaannn"
Very well explained with a high quality microphone lol. Thanks!
What if you go through the list, if you see a negative int, square it and throw it in a max heap (or max priority queue, or I'm assuming a normal queue would work also since array is sorted so negative ints thrown into the queue will be smallest square at the back biggest square at the front), If you see a positive int then square it? check max heap and keep popping until max element is less than your positive int, then just keep going, throwing elements in whatever new list you initialized at the beginning
Should be O(n) if I'm not missing anything
I find your videos so easy to understand
which coding platform was that?
i thought of travelling to 1st positive number and then use 2 pointers on right and left list just like in video, but i guess that travelling is unnecessary now
You have explained quite well.
Is there any way to improve the space complexity? Is it possible to do an in-place merge?
We can also make 2 vectors with squared value of - ve and +ve no.s and mergesort them
can i use sort function of python to sort it after squaring every element.. i think it's time will be O(n)..??
Great explanation !!
A sorting algo with liner time complexity..
There are no comparison-based sorting algorithm with a complexity of less than O(n logn) and there can never be as well.
There is no sorting either, it is essentially just merging two already sorted arrays.
@@Akash-Watched While there is no comparison-based sorting algorithm with time complexity less than O(nlogn), this is done on an array of integers. You can use counting sort on this. The problem is counting sort is inefficient on arrays with a large range of values.
What Nick is doing has time complexity O(n)/linear and he said it's the best way to do it, what if we square numbers of array and then apply MERGE SORT that has time complexity O(nlog(n)) in all the best to worst case time complexity scenarios (we would need left and right(start and end) values in array for merge sort).
Which one would be good way to solve the problem please tell me, I'm confused?
P.S. Nick's solution and mindset towards problem was absolutely amazing! :)
Thank you Nick !
What will be the time complexity of the solution algorithm given by Nick??
two pointer algorithms are so satisfying
This is so easy in javascript with the sort command.
You just sort the array and then use the forEach command to get the output.