Scripting Helpers is winding down operations and is now read-only. More info→
Ad
Log in to vote
0

How do I check if there's any 2 items in a table?

Asked by
yumtaste 476 Moderation Voter
10 years ago

I'm making a sort of crafting system for my game, and I imagine the most efficient way to carry it out would be if I had a table of selected ingredients, and a method to see if there's 2 items in any order in the selected table. Example: I want to check if there's the string "foo1" anywhere in the table, and I also want to see if there's "bar2" in the table anywhere. Feel free to ask questions to verify what I'm asking, and thanks for your help!

1 answer

Log in to vote
2
Answered by
BlueTaslem 18071 Moderation Voter Administrator Community Moderator Super Administrator
10 years ago

First, solve the problem of finding an item in a table (we will say where it is in particular):

1function find(tab, element)
2    for index, value in pairs(tab) do
3        if value == element then
4            return index
5        end
6    end
7end

find will return nil if element wasn't in tab, otherwise it will return where in tab it is, ie, tab[find(tab, element)] == element.

Thus to check if something is in a table:

1if find(tab, "foo") then
2    -- "foo" is in `tab`

We could just string a few together to see if they have multiple things:

1if find(tab, "foo") and find(tab, "bar") then
2    -- they have both "foo" and "bar"

It's also straightforward to use a loop to check if all of the things are in the table:

1local required = {"foo", "bar"}
2local haveAll = true -- (have all *so far*)
3for _, thing in pairs(required) do
4    haveAll = haveAll and find(tab, thing)
5end
6 
7if haveAll then
8    -- they have all the things

The limitation of this solution is that we can't require more than one a thing.

1if find(tab, "foo") and find(tab, "foo") then
2-- same thing as
3if find(tab, "foo") then
4-- since both calls will return the same thing

Thus we could make a "count" (or find multiple) function:

1function findAll(tab, element)
2    local where = {}
3    for index, value in pairs(tab) do
4        if value == element then
5            table.insert(where, index)
6        end
7    end
8end

Now findAll will return an empty list if it isn't in tab, otherwise it will return a list of elements.

Thus

1if #findAll(tab, "foo") >= 1 then
2    -- they have at least one foo

And we can require 2 foo and 3 bar like this:

1if #findAll(tab, "foo") >= 2  and findAll(tab, "bar") >= 3 then

To put this into a loop, we can make a bunch of pairings:

1local required = {    {"foo", 2}, {"bar", 3}    }
2local haveAll = true
3for _, thing in pairs(required) do
4    -- thing[1]: ingrediant
5    -- thing[2]: quantity
6    haveAll = haveAll and ( #findAll(tab, thing[1]) >= thing[2] )
7end

Bonus option: Say you don't want to make the tables of tables like above. E.g., you want ingredients to look like this:

1local required = {"foo", "foo", "bar", "bar", "bar"}
2-- Arguably this is a worse format because you could miscount the
3-- number; still, it is probably more natural.

Then we can just go through all of the elements of required and count how many times they appear!

1local haveAll = true -- (have all *so far*)
2for _, thing in pairs(required)
3    haveAll = haveAll and ( #findAll(tab, thing) >= #findAll(required, thing) )
4end
5 
6if haveAll then

This is somewhat inefficient because it will re-check for, say, "bar" three times -- but the behavior will be correct, and this will still be very fast for small lists like ingredient lists.

0
Good answer Mystdar 352 — 10y
0
Excellent answer. This is EXACTLY what I was looking for. Thank you, my good man! yumtaste 476 — 10y
0
Wow. It's even better htan before. BlueTaslem, you are a genius! yumtaste 476 — 10y
Ad

Answer this question