Mysql搜索,按优先顺序排序

时间:2011-07-20 10:17:44

标签: mysql sql codeigniter search

我决定在我的网站上想要一个搜索字段,事情是我对搜索算法一无所知。搜索框的工作方式类似于您可以输入任意数量的名称(用户名,名字,姓氏)或用空格分隔的电子邮件。所以我想出了以下使用codeigniter的活动记录类(非常简单):

    $emails = FALSE;
    $names = FALSE;

    foreach($search_array as $value){
        if(valid_email($value)){
            $emails[] = $value;
        } else{
            $names[] = $value;
        }
    }

    if($emails){
        $this->db->where_in('email', $emails);
        if($names){
            $this->db->or_where_in('username', $names);
            $this->db->or_where_in('firstname', $names);
            $this->db->or_where_in('lastname', $names);
        }
    } else{
        if($names){
            $this->db->where_in('username', $names);
            $this->db->or_where_in('firstname', $names);
            $this->db->or_where_in('lastname', $names);
        }
    }

    $this->db->select('id, username, firstname, lastname');
    $query = $this->db->get('users');

    if($query->num_rows() > 0){
        return $query->result();            
    }

    return FALSE;

这段代码的作用是运行一堆where语句,然后简单地将它们全部发回。 也许我可以清理if语句以及电子邮件的工作方式,但这不是重点。这段代码完美无缺(至少对于一个非常小的数据库:P),但我希望它能做得更多:

  • 现在您需要输入正确的名称或电子邮件才能获得任何好结果。如果你几乎是正确的,那么得到一些建议会很好,但我可能会自己做一些喜欢的事情。

  • 我想对整个结果数组进行排序,这就是我磕磕绊绊的地方。由于电子邮件和用户名都是唯一的,我希望使用以下优先级进行搜索:

    1. 电子邮件
    2. 用户名
    3. 名字+姓氏
    4. 姓氏

目前我并不关心电子邮件+用户名+名字+姓氏,因为我相信这会让人不知所措:P

我确实发现了这个话题Search query, 'order by' priority但是因为我不知道在我的搜索中它是什么,它对我没有帮助,但也许我可以使用类似的东西?

无论如何,你们都知道解决优先级问题的方法吗?我是否可能错过了一些可能有用的主要功能?当我的数据库变大时会发生什么? ;)

感谢您的帮助!

1 个答案:

答案 0 :(得分:5)

你可以这样做:

ORDER BY email != 'value',
         username != 'value',
         firstname != 'value' AND lastname != 'value',
         firstname != 'value',
         lastname != 'value'

如果表达式为假(意味着它们是相同的),那么它等于0并且将排序高于表达式,该表达式为真(意味着它们不同),等于1并在0之后排序。

如果你发现!=令人困惑,那么这就是同样的事情:

ORDER BY email = 'value' DESC,
         username = 'value' DESC,
         firstname = 'value' AND lastname = 'value' DESC,
         firstname = 'value' DESC,
         lastname = 'value' DESC
相关问题