8

我目前有一个只有两个元素的小型搜索表单:一个名为的文本输入search和一个名为param. 在该框中选择的选项应用作参数来选择将获取我的数据库中的哪个表列。

使用mysql_函数,可以这样做:

$param  = $_POST['param' ];
$search = $_POST['search'];

$query = 'SELECT * FROM table WHERE $param LIKE "%$search%"';

但我无法让它与mysqli_语法一起工作。我正在尝试使用准备好的语句,但到目前为止我所做的最好的是:

$param  = $_POST['param' ];
$search = $_POST['search'];

if($param == 'first_name')
{
    if($prep = $link->prepare('SELECT * FROM table
                               WHERE first_name
                               LIKE CONCAT("%", ?, "%")'))
    {
        $prep->bind_param('s', $search);
        $prep->execute();
        $prep->bind_result($first_name, $last_name);

        while($prep->fetch())
            echo $first_name . ' ' . $last_name;

        $prep->close();
     }
     else
         echo 'Error while preparing statement.';
}
else if($param == 'last_name')
{
    ...
}

但是仅仅使用一堆else ifs 似乎很重复且没有效率,特别是如果我有很多列要处理。

我尝试的第一件事是参数绑定 -... WHERE ? LIKE ...$prep->bind_param('ss', $param, $search)-,但它不起作用(我仍然不知道为什么)。

有没有更智能的方法呢?

4

3 回答 3

3

如果您对每个参数使用相同的 SQL 代码,只需创建可能参数的哈希:(CGI param name => table column name)

$params = array(
    'first_name' => 'first_name',
    'last_name' => 'last_name',
);

从安全的角度来看,这要好得多,因为您可以免受 SQL 注入的影响。

然后从哈希中获取列名并将其放入查询中-您将摆脱 if-s:

$name = $params[$param];
$sql = "SELECT * FROM table
WHERE 
$name LIKE ?";

if($prep = $link->prepare($sql))
{
    $prep->bind_param('s', "%$search%");
    ...

正如@Akam 所说,不需要在查询中使用 CONCAT("%", ?, "%") - 最好将值与前面的百分比绑定。

于 2013-07-24T22:43:16.993 回答
2

根据 PHP 手册中的这个例子

http://www.php.net/manual/en/mysqli-stmt.bind-param.php#108790

您似乎最好将 '%' 添加到您正在绑定的字符串变量中 - 而不是在查询中,例如在您的示例中:

if($prep = $link->prepare('SELECT * FROM table
                           WHERE first_name
                           LIKE ?'))
{
    $search='%'.$search.'%';
    $prep->bind_param('s', $search);
    $prep->execute();
    $prep->bind_result($first_name, $last_name);

    while($prep->fetch())
        echo $first_name . ' ' . $last_name;

    $prep->close();
 }

尚未对其进行测试,但它似乎是一个明智的解决方案。

于 2013-07-24T22:53:13.083 回答
1

您不能对列名使用占位符,因此您必须正常连接列名。但是,不要仅仅用 mysqli 转义它,因为你有一组有限的列,我建议使用 while list 方法:

$allowed_params = array('first_name', 'last_name', etc);

$param  = $_POST['param' ];
$search = $_POST['search'];


if(!in_array($param, $allowed_params))
    die("Uh oh, the request seems to have an invalid param!");

if($prep = $link->prepare('SELECT * FROM table
                           WHERE ' . $param . '
                           LIKE ?'))
{
    $prep->bind_param('s', '%' . $search . '%');
    $prep->execute();
    $prep->bind_result($first_name, $last_name);

    while($prep->fetch())
        echo $first_name . ' ' . $last_name;

    $prep->close();
}
else
    echo 'Error while preparing statement.';

还要注意语句的删除concat,而是在 PHP 中连接。这对于准备好的语句很重要,因为一旦它到达服务器它实际上并没有组合它们(因此保护准备好的语句),因此除非通配符与$search字符串一起发送,否则它将无法正常工作。

于 2013-07-24T23:07:38.257 回答