What are Metatables?
Metatables, despite what most people say, are not as complicated as they seem at first glance. People like to describe them as tables within tables, or tables you can attach to other tables, but I personally don't like these descriptions because they don't seem very clear and they certainly don't emphasize the power of metatables. If I had to sum them up in my own words, I'd describe them as such:
Metatables allow you to monitor what is happening to a table. Is it gaining a new entry? Is it invoking tostring
? Is it trying to be called as a function t = {}; t()
? Etc.
Metatables exist to allow the developer to run code when an action is performed on a table.
Some people will also compare metatables to behaving like an event listener for a table, which is not a bad comparison so long as you know the differences between the two. Now, let's move on to how they're used.
Constructing Metatables
There is a built-in function in Lua called setmetatable
-- this function is the only reason metatables exist. You can think of setmetatable as a constructor for creating metatables. However, the function itself only takes two regular tables as it's arguments. Here's an example:
The first argument is the table that will keep it's regular behavior, and the second argument is the table that will now behave as the first table's metatable. Here's a more explicit example:
Once setmetatable is called, it will return the table it was given as the first argument. In this case, it will return t
. However, t
now has a metatable attached to it. In this case, the metatable is mt
.
This is why people describe metatables as "tables you can attach to other tables," but do you see how much explaining had to be done before we reached that conclusion? Not the best beginner-friendly description in my opinion.
Metamethods
Now that you hopefully have a better understanding of metatables, let's table about metamethods. Metamethods are the fields that go inside of the metatable to control what happens in your program when some action is performed on a table you set a metatable to.
If you like the event listener analogy, you can think of metamethods as events that fire when specific actions are performed to on table.
Here is a list of some available metamethods in Lua 5.1:
__index
__newindex
__mode
__call
__mul
__div
__mod
__metatable
__tostring
__len
... and the list goes on.
Don't worry; you don't have to know them all by memory. Most of these metamethods are rarely used anyway. The most frequent metamethods you will probably use are: __index
, __newindex
, __call
, __metatable
, and __tostring
.
The rest you'll only use if you're doing something incredibly specific. But anyway, these are the fields that you will set in your metatable. Let's try __index
, for example:
3 | __index = function (t, key) |
4 | return "the index " .. key .. " does not exist" |
The __index metamethod will be invoked when you try to retrieve a value from a table with an index that does not exist in the table. __index can be a table or a function, but I won't go in depth about the description of each metamethod in this answer. You can easily find those anywhere online.
Anyway, in this case, when __index is invoked, it will provide you with the index that you searched t
for. This value is the second parameter of the __index function in the example above (the key
parameter). Let's watch put it to use:
03 | __index = function (t, key) |
04 | return "the index " .. key .. " does not exist" |
We indexed t
for "someKey"
, but since that index does not exist, Lua checks to see if t
has a metatable with an __index metamethod. If so, return whatever __index returns.
Conclusion
This was a very lengthy answer, and it only scratches the surface of how metatables can be used. My goal here was to broaden your understanding of metatables and metamethods more than covering some examples of classes or however else they can be used. If you'd still like an explanation on that, let me know and maybe we can chat through discord.
Hope this helped!
External Sources