Author |
Topic |
samham
Starting Member
12 Posts |
Posted - 2003-04-28 : 12:43:22
|
I have 3 tablesPara(paraID,paraTitle,paraDate)Country(countryID,countryName)paraCountry(paraID,countryID)a paragraph can be related to many countriesi want the user to be able to get the paragraph based on the countries he choosesi have 2 possibilities:the user can choose to have ALL the countries he selects included for a paragraph or ANY country.the ANY possibility is done like this:SELECT paraID, paraTitle FROM para a WHERE EXISTS( SELECT NULL FROM paraCountry cwhere a.paraID = c.paraIDand countryID in (1,2,3))As for the ALL possibility i want that all the countries he chooses included in a paragraph.. so if a paragraph is related to countryID 1,2,3,4,5 and the user chooses 1,3,5 the paragraph is selectedthx |
|
simondeutsch
Aged Yak Warrior
547 Posts |
Posted - 2003-04-28 : 13:46:52
|
SELECT DISTINCT ParaID,ParaTitle FROM Para A INNER JOIN ParaCountry PC ON A.ParaID = PC.ParaID WHERE PC.CountryID IN(Your Choice1, Your Choice2,etc.)OR:SELECT ParaID,ParaTitle FROM Para A INNER JOIN ParaCountry PC ON A.ParaID = PC.ParaID WHERE PC.CountryID IN(Your Choice1, Your Choice2,etc.) GROUP BY A.ParaID,A.ParaTitleSarah Berger MCSD |
 |
|
samham
Starting Member
12 Posts |
Posted - 2003-04-28 : 14:32:19
|
ur query is identical to mine.. this is my first possibility, the ANY possibility.this query will return results if ANY of the choices is found ... what i want is the other possibility which returns results only if ALL the possibilities matchso if i want choice1, choice2 i only get results where paragraphs are related to choice1, choice2 in the same time , not if choice1 OR choice2 r presentso the idea is to get the results if the user choices of countries(many choices) are INCLUDED in the paragraphs choices in the paraCountry tablehope this makes it more clear |
 |
|
X002548
Not Just a Number
15586 Posts |
Posted - 2003-04-28 : 15:06:21
|
How About:CREATE TABLE Para(paraID int,paraTitle varchar(255),paraDate datetime) GOCREATE TABLE Country(countryID int,countryName varchar(50)) GOCREATE TABLE paraCountry(paraID int,countryID int)GOINSERT INTO Para (ParaId, ParaTitle, ParaDate) SELECT 1, 'This Article', Getdate()GOINSERT INTO Country(countryId, countryName) SELECT 1, 'USA' UNION ALL SELECT 2, 'GB' UNION ALL SELECT 2, 'Canada'GOINSERT INTO paraCountry (ParaId, CountryID) SELECT 1,1 UNION ALL SELECT 1,2 UNION ALL SELECT 1,3GODECLARE @UserSelect1 Int, @UserSelect2 Int, @UserSelect3 Int SELECT @UserSelect1 = 1, @UserSelect2 = 2, @UserSelect3 = 3CREATE TABLE #Temp(UserSelect Int)INSERT INTO #Temp(UserSelect) SELECT @UserSelect1 UNION ALL SELECT @UserSelect2 UNION ALL SELECT @UserSelect3SELECT Distinct ParaId o FROM ParaCountry WHERE EXISTS (SELECT 1 FROM #Temp t LEFT JOIN paraCountry i ON t.UserSelect = i.CountryId WHERE o.ParaId = i.ParaId AND i.CountryId Is Not Null HAVING Count(*) = (SELECT Count(*) FROM #Temp))DELETE FROM ParaCountry WHERE CountryID = 3SELECT Distinct ParaId o FROM ParaCountry WHERE EXISTS (SELECT 1 FROM #Temp t LEFT JOIN paraCountry i ON t.UserSelect = i.CountryId WHERE o.ParaId = i.ParaId AND i.CountryId Is Not Null HAVING Count(*) = (SELECT Count(*) FROM #Temp))GODROP TABLE ParaCountryGODrop TABLE CountryGODROP TABLE ParaGODROP Table #TempGO Brett8-)EDIT: <homer>dooooooh forgot to add the coorelation</homer>Edited by - x002548 on 04/28/2003 15:10:47 |
 |
|
samham
Starting Member
12 Posts |
Posted - 2003-04-28 : 15:46:31
|
thank u X002548i tried it and it worksin fact i have a lot of tables other than the country table and want to search in the same way, wont creating multiple temp tables affect performance ? can this be done with a table datatype for example, or with my comma seperated list of values without creating a temp table.what am doing is building the query dynamically and then send it back to sql server to be executed, so i have a lot of search criteria before i reach this one so creating a temp table will be a little comlicated but if it's the only way i'll do itthx again |
 |
|
X002548
Not Just a Number
15586 Posts |
Posted - 2003-04-29 : 09:54:15
|
That's the only way I could think of...certainly doesn't mean the only way.And if you're worried about a performance hit, you will certainly get one with using dynamic sql.As far as temp tables versus a table variable, I've seen comments in both directions. Unless you have SQL 2000 then it's not even an option. Temp tables Use tempdb, where as table datatypes are in memory. I believe the trade offs are in how you manage those resources, how much data you have, number of transactions, ect.I wonder though how search engines do this (or do they?) Like in Google, their advanced options have"All words must match", or "At least one word must match" which is similar to what you're doing.btw, what are you doing?Brett8-) |
 |
|
samham
Starting Member
12 Posts |
Posted - 2003-04-29 : 11:59:52
|
am using sql server 2000 and what am doing is a search engine a little faster than google no what am doing is a search engine for news, each paragraph is related to countries,categories etc...i will try to use ur method by using a function which returns a table from a comma delimited list of valuesam not sure if using the function and looping throught the list of values and inserting them in a table to be returned from the function is faster then using the temporary table |
 |
|
jsmith8858
Dr. Cross Join
7423 Posts |
Posted - 2003-04-29 : 12:17:26
|
This is very generic, but:If you have a list of 3 words to search for, and you can put them into a table or a table variable called "Words", you can make sure that ALL words match this way:Select DocumentIDFROMDocumentsINNER JOINWordsONDocumentText LIKE '%' + Word + '%'GROUP BY DocumentIDHAVING COUNT(*) = (SELECT COUNT(*) FROM Words)And, of course, to check if at least 1 matches, eliminate the HAVING clause:Select DocumentIDFROMDocumentsINNER JOINWordsONDocumentText LIKE '%' + Word + '%'GROUP BY DocumentID- Jeff |
 |
|
X002548
Not Just a Number
15586 Posts |
Posted - 2003-04-29 : 12:56:57
|
Yeah, I knew there was an easier way...got myself in the over-thinking mode....Thanks JeffBrett8-) |
 |
|
samham
Starting Member
12 Posts |
Posted - 2003-04-29 : 16:36:03
|
thank you very much Jeff it works great |
 |
|
jsmith8858
Dr. Cross Join
7423 Posts |
Posted - 2003-04-29 : 16:59:27
|
What's cool is if you are matching more than 1 word, you can also return a matching % per document:Select DocumentID, 1.0 * COUNT(*) / (SELECT COUNT(*) FROM Words) as MatchPctFROM Documents INNER JOIN Words ON DocumentText LIKE '%' + Word + '%' GROUP BY DocumentID and do all kinds of neat things like that. - Jeff |
 |
|
AjarnMark
SQL Slashing Gunting Master
3246 Posts |
Posted - 2003-04-30 : 14:05:30
|
And if you're looking for a handy way to turn that CSV list into rows in a table, so that you can then use Jeff's excellent solution, you might want to read Rob Volk's article: Parsing CSV Values into Multiple Rows------------------------------------------------------The more you know, the more you know you don't know. |
 |
|
samham
Starting Member
12 Posts |
Posted - 2003-05-01 : 05:02:52
|
to parse my csv list i wrote the following function:CREATE FUNCTION ParseArrayOfInt(@string varchar(100), @delimiter char(1))RETURNS @intTable table (intElement int)BEGIN declare @myValue varchar(5) declare @i int--first position of an element to extract from the csv list declare @j int--last position of the element set @i=1 set @j=1 while charindex(@delimiter,@string,@j)<>0 begin set @j=charindex(@delimiter,@string,@j)+1 set @myValue=substring(@string,@i,@j-@i-1) insert into @intTable(intElement) values(cast(@myValue as int)) set @i=@j end RETURN ENDNB: the list must end with the delimiter |
 |
|
|