2

我正在编写 PDO 准备,但我想知道是否有另一种编写它的好方法,以便它运行得更快。这是代码:

function comp_post_code($cat, $comp_post_code){
            global $DBH;
            $STH = $DBH->prepare("SELECT * from uk_data where 
                                    cat10 like :comp_post_code  AND (
                                    cat1 like :cat OR
                                    cat2 like :cat OR
                                    cat3 like :cat OR
                                    cat4 like :cat OR
                                    cat5 like :cat OR
                                    cat6 like :cat OR
                                    cat7 like :cat 

                                    )") ;
            $STH->bindValue(':cat', "%$cat%", PDO::PARAM_STR);
            $STH->bindValue(':comp_post_code', "$comp_post_code%", PDO::PARAM_STR);
            $STH->execute();
            $STH->setFetchMode(PDO::FETCH_ASSOC);
            return $STH;
            $DBH = Null;
            }

我想要表中的完整数据,所以我正在使用 Select。* ....谢谢

编辑:- cat10 是邮政编码,cat 1 到 cat7 是类别。我需要搜索给定邮政编码中的类别。

这是表格格式:

CREATE TABLE IF NOT EXISTS `uk_data` (
  `slno` int(10) NOT NULL AUTO_INCREMENT,
  `comp_name` varchar(150) DEFAULT NULL,
  `comp_no` varchar(50) DEFAULT NULL,
  `comp_street` varchar(100) DEFAULT NULL,
  `comp_area` varchar(100) DEFAULT NULL,
  `comp_post_code` varchar(15) DEFAULT NULL,
  `comp_phone` varchar(100) DEFAULT NULL,
  `comp_phone1` varchar(100) DEFAULT NULL,
  `cat1` varchar(100) DEFAULT NULL,
  `cat2` varchar(100) DEFAULT NULL,
  `cat3` varchar(100) DEFAULT NULL,
  `cat4` varchar(100) DEFAULT NULL,
  `cat5` varchar(100) DEFAULT NULL,
  `cat6` varchar(100) DEFAULT NULL,
  `cat7` varchar(100) DEFAULT NULL,
  `cat8` decimal(9,6) DEFAULT NULL,
  `cat9` decimal(9,6) DEFAULT NULL,
  `cat10` varchar(15) DEFAULT NULL,
  PRIMARY KEY (`slno`),
  UNIQUE KEY `Phone` (`comp_phone`),
  KEY `cat10` (`cat10`)
) ENGINE=MyISAM  DEFAULT CHARSET=latin1 AUTO_INCREMENT=74504 ;

我正在使用的输出:

$uk_data_radious = comp_post_code($q,$pc_o);
            while (($row = $uk_data_radious->fetch()) !== false) {
            $comp_name[] = $row['comp_name'];
            $comp_phone[] = $row['comp_phone'];
            $comp_phone1[] = $row['comp_phone1'];
            $post_code[] = $row['cat10'];
            $post_code1[] = $row['comp_post_code'];
            $comp_no[] = $row['comp_no'];
            $comp_street[] = $row['comp_street'];
            $comp_area[] = $row['comp_area'];
            $cat1[] =  $row['cat1'];
            $cat2[] =  $row['cat2'];
            $cat3[] =  $row['cat3'];
            $cat4[] =  $row['cat4'];
            $cat5[] =  $row['cat5'];
            $cat6[] =  $row['cat6'];
            $cat7[] =  $row['cat7'];
            $distance_m[] = distance($Latitude[0],$Longitude[0],$row['cat8'],$row['cat9'],"M");
            }
4

1 回答 1

2

全文搜索在长字符串上的搜索效果更好。但它需要创建一个特殊的索引,并更改查询(并且与 InnoDB 表不兼容)。

这是相当多的工作。但是如果类似查询真的很慢,FTI(全文索引)是一个快速的选择。

--

保留like查询,您无能为力 - 当然除非改变您在列中组织数据的方式 - 即为了避免like

为了以某种方式优化like查询,您可以将所有猫合并到一个列中(即添加一个新列catx),用一个不会出现在您的中的特殊字符分隔, like :,并且只执行一个“like”(这应该会加快一点)。例如

cat1: alpha
cat2: beta
cat3: gamma
...

catx is :alpha:beta:gamma:

并在 catx 上做类似的事情

catx like :cat
  • 要搜索特定的,请搜索“:mycat:”
  • 搜索cat的一部分,以单词开头,搜索 ":start" 或
  • 要搜索结束词,请搜索“end:”

类似搜索算法长字符串上比在较小字符串上运行多次更有效。当搜索字符串超过 3 个字符时,MySQL 使用非常高效的Turbo Boyer-Moore 算法。

但我必须警告你:这个策略有几个限制

  • 不应出现在猫中的特殊分隔符
  • 对cat[1-7]的任何更新都需要catx调整,因此如果可能会发生很大变化,那么这可能不是一个好的解决方案
  • 这个解决方案通常适用于众所周知的数据——比如标识符——格式很少改变

根据经验,我认为您不能期望通过这种策略获得超过 25% 的收益。

于 2013-02-05T17:31:19.090 回答