2016-01-21 92 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子句。