rstar - an advanced space indexing library

Showcase your libraries, tools and other projects that help your fellow love users.
User avatar
Rick Astley
Prole
Posts: 16
Joined: Sat Mar 13, 2021 11:51 am

rstar - an advanced space indexing library

Post by Rick Astley »

rstar
This is the official rstar library topic, welcome!

Introduction
rstar is a Lua implementation of the R*Tree data structure. Briefly the R*Tree is a improved version of the classic R-Tree: a structure born to store rectangles of any dimensions, in a way that makes queries like "nearest-neighbor searching" and "all rectangles in a area" quick and efficient.
To achieve this, rectangles are distributed in groups: each of these groups contains at least m members, and never more than M members. A group is called leaf node, which also holds a rectangle sized in a way that contains all its entries(members). Leaf nodes can be treated as entries, thus they can be distributed in groups with the same rules mentioned before, giving birth to internal nodes (or branches).
The node at highest level, which contains all the structure, is called the root, and is the starting point for any kind of operation on the tree.
The R*Tree variant improves the original in the way it builds the structure: creates nodes trying to minimize overlap between them and using as much space as it can.

Overview and Features
R*Tree is good for game developement because its nature makes it suitable in many situations. It must be said that it's not always the best choice: to grow a optimal structure, insertion and deletion take a lot of effort, meaning that for a dynamic set of objects this is not a convenient technique.
On the other hand for static objects with occasional updating, this is going to be lövely!
These are some examples where this library can be really helpful:

Storing solid objects
A advantage of this structure is that it does not need to know how big is the medium object, like a spatial hash does; thus making it great to store objects with a rich range of sizes.

Representing "infinite" worlds
Another advantage is that, unlike Quadtrees, R*Trees are not constrainted to a rigid boundary as they stretch as needed. Worlds which expand over time, like Minecraft's ones, can be handled easily.

Filtering objects
If you want to optimize the drawing of game objects, you can retrieve ones which intersect with the screen rectangle to discard invisible ones. Also finding all objects at a point is a very quick operation: sweet to know what the user is clicking!

Detecting collisions
To detect if the player is bumping into walls, does not make sense to test on a wall a mile away from them; this is a task where the ability to search quickly for walls around the player is a big deal.

Raycasting
This implementation comes with a method to retrieve all rectangles within a circular range, which could speed up raycasting by excluding objects not in the radius of sight.

AI
Nearest-neighbor search is included, and works both with rectangles inserted in the tree and arbitrary boxes.

That's all for now, thanks for checking this out!

P.S. If you'd like to use this with dynamic objects, wait for version 1.1.

Links Try the demo
You can download it from here or GitHub, where you can find also source code as separate files.
Attachments
tank - rstar demo.love
runnable demo
(22.41 KiB) Downloaded 318 times
Last edited by Rick Astley on Sun Sep 05, 2021 10:11 am, edited 5 times in total.

Code: Select all

if self.thoughts[1] == 'give you up' then
   table.remove(self.thoughts,1)
end
 
User avatar
Gunroar:Cannon()
Party member
Posts: 1143
Joined: Thu Dec 10, 2020 1:57 am

Re: rstar is here for you!

Post by Gunroar:Cannon() »

This seems neat. Is it really fast? Like how fast?
The risk I took was calculated,
but man, am I bad at math.

-How to be saved and born again :huh:
User avatar
Rick Astley
Prole
Posts: 16
Joined: Sat Mar 13, 2021 11:51 am

Re: rstar is here for you!

Post by Rick Astley »

All search queries share the same base idea. Starting from the root: for each child see if it satisfies the input condition (intersects with rectangle or circle/contains a point); if yes, repeat on its children the same loop and go on until leaf level is reached.
On the average case, that loop is done for one node per tree level. In numbers this means that M x tree height rectangles are cheked. Tree height can be calculated using the formula logMN, where N is the number of rectangles you inserted in the tree.
For example if N=1000 and M=20, height is equal to 3: on average 20x3=60 rectangles are checked instead of a thousand.
In conclusion this is that fast, and if correctly tuned to the desired situation even faster.

Code: Select all

if self.thoughts[1] == 'give you up' then
   table.remove(self.thoughts,1)
end
 
User avatar
Gunroar:Cannon()
Party member
Posts: 1143
Joined: Thu Dec 10, 2020 1:57 am

Re: rstar is here for you!

Post by Gunroar:Cannon() »

Wow. Wait... so , like, if I want to get all bullet type entities in a tree that has entities added to it, for example, will it be faster than iterating through a table of all entites and getting which one is a bullet?
The risk I took was calculated,
but man, am I bad at math.

-How to be saved and born again :huh:
User avatar
pgimeno
Party member
Posts: 3674
Joined: Sun Oct 18, 2015 2:58 pm

Re: rstar is here for you!

Post by pgimeno »

Rick Astley wrote: Sun Aug 01, 2021 2:53 pmIt must be said that it's not always the best choice: to grow a optimal structure, insertion and deletion take a lot of effort, meaning that for a dynamic set of objects this is not a convenient technique.
How slow is that? Like, how many insertions/deletions per frame are possible? Few? Hundreds? Thousands?
User avatar
Rick Astley
Prole
Posts: 16
Joined: Sat Mar 13, 2021 11:51 am

Re: rstar is here for you!

Post by Rick Astley »

If you don't mind I would like to answer both your questions in a single post.

R-Tree seems to be the king in handling rectangles (R stands for Rectangle). It's so popular that there are a lot of variants, which are improvements of the base idea. I chose R* variant because in my opinion it is the one which does a better job in making searches fast.
E.g. Google Maps probably uses a couple of those: one on the server to find routes and places nearby, and one on your phone to render the map on the screen.

However, this seems to be the first public implementation of the R*Tree written in Lua; I didn't find anything on Internet.
As you can see in the paper I linked at the bottom of the main post, authors give detailed benchmark data but:
  1. Their implementation is written in a weird language.
  2. They did it in the 90s
pgimeno wrote: Sun Aug 01, 2021 9:50 pm
Rick Astley wrote: Sun Aug 01, 2021 2:53 pmIt must be said that it's not always the best choice: to grow a optimal structure, insertion and deletion take a lot of effort, meaning that for a dynamic set of objects this is not a convenient technique.
How slow is that? Like, how many insertions/deletions per frame are possible? Few? Hundreds? Thousands?
That said I can't give you an answer now because I'm missing data, but I think my next step is going to be to perform some tests and post here my results. I can say that if you put static objects(like rocks, trees, walls) in it, which won't move much during gameplay, any physics-related task regarding them is going to be light-speed fast.
Gunroar:Cannon() wrote: Sun Aug 01, 2021 9:40 pm Wow. Wait... so , like, if I want to get all bullet type entities in a tree that has entities added to it, for example, will it be faster than iterating through a table of all entites and getting which one is a bullet?
Well this tree organises entities using their position in the world, so is useful for tasks like "find which bullet entities are hitting a wall, and should therefore stop or bounce". For your example a simple table is good; in that kind of tasks this structure isn't very helpful.
Anyway in the example I made I would suggest to keep bullets outside the tree, and ask the tree for each bullet which walls is touching on every frame.

Code: Select all

if self.thoughts[1] == 'give you up' then
   table.remove(self.thoughts,1)
end
 
User avatar
GVovkiv
Party member
Posts: 686
Joined: Fri Jan 15, 2021 7:29 am

Re: rstar is here for you!

Post by GVovkiv »

Does rstar stands for Rick Star?
User avatar
Gunroar:Cannon()
Party member
Posts: 1143
Joined: Thu Dec 10, 2020 1:57 am

Re: rstar is here for you!

Post by Gunroar:Cannon() »

Haha. no. I don't think the algorithm was made by him, just this implementation. Stands for rectangle (?)

Really liking the sound of all this :awesome: . And then does that mean it's really fast in getting, let's say, all objects touching/inside an entity?
The risk I took was calculated,
but man, am I bad at math.

-How to be saved and born again :huh:
User avatar
Rick Astley
Prole
Posts: 16
Joined: Sat Mar 13, 2021 11:51 am

Re: rstar is here for you!

Post by Rick Astley »

GVovkiv wrote: Mon Aug 02, 2021 3:01 pm Does rstar stands for Rick Star?
I really wish it was like that haha :rofl:

Code: Select all

if self.thoughts[1] == 'give you up' then
   table.remove(self.thoughts,1)
end
 
User avatar
Rick Astley
Prole
Posts: 16
Joined: Sat Mar 13, 2021 11:51 am

Re: rstar is here for you!

Post by Rick Astley »

Gunroar:Cannon() wrote: Mon Aug 02, 2021 4:40 pm Haha. no. I don't think the algorithm was made by him, just this implementation. Stands for rectangle (?)

Really liking the sound of all this :awesome: . And then does that mean it's really fast in getting, let's say, all objects touching/inside an entity?
You're right. I'm going to add a demo project soon with some collision detection. Stay tuned ;)

Code: Select all

if self.thoughts[1] == 'give you up' then
   table.remove(self.thoughts,1)
end
 
Post Reply

Who is online

Users browsing this forum: No registered users and 1 guest