I'm using PDO to execute a statement with an IN
clause that uses an array for it's values:
$in_array = array(1, 2, 3);
$in_values = implode(',', $in_array);
$my_result = $wbdb->prepare("SELECT * FROM my_table WHERE my_value IN (".$in_values.")");
$my_result->execute();
$my_results = $my_result->fetchAll();
The above code works perfectly fine, but my question is why this doesn't:
$in_array = array(1, 2, 3);
$in_values = implode(',', $in_array);
$my_result = $wbdb->prepare("SELECT * FROM my_table WHERE my_value IN (:in_values)");
$my_result->execute(array(':in_values' => $in_values));
$my_results = $my_result->fetchAll();
This code will return the item who's my_value
equals the first item in the $in_array
(1), but not the remaining items in the array (2, and 3).
I've just come up against this problem and coded a small wrapper. It's not the prettiest or best code I'm sure, but it might help somebody so here it is:
E.g, passing these in:
Becomes:
It's not bulletproof, but as a sole dev for my needs it does the job fine, so far anyway.
I was able to do it like this. The idea is one value is coming in from a search form. They are looking for something and the value might be in one of the following two fields: thisField, thatField, or equal to someField.
PDO is not good with such things. You need to create a string with question marks dynamically and insert into query.
In case there are other placeholders in the query, you could use the following approach (the code is taken from my PDO tutorial):
You could use
array_merge()
function to join all the variables into a single array, adding your other variables in the form of arrays, in the order they appear in your query:In case you are using named placeholders, the code would be a little more complex, as you have to create a sequence of the named placeholders, e.g.
:id0,:id1,:id2
. So the code would be:Luckily, for the named placeholders we don't have to follow the strict order, so we can merge our arrays in any order.
Variable substitution in PDO prepared statements doesn't support arrays. It's one for one.
You can get around that problem by generating the number of placeholders you need based on the length of the array.
As I understand it it is because PDO will treat the $in_values contents as a single item and will quite it accordingly. PDO will see 1,2,3 as a single string so the query will look something like
SELECT * FROM table WHERE my_value IN ("1,2,3")
You may think that changing the implode to have quotes and commas will fix it, but it will not. PDO will see the quotes and change how it quotes the string.
As to why your query matches the first value, I have no explanation.
As PDO doesn't seem to provide a good solution, you might as well consider using DBAL, which mostly follows PDO's API, but also adds some useful features http://docs.doctrine-project.org/projects/doctrine-dbal/en/latest/reference/data-retrieval-and-manipulation.html#list-of-parameters-conversion
There are probably some other packages out there that don't add complexity and don't obscure the interaction with the database (like most ORM do), but at the same time make small typical tasks bit easier.