如何在 Java 中构建 KDTree
How do you build a KDTree in Java
我看过几个实现,它们有点令人困惑,我希望能从点列表中对构建 KDTree 所需的内容进行某种细分。我将最终使用此 KDTree 执行 2D K 最近邻搜索。
这是我目前所拥有的,并不多。
点Class
class Point{
public PVector p;
public Point(float x, float y){
p = new PVector(x,y);
}
public Point(PVector _p0 ){
p = _p0;
}
float getX(){ return p.x; }
float getY(){ return p.y; }
float x(){ return p.x; }
float y(){ return p.y; }
public float distance( Point o ){
return PVector.dist( p, o.p );
}
节点class
class Node{
Node left;
Node right;
Point p;
Node(Point _p, Node l, Node r){
p = _p;
left = l;
right = r;
}
}
KDTree class
class KDTree{
Node root;
KDTree()
{
root = null;
}
}
如您所见,Node 和 KDTree class 并未真正实现,这就是我遇到的问题。
我想通过给它提供类似这样的东西来构建这个 KDTree:ArrayList< Point > points
如有任何帮助,我们将不胜感激!
关于您的代码结构和您需要实现的方法,我建议它应该类似于以下内容:
enum Axis {
X, Y;
public Axis next() {
return values()[(ordinal() + 1) % values().length];
}
}
interface Node {
Point closestTo(Point point);
}
class Leaf implements Node {
private final Point point;
public Leaf(Point point) {
this.point = point;
}
public Point closestTo(Point point) {
return this.point;
}
}
class Branch implements Node {
private final Axis axis;
private final Point median;
private final Node smaller;
private final Node larger;
public Branch(Axis axis, Point median, Node smaller, Node larger) {
this.axis = axis;
this.median = median;
this.smaller = smaller;
this.larger = larger;
}
public Point closestTo(Point point) {
...
}
}
class KD_Tree {
private final Optional<Node> root;
public KD_Tree(List<Point> points) {
this.root = makeNode(points);
}
private Optional<Node> makeNode(Axis axis, List<Point> points) {
switch (points.size()) {
case 0:
return Optional.empty();
case 1:
return Optional.of(new Leaf(points.get(0));
default:
Point median = medianOf(points);
Node smaller = makeNode(axis.next(), lessThan(median, points)).get();
Node larger = makeNode(axis.next(), greaterThan(median, points)).get();
return Optional.of(new Branch(axis, median, smaller, larger));
}
}
}
那里还有很多逻辑要写,但希望这能让你入门。
我看过几个实现,它们有点令人困惑,我希望能从点列表中对构建 KDTree 所需的内容进行某种细分。我将最终使用此 KDTree 执行 2D K 最近邻搜索。
这是我目前所拥有的,并不多。
点Class
class Point{
public PVector p;
public Point(float x, float y){
p = new PVector(x,y);
}
public Point(PVector _p0 ){
p = _p0;
}
float getX(){ return p.x; }
float getY(){ return p.y; }
float x(){ return p.x; }
float y(){ return p.y; }
public float distance( Point o ){
return PVector.dist( p, o.p );
}
节点class
class Node{
Node left;
Node right;
Point p;
Node(Point _p, Node l, Node r){
p = _p;
left = l;
right = r;
}
}
KDTree class
class KDTree{
Node root;
KDTree()
{
root = null;
}
}
如您所见,Node 和 KDTree class 并未真正实现,这就是我遇到的问题。
我想通过给它提供类似这样的东西来构建这个 KDTree:ArrayList< Point > points
如有任何帮助,我们将不胜感激!
关于您的代码结构和您需要实现的方法,我建议它应该类似于以下内容:
enum Axis {
X, Y;
public Axis next() {
return values()[(ordinal() + 1) % values().length];
}
}
interface Node {
Point closestTo(Point point);
}
class Leaf implements Node {
private final Point point;
public Leaf(Point point) {
this.point = point;
}
public Point closestTo(Point point) {
return this.point;
}
}
class Branch implements Node {
private final Axis axis;
private final Point median;
private final Node smaller;
private final Node larger;
public Branch(Axis axis, Point median, Node smaller, Node larger) {
this.axis = axis;
this.median = median;
this.smaller = smaller;
this.larger = larger;
}
public Point closestTo(Point point) {
...
}
}
class KD_Tree {
private final Optional<Node> root;
public KD_Tree(List<Point> points) {
this.root = makeNode(points);
}
private Optional<Node> makeNode(Axis axis, List<Point> points) {
switch (points.size()) {
case 0:
return Optional.empty();
case 1:
return Optional.of(new Leaf(points.get(0));
default:
Point median = medianOf(points);
Node smaller = makeNode(axis.next(), lessThan(median, points)).get();
Node larger = makeNode(axis.next(), greaterThan(median, points)).get();
return Optional.of(new Branch(axis, median, smaller, larger));
}
}
}
那里还有很多逻辑要写,但希望这能让你入门。