The main concept of 2D ICP is to find the closest point for each source point. In addition, the key is that after determining the closest points' coorespondences, how do we calculate the rotation. Below is the big picture of computing the rotation part.
We have two points set: closest points set and source points set.
After centering at each mean, every point lives in its own coordinate.
Suppose the rotation we want to compute is r.
Let r = r_closest - r_src.
r_closest is the shifted closest points' coordinate relative to the global coordinate.
r_src is the shifted source points' coordinate relative to the global coordinate.
The formula below is an approximation to r.
r = atan2(y'/x') - atan2(y/x), (x, y) is the source coordinate and (x', y') is the closest points' coordinate.
app(r) = (atan2(y'/x') - atan2(y/x))/(1+yy'/xx')
app(r) is proportional to r and up to scale by 0.5 to 1.0.
Another big problem is the local minima. The way I deal with local minima is by generating multiple rotation hypothesis at local minima and doing ICP again to find the minimum error.
Here is my demonstration video: 2D Iterative Closest Point.
-
Get the closest points' coorespondences via Approximate Nearest Neighbors (ANN).
-
Compute delta translation and rotation.
-
Calculate the error function based on the transformed points set and their cooresponding closest points set.
-
Suppose we reach a local minima.
-
Generate rotation hypothesis and repeat steps 1 to 3.
To show the peroformace and accuracy, I randomly generat a pair of translation and rotation to move current points set to a target position.
In the video, red points set is the source points set and green points set is the target points set.
During the process, magenta points set is the local minima.
Yellow points set is the rotation hypothesis. (This part can be parallelized.)
Blue points set is our find result.
Meanwhile, in the terminal, you can see the tranlation and rotation result when reaching local minima.
At the end, you can see the final translation and rotation and the original randomly generated translation and rotation.
- Please make sure you have installed OpenCV and ANN library.
mkdir build
cd build
cmake ..
make
-
./ICP (The default img is ../img/curve.png.)
-
./ICP ../img/corner.png (Change img.)
-
./ICP ../img/corner.png 1000 (Uniformly add 1000 noise points to the target points set.)
-
./ICP ../img/corner.png 10 2000 (Uniformly add 10 noise points and remove 2000 points from the target points set.)
-
How do we extend 2D ICP into 3D ICP? Is there a way to compute 3D rotation? RANSAC+SVD?
-
How do we speed up the performance by doing down-sampling? We can try uniform sampling first to realize down-sampling. If we try to keep the shape information when doing down-sampling, we need to have more samples agound the shape features such as corners.
-
Is there another way to solve local minima problem?
-
Is is reasonable to set the error difference as 1e-6?
-
Could we use fewer iterations as a termination condition? When the error difference is small, we can deduct the termintation iteration. In other words, an adaptive iteration bound can be used to improve this issue.