2016-01-21 173 views
1

搜索字符串被传递给一个存储过程,我们有,这一点,例如现在逗号分隔的关键字

"this is a search string". 

,用于搜索的查询是这样的:

Select * From Table Where Keywords Like '%this is a search string%' 

的“在数据库中存储关键字”列值以逗号分隔,所以在这种情况下:

"this, is, a, search, string" 

显然,上面的查询会沤瓮没有结果,我今天下午花了我怎么能做到这一点。

+3

你可以将它与'replace(关键词,',','')'进行比较。 – SomeJavaGuy

+12

这是在将多个值存储在单个列中时出现的众多问题之一,请考虑一个设计,其中每个关键字位于关键字表中的自己的行中,并且使用一对多表将各个关键字拼接在一起。 –

+2

嗨亚历克斯,这是我的未来计划,因为我遗憾地继承了这个项目。 –

回答

1

使用功能在这里找到:http://sqlperformance.com/2012/07/t-sql-queries/split-strings

CREATE FUNCTION dbo.SplitStrings_Moden 
(
    @List NVARCHAR(MAX), 
    @Delimiter NVARCHAR(255) 
) 
RETURNS TABLE 
WITH SCHEMABINDING AS 
RETURN 
    WITH E1(N)  AS (SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 
         UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 
         UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1), 
     E2(N)  AS (SELECT 1 FROM E1 a, E1 b), 
     E4(N)  AS (SELECT 1 FROM E2 a, E2 b), 
     E42(N)  AS (SELECT 1 FROM E4 a, E2 b), 
     cteTally(N) AS (SELECT 0 UNION ALL SELECT TOP (DATALENGTH(ISNULL(@List,1))) 
         ROW_NUMBER() OVER (ORDER BY (SELECT NULL)) FROM E42), 
     cteStart(N1) AS (SELECT t.N+1 FROM cteTally t 
         WHERE (SUBSTRING(@List,t.N,1) = @Delimiter OR t.N = 0)) 
    SELECT Item = SUBSTRING(@List, s.N1, ISNULL(NULLIF(CHARINDEX(@Delimiter,@List,s.N1),0)-s.N1,8000)) 
    FROM cteStart s; 

你可以做这样的事情

SELECT a.pk, COUNT(*) AS hits 
FROM Table AS a 
CROSS APPLY dbo.SplitStrings_Moden(a.Keywords, ',') b 
WHERE 
    CHARINDEX(b.Item, 'this is a search string') > 0 
GROUP BY a.pk 
ORDER BY COUNT(*) DESC 

基本上你正在创建Keyword值的爆发表。然后搜索查看哪些包含搜索字符串中的Keyword,按主键进行分组并按命中次序排序。

+0

谢谢,这似乎工作。我将与此合作,并着眼于重新设置这个设置。 –

+1

这样更好,但是如果你阅读那篇文章,你会意识到这是来自Jeff Moden的分离器,而Aaron修改了它,因此它适合于他的其他测试。问题是,当你将输入转换为varchar(MAX)时,性能会受到严重影响。杰夫的原始文章可以在这里找到。 http://www.sqlservercentral.com/articles/Tally+Table/72993/ –

1

正如在评论中提到的,这是存储关键字的错误方法。您应该有一个表,每个实体一行,每个关键字一行。

有时,我们被其他人的糟糕的设计决定所困扰。如果是这样,你可以谷歌分拆功能,并做类似:

Select * 
From Table t cross apply 
    (select ltrim(rtrim(item)) as keyword 
     from dbo.split(t.keywords, ',') 
    ) tk cross apply 
    (select ltrim(rtrim(@item)) as keyword 
     from dbo.split(@Keywords) 
    ) input 
where input.keyword = tk.keyword; 

这给你的比赛。如果你想andor他们(你的问题没有指定),那么你会使用table主键上的聚合,并包括合适的having子句。