A lot of people are using LeetCode. Today, I also registered for a game, and found that many of them are algorithm problems. Well, ten years after graduation, all the poor math knowledge I learned has been returned to the school. So without further ado, let’s take a look at the first problem I tried in today’s video, which is kind of interesting, not just the algorithm, but the data and the fit.
ChengTi
Click on the question bank, look at the first question, let’s look at this question:
Given an array of integers and a target value, find two numbers that neutralize the target value in the array. You can assume that there is only one answer for each input and that the same elements cannot be reused. Example: Given nums = [2, 7, 11, 15], target = 9
Find the index of the two numbers in the array that you want to add or equal to the target value.
The problem is not difficult. Leetcode gives two algorithms:
- Brute force, iterating, time O(n^2), space O(1).
- One hash table, order n in time and space.
Violence law
I use Go language (Golang) to implement the violence method, let’s look at the code.
func TwoSum1(nums []int, target int) []int {
n:=len(nums)
for i,v:=range nums {
for j:=i+1; j<n; j++ {if v+nums[j] == target {
return []int{i,j}
}
}
}
return nil
}
Copy the code
Two layer cycle nesting, very yellow very violent. The algorithm is if you’re lucky, you loop it twice, and if you’re unlucky, you’re looking for elements in the last two digits, then it’s really O(n^2).
Hash method
The Go language has the map type, which is the default Hash implementation, and based on that we use Golang to implement the Hash.
func TwoSum2(nums []int, target int) []int {
m:=make(map[int]int.len(nums))
for i,v:=range nums {
sub:=target-v
ifj,ok:=m[sub]; ok{return []int{j,i}
}else{
m[v]=i
}
}
return nil
}
Copy the code
It’s a nice algorithm, O(n) in both time and space, and if you’re lucky enough to have a lot of duplicate elements in the array, you’ll get a little bit less space.
test
You write the algorithm, you test it, you make sure it’s correct, you don’t screw up.
package main
import (
"flysnow.org/hello/lib"
"fmt"
)
func main(a){
r1:=lib.TwoSum1([]int{2.7.11.15},9)
fmt.Println(r1)
r2:=lib.TwoSum2([]int{2.7.11.15},9)
fmt.Println(r2)
}
Copy the code
Run output:
[0, 1] is [0, 1]Copy the code
As expected, our algorithm is fine.
Performance expectations
These two algorithms, Leetcode also give space and time complexity, and from our own code implementation analysis, the second hash method is much better than the brute force method, is that really the case? Let’s test this using the Go language Benchmark.
About the Benchmark (Benchmark) can refer to the language field notes (22) | Go benchmarks, here no longer expatiatory.
func BenchmarkTwoSum1(b *testing.B) {
b.ResetTimer()
for i:=0; i<b.N; i++{ TwoSum1([]int{2.7.11.15},9)}}func BenchmarkTwoSum2(b *testing.B) {
b.ResetTimer()
for i:=0; i<b.N; i++{ TwoSum2([]int{2.7.11.15},9)}}Copy the code
Run ➜ lib go test-bench =. -benchmem-run =none to see the results of the Golang Benchmark test.
pkg: flysnow.org/hello/lib benchmarkTwoSUM1-8 50000000 26.9 ns/op 16 B/ OP 1 Allocs/OP benchmarkTWOSUM2-8 20000000 73.9 ns/op 16 B/op 1 allocs/opCopy the code
So the test case THAT I used, the test case that they gave us, we found that in this test case, instead of looking at the good brute force method, the performance was 2.5 times better than the hash method, which is a little bit different from what we expected.
Array position adjustment
If we look at the array of tests, the answers are in the first two parts of the array, which does have an advantage for the brute force method, we adjust the two answers, 2 and 7, to the end of the array, so that the test array is {11, 15, 2, 7}, and look at the test result.
Benchmarktwosum1-8 50000000 29.1 NS/OP 16 B/ OP 1 ALlocs/OP benchmarkTWOSUM2-8 10000000 140 NS/OP 16 B/ OP 1 Allocs/OPCopy the code
Well, this one, the brute force is as strong as ever, but the performance of the hash drops by a factor of two, killing the hash.
Expanding the number of arrays
We find that brute force is superior and performs best when the number of arrays is small. Let’s adjust the number of arrays and test again.
const N = 10
func BenchmarkTwoSum1(b *testing.B) {
nums:=[]int{}
for i:=0; i<N; i++{ nums=append(nums,rand.Int())
}
nums=append( nums,7.2)
b.ResetTimer()
for i:=0; i<b.N; i++{ TwoSum1(nums,9)}}func BenchmarkTwoSum2(b *testing.B) {
nums:=[]int{}
for i:=0; i<N; i++{ nums=append(nums,rand.Int())
}
nums=append( nums,7.2)
b.ResetTimer()
for i:=0; i<b.N; i++{ TwoSum2(nums,9)}}Copy the code
If you look closely at the code above, I randomly generate the array elements automatically, but the last two digits of the array remain 7,2 to ensure the answer. Let’s test for arrays of 10.
Benchmarktwosum1-8 20000000 73.3 NS/OP 16 B/ OP 1 ALLOCs/OP benchmarktwoSUM2-8 2000000 660 NS/OP 318 B/ OP 2 Allocs/OPCopy the code
Ten elements, brute force is 10 times faster than hashing.
Continue resizing the array to 50, and just change the constant N to test for 50 elements.
BenchmarkTwoSum1-8 2000000 984 ns/op 16 B/op 1 allocs/op
BenchmarkTwoSum2-8 500000 3200 ns/op 1451 B/op 6 allocs/op
Copy the code
As the size of the array increases, the advantage of hashing becomes apparent, with only a 4-fold difference at 50 array elements.
You start by increasing the size of the array. On my computer, when the array size is 300, the two are even and the performance is the same.
When the array size is 1000, hash performance is already 4 times better than brute force, reversed.
When the array size is 10000, the performance of hash method is 20 times that of brute force method. The test data is as follows:
BenchmarkTwoSum1-8 100 21685955 ns/op 16 B/op 1 allocs/op
BenchmarkTwoSum2-8 2000 641821 ns/op 322237 B/op 12 allocs/op
Copy the code
Based on the benchmark data, the larger the array, the longer each operation takes, but the brute force method increases too much, resulting in poor performance.
It can also be seen from the data that the hash method is the way of space for time, and the memory occupation and allocation are relatively large.
summary
From this testing and performance analysis, there is no optimal algorithm, only the most suitable one.
If you have fewer elements in your array, then brute force is more suitable for you. If the array has a lot of elements, then hashing is a good choice.
Therefore, according to the actual situation of our own system, to choose the appropriate algorithm, such as dynamic judgment array size, using different algorithms, to achieve the maximum performance.
This article is an original article, reprinted with notes of origin, “there are always bad people grab the article when also remove my original description” welcome to scan the code to follow the public number flysnow_org or www.flysnow.org/, the first time to see the follow-up wonderful article. “Anti-rotten person remarks **…… &*¥” feel good, feel free to share it in moments, thank you for your support.