5

给定一个搜索字符串,我需要选择每条记录,其中(在执行搜索的字段中)至少有一个以给定文本开头的单词。

例如:

'John Doe'

必须使用搜索字符串进行选择,例如:

'joh'
'do'
'JOHN doe'

不必选择

'ohn'
'oe'

我需要(可能)避免全文搜索。

我发现工作的是

$query = 'SELECT * FROM MYTABLE WHERE SEARCHFIELD LIKE "' . $searchText . '%"'
                                . 'OR SEARCHFIELD LIKE "% ' . $searchText . '%"'

我在问是否有更好的方法来做到这一点。

(对于“更好的方式”,我的意思是更好的性能相同的性能但更优雅

此外,由于查询将使用准备好的语句构建,我应该如何在搜索字符串中取消转义LIKE元字符?

4

2 回答 2

3

如问题中所述,查询

$query = 'SELECT * FROM MYTABLE WHERE SEARCHFIELD LIKE "' . $searchText . '%"'
                                . 'OR SEARCHFIELD LIKE "% ' . $searchText . '%"'

适用于匹配SEARCHFIELD包含以(或等于)开头的单词的记录$searchText


关于性能,我已经在我的开发机器上进行了测试MBP 2,2 GHz i7 quad core

在 4.000 条记录上搜索一个单词大约需要 40 毫秒。

记录通常被索引(没有全文)。

我有几千条记录,查询不经常运行,所以对我来说很好。
该解决方案可能不适用于其他环境。


为了使用上面的查询构建一个准备好的语句,我使用了这里描述的技术:

转义 MySQL 通配符

结果代码如下:

function like($s, $e)
{
    return str_replace(array($e, '_', '%'), array($e . $e, $e . '_', $e . '%'), $s);
}

/* ... */

/* create a prepared statement */
$stmt = $mysqli->prepare(
    'SELECT * FROM MYTABLE WHERE SEARCHFIELD LIKE ? ESCAPE "=" OR SEARCHFIELD LIKE ? ESCAPE "="'
); 

if( $stmt )
{
    /* escape the text */
    $escSearchText = like( $searchText, "=" );

    /* 'like' parameters */
    $like1 = $escSearchText . "%";
    $like2 = "%" . $escSearchText . "%";

    /* bind parameters for markers */
    $stmt->bind_param( "ss", $like1, $like2 );

/* ... */
于 2014-05-14T14:44:55.527 回答
2

用这个:

$query = "SELECT * FROM MyTable WHERE searchfield LIKE CONCAT('%', ?, '%')";

您不需要OR条件 - 如果一个字段匹配search%,它也将匹配%search%

于 2013-10-31T22:36:42.773 回答