Hi everyone! I recently made a module for object pooling system, and I wanted to share it
Why Use Object Pooling?
Object pooling helps improve performance by reusing instances instead of constantly creating and destroying them. This is especially useful for:
- Custom particle systems
- Projectiles & bullets
- Sound effects
- Other frequently spawned objects
Instead of calling Instance:Clone()
every time you need an object, you can pull from a pre-allocated pool.
Since objects aren’t destroyed and created anew all the time, it will help prevent memory leaks associated with connecting to events, such as .Changed
How to use it?
Get started by requiring the module, and simply create an object pool with
local Pool = ObjectPoolingModule.new(Count,Instance)
This creates an object pool size of Count
filled with clones of Instance
Parameters
The .new()
function also accepts parameters, that affect how the object pool acts, the parameters are passed in a table like
local Pool = ObjectPoolingModule.new(Count,Instance,{Parameter = State})
Available parameters include:
-
LiveReusing:boolean -- false by default
- whether or not objects that have already been loaded should be reused. If enabled, this will take the object that has been loaded for the longest time -
LoadWait:number -- 0 by default
– how many heartbeats should the.new()
function wait between the addition of each instance to the pool -
AutoExpand:boolean -- false by default
- should the table automatically expand itself if it runs out of available instances. RequiresInstance
be passed during the.new()
function -
MaxPoolSize:number -- math.huge by default
– if the table automatically expands, at what point should it stop expanding -
UnloadTimer:number -- -1 by default
– how many seconds should the instances wait after being loaded, before being unloaded automatically. Anything below 0 disables this functionality
Loading objects
To load an object into the game, simply do
local Item = Pool:LoadItem(Parent?) -- optionally sets the parent of the item, if none is provided, sets the parent to Workspace
-- IMPORTANT, the 'Item' object returned is a custom class, to refer to the Instance within the class itself, use Item.Instance
Then, the item can be unloaded again with
Item:Unload()
Events
This module comes with events you can track to make utilisation easier
-- The pool has events like
Pool.ItemLoaded:Connect(function(item)
-- when an item is loaded via :LoadItem(), this event fires and passes the item that has been loaded
end)
Pool.ItemUnloaded:Connect(function(item)
-- when an item is unloaded, this event fires and passes the item that has been unloaded
end)
Item Events
-- Items themselves have events like
Item.Unloaded:Connect(function()
-- fires AFTER the item has been unloaded from the game
end)
Item.Loaded:Connect(function()
-- fires when an item is loaded into the game
end)
Item.OnUnloading:Connect(function()
-- fires BEFORE an item is unloaded, yields for 1.5 seconds if any functions are attached to this event
-- useful if you wanna, for example, tween an object's size to 0,0,0 before its unloaded
end)
That & much much more available in this module. Reading the documentation is recommended
How I use it
Here’s a snippet of code from a custom particle system I have made, this shows how you can utilise the pooling system efficiently
for _,Obj in pairs(BloodPool:GetAll()) do
local Particle = Obj.Instance :: BasePart
local Con : RBXScriptConnection
Obj.Loaded:Connect(function()
local Size = math.random(80,140)/100
Particle.Size = Vector3.new(.1,.1,Size)
local DefaultPos = game.Workspace.Folder:GetChildren()[math.random(1,#game.Workspace.Folder:GetChildren())].CFrame
Particle.CFrame = CFrame.new(DefaultPos.Position,DefaultPos.Position + DefaultPos.LookVector)*CFrame.Angles(math.rad(math.random(-40,40)),math.rad(math.random(-40,40)),0)
local P = Particle.Position
local V = 7
local G = Vector3.new(0,-workspace.Gravity/22,0)
local D = Particle.CFrame.LookVector * 6
local T = 0
Con = game:GetService("RunService").Heartbeat:Connect(function(dt)
T += dt * V
local Pos = PositionCalc(P,D,G,T)
Particle.CFrame = CFrame.new(Particle.Position,Pos)
Particle.Position = Pos
end)
end)
Obj.Unloaded:Connect(function()
if Con then
Con:Disconnect()
end
end)
end
Install
GitHub | Roblox Asset | Documentation
I hope you find this module useful in your next project! I’m happy to answer any questions and take suggestions!