Kom ihåg mig?
Home Menu

Menu


SQL-fråga

Ämnesverktyg Visningsalternativ
Oläst 2009-06-26, 07:39 #1
Innocast Innocast är inte uppkopplad
Mycket flitig postare
 
Reg.datum: Aug 2004
Inlägg: 688
Innocast Innocast är inte uppkopplad
Mycket flitig postare
 
Reg.datum: Aug 2004
Inlägg: 688
Hej,

Sitter här på morgonkvisten och funderar lite över hur jag ska formulera en SQL-fråga. Vad jag har är en ständigt växande tabell. Jag har en kolumn som är en ENUM( 'true', 'false' ), låt oss kalla den `active`. Default-värdet är 'true'.

Vad jag är ute efter är att endast ha de senaste X raderna satta till true och resten till false. Prövade med följande:

UPDATE `table` SET `active` = 'false' ORDER BY `id` DESC LIMIT X,1000

I mina ögon så sorterar den tabellen bakvänt och tar 1000 rader efter rad X och sätter dom till false. Men det fungerade inte. Prövade även denna sub query variant som inte heller fungerade...

UPDATE `table` SET `active` = 'false' WHERE `active` = 'true' ORDER BY `id` DESC LIMIT ( SELECT COUNT(*) FROM `table` WHERE `active` = 'true') - X

Notera att jag bytt ut X mot ett värde.

Jag skulle ju kunna göra två frågor av det hela, en som räknar ut alla rader och därefter gör en update på det föregående resultatet minus 20. Men det känns inte optimalt.

Scriptet kommer att köras var 5:e minut via crontab och se till att tabellen aldrig blir överfull. Har nämligen en indexerng på `active`och gör väldigt många urval på just rader med `active` = 'true'.

Några tips?
Innocast är inte uppkopplad   Svara med citatSvara med citat
Oläst 2009-06-26, 09:45 #2
greenio.se greenio.se är inte uppkopplad
Medlem
 
Reg.datum: May 2009
Inlägg: 106
greenio.se greenio.se är inte uppkopplad
Medlem
 
Reg.datum: May 2009
Inlägg: 106
Jag tror inte ORDER BY fungerar på en UPDATE över huvud taget faktiskt.

Vad sägs om att ha ett datetime-fält i tabellen och fyll i aktuell tid där när du skriver till tabellen så kan du köra en update WHERE created > $femminutersedan
greenio.se är inte uppkopplad   Svara med citatSvara med citat
Oläst 2009-06-26, 10:23 #3
Roberts avatar
Robert Robert är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Jan 2004
Inlägg: 2 103
Robert Robert är inte uppkopplad
Klarade millennium-buggen
Roberts avatar
 
Reg.datum: Jan 2004
Inlägg: 2 103
Lite pseudokod (sql dialekten kan variera):

UPDATE `table` SET `active` = 'false'
UPDATE `table` SET `active` = 'true' WHERE `id` IN (SELECT TOP(10) `id`FROM `table`ORDER BY `id`)

Mao: sätt alla till false, sen plockar du alla x antal id'n sorterade. Ifrån dessa gör du alla active till true.
Robert är inte uppkopplad   Svara med citatSvara med citat
Oläst 2009-06-26, 11:25 #4
Conny Westh Conny Westh är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Aug 2005
Inlägg: 5 166
Conny Westh Conny Westh är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Aug 2005
Inlägg: 5 166
Frågan är vad du ska åstadkomma egentligen, att sätta ett värde "tillfälligt" som du verkar göra i detta fall är ingen bra lösning för en SQL-fråga.

Du borde i stället välja en fråga där du sorterar i omvänd ordning och sedan tar de översta X raderna, ungefär så här (SQL-server):

SELECT TOP 100 * FROM tabellen ORDER BY createdate DESC

Minns inte hur syntaxen var i MySQL men det går att göra samma sak ....
Conny Westh är inte uppkopplad   Svara med citatSvara med citat
Oläst 2009-06-26, 12:04 #5
Jonas Jonas är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Feb 2004
Inlägg: 3 364
Jonas Jonas är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Feb 2004
Inlägg: 3 364
Citat:
Originally posted by ConnyWesth@Jun 26 2009, 11:25
Minns inte hur syntaxen var i MySQL men det går att göra samma sak ....
Kolla svaret ovanför ditt...
Jonas är inte uppkopplad   Svara med citatSvara med citat
Oläst 2009-06-26, 22:36 #6
Conny Westh Conny Westh är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Aug 2005
Inlägg: 5 166
Conny Westh Conny Westh är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Aug 2005
Inlägg: 5 166
Citat:
Ursprungligen postat av Jonas
Citat:
Ursprungligen postat av ConnyWesth
Minns inte hur syntaxen var i MySQL men det går att göra samma sak ....

Kolla svaret ovanför ditt...
Roberts SQL-sats gör förvisso vad trådskaparen frågade efter, men mitt inlägg var ämnat att ifrågasätta detta och försöka förstå vad trädskaparen egentligen vill ha ut av frågan.

Roberts SQL-sats är mycket ineffektiv eftersom den kräver en "tablescan med sin första SQL-sats där han sätter alla värden till false.

Jag misstänker att ett bättre sätt är att endast köra en fråga som använder ett befintligt index så att de förekomster som efterfrågas visas utan att det sker en fysisk uppdatering av databasen.

Min gissning var att det kanske löser trådskaparens egentliga problem, men det skulle gå hiskeligt mycket snabbare att köra den frågan, exempelvis:

Kod:
SELECT TOP(10) * FROM `table`ORDER BY `id` DESC )
Alternativt...

Kod:
SELECT TOP(10) * FROM `table`ORDER BY `createdate` DESC )
Conny Westh är inte uppkopplad   Svara med citatSvara med citat
Oläst 2009-06-27, 01:38 #7
Roberts avatar
Robert Robert är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Jan 2004
Inlägg: 2 103
Robert Robert är inte uppkopplad
Klarade millennium-buggen
Roberts avatar
 
Reg.datum: Jan 2004
Inlägg: 2 103
ConnyWesth; utan att jag (eller vi) ska spekulera i syftet med trådskaparens sqlfråga så kan man också anta att det booleanska värdet är en form av aggregering, dvs att slippa köra en "order by" om tabellen exempelvis används på andra sätt (kanske i joinar etc). Tabellen kan bli mer logiskt hanterbar om man sätter flaggor på de ställen där vissa vilkor är uppfyllda. Dumt eller ej, som sagt beror det på....

Sen har vi det vanliga problemet med folk som försöker optimera sina updates när de egentligen ska optimera sina selects. Ofta sker det xxx antal selects per x update/insert.

[edit: rättstavning]
Robert är inte uppkopplad   Svara med citatSvara med citat
Oläst 2009-06-27, 01:49 #8
Conny Westh Conny Westh är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Aug 2005
Inlägg: 5 166
Conny Westh Conny Westh är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Aug 2005
Inlägg: 5 166
Min misstanke är att trådskaparen vill kunna visa de 10 senast registrerade medlemmarna på webbsidan, och då skulle min enkla SELECT vara exakt det som är det han vill åstadkomma.

Att över huvud aget ha en extra kolumn med data med i detta fall reduntant information som enkelt kan fås genom en simpel SELECT är inte helt optimalt.
Conny Westh är inte uppkopplad   Svara med citatSvara med citat
Oläst 2009-06-27, 11:44 #9
Westmans avatar
Westman Westman är inte uppkopplad
Klarade millennium-buggen
 
Reg.datum: Jun 2004
Inlägg: 4 021
Westman Westman är inte uppkopplad
Klarade millennium-buggen
Westmans avatar
 
Reg.datum: Jun 2004
Inlägg: 4 021
En stored procedure med while loop fungerar också.
Westman är inte uppkopplad   Svara med citatSvara med citat
Svara


Aktiva användare som för närvarande tittar på det här ämnet: 1 (0 medlemmar och 1 gäster)
 

Regler för att posta
Du får inte posta nya ämnen
Du får inte posta svar
Du får inte posta bifogade filer
Du får inte redigera dina inlägg

BB-kod är
Smilies är
[IMG]-kod är
HTML-kod är av

Forumhopp


Alla tider är GMT +2. Klockan är nu 01:31.

Programvara från: vBulletin® Version 3.8.2
Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
Svensk översättning av: Anders Pettersson
 
Copyright © 2017