HyperLink
Bejelentkezés
E-mail: 
Jelszó: 





Skip Navigation Links
 

MS SQL adatbázis kezelés Delphi-ből


7. rész

Példaprogram letöltése

9335 bájt

Aktuális cikkünkben az Identity mező használatát mutatjuk be.

Az SQL Server identity mezője nagyon hasonló a Paradox autoincrement mezőjéhez vagy a Microsoft Access counter mezőjéhez, vagyis minden alkalommal automatikusan megnövekszik az értéke, amikor a táblába rekordot szúrunk be, mégpedig annyival, amennyit a tábla definiálásakor beállítottunk.

IDENTITY (seed, increment)
A mezőt remekül használhatjuk egyedi azonosító érték tárolására akár a Primary Key constraint-tel együtt is. Meg kell adnunk egy kezdeti értéket, ami a legelső beszúrás alkalmával lesz kiosztva (seed), és egy növekményt (increment). Ennyivel lesz több a mező értéke a következő beszúráskor. Ha elhagyjuk a két paramétert, (1,1)-nek tekinti az SQL Server. A identity mező csak egész típusú lehet, tehát tinyint, smallint, int, decimal(p,0), vagy numeric(p,0). Nézzünk egy példát:
CREATE TABLE jobs
(
 job_id     smallint  IDENTITY(1,1) PRIMARY KEY CLUSTERED,
 job_desc varchar(50)  NOT NULL,
 min_lvl    tinyint   NOT NULL CHECK (min_lvl >= 10),
 max_lvl   tinyint   NOT NULL CHECK (max_lvl <= 250)
)
Amiért érdemes ezzel a mezővel külön is foglalkozni, az az a probléma, hogy egy beszúrás után a BDE nem tudja, hogy az identity mező milyen értéket kapott. Bizonyos alkalmazásoknál ez problémát jelenthet. Például ha egy táblában identity mező a primary key, a táblát egy TTable komponensen keresztül érjük el és az adatokat a primary key szerint rendezve nézzük, egy beszúrás után a BDE nem fogja megtalálni az új rekordot. Mégpedig azért nem, mert a rekord beszúrása előtt a kulcs értéke Null, és ez csak az SQL Serverben kap helyes értéket, amiről a BDE nem kap értesítést. Az aktuális rekordra mutató pointer ezért a tábla első rekordjára mutat ahelyett, hogy a most beszúrt rekordra mutatna.

A problémát a példaprogram bal oldali panelje szemlélteti. A program a 4. fejezetben létrehozott test1 nevű aliasra csatlakozik, ami az SQL Server pubs nevű példa adatbázisára mutat. Innen megnyitjuk a jobs nevű táblát, ami tartalmaz identity mezőt és ugyanez a mező primary key is. Ha most a navigator segítségével beszúrunk egy értéket, látni fogjuk, hogy az identity érték nem jelenik meg és a rekord az első helyre kerül. Több, egymást követő beszúrás esetén pedig mindig csak az utolsó jelenik meg. Amíg nem kérünk egy frissítést, nem látjuk az identity értékeket.

Beszúráskor vigyázzunk arra, hogy magát az identity értéket nem adhatjuk meg, továbbá a min_lvl és max_lvl mezőkre check constraint van definiálva, ezért itt sem adhatunk meg akármekkora értéket. (A tábla definícióját lásd a fenti példában.)

A probléma feloldására számtalan megoldás létezik, mutatunk egy párat.


1. Alternatív mező használata a rendezésre

Ne az identity mező szerint rendezzük a táblát. Ha a tábla szerkezete megengedi, ügyesen válasszunk olyan rendezést, ami hasonló vagy ugyanolyan sorrendet produkál, mint az identity mező szerinti.


2. @@identity változó lekérdezése

Az SQL Server az utolsó beszúráskor kalkulált identity értéket a @@identity globális változóban tárolja. Ha az utolsó INSERT vagy SELECT INTO nem érintett identity mezőt, a @@identity változó értéke NULL lesz. Ha a művelet egyszerre több sort is érintett, az utolsó beszúrt identity mező értéke lesz a nyerő. Írhatunk egy egyszerű tárolt eljárást, ami a beszúrás után ezt rögtön lekérdezi. A kapott érték alapján már könnyen megtaláljuk az utoljára beszúrt rekordot a Delphi-s alkalmazásunkban is.

Ez a megoldás azonban hiába elegáns, nem maradéktalanul problémamentes. Ha ugyanis a táblán olyan INSERT trigger van, ami szintén végrehajt néhány beszúrást más identity mezőt tartalmazó táblákba, a @@identity értéke már ezt a legutolsó értéket fogja tartalmazni, és nem azt, ami nekünk kell. Ha pedig a trigger olyan táblába szúr be, ahol nincs identity mező, a globális változó értéke NULL lesz. Szintén probléma, hogy ha egy INSERT vagy SELECT INTO művelet meghiúsul, vagy egy tranzakciót visszagörgetünk (rollback), a @@identity értéke nem áll vissza az utolsó, konzisztens értékre.

Ha ezt a megoldást választjuk, a fentieket is figyelembe kell vennünk.


3. UNIQUE mező felhasználása

Szerencsénk van, ha az adattáblánkban az identity mező mellett szerepel egy UNIQUE mező is. Ekkor az INSERT után ezt a mezőt felhasználva még az SQL Serverben visszakeressük a rekordot, és kiolvassuk belőle az identity mező értékét, amit visszaadunk az alkalmazásunknak.

Például tegyük fel, hogy a jobs táblában a job_desc mezőt UNIQUE-nak definiáltuk. Ekkor az adatok beszúrását végző tárolt eljárás a következőképpen nézne ki:
CREATE PROCEDURE sp_add_job
    @job_desc  varchar(50),
    @min_lvl   tinyint,
    @max_lvl   tinyint
AS

insert into jobs (job_desc, min_lvl, max_lvl) 
          values (@job_desc, @min_lvl, @max_lvl)

select job_id
  from jobs
  where job_desc = @job_desc
Mivel a job_desc mező egyedi, biztosan a helyes identity értéket kapjuk. Ha nincs ilyen UNIQUE mezőnk, akkor is gondolkodhatunk úgy, hogy van-e olyan kombinációja a mezőknek, amelyekre igen valószínűtlen, hogy ugyanazokat az értékeket veszik fel.


4. Nem használunk identity mezőt kulcsmezőként

Valami más módon generálunk egyedi azonosítót, például egy táblában minden beszúráskor inkrementálunk egy egész típusú mezőt, és minden beszúrás előtt kiolvassuk az értékét. Az INSERT utasításban ekkor minden mezőt megadunk, így föl sem merülnek az előbb említett problémák. Persze nem valami elegáns, hogy minden beszúrás előtt egy extra alkalommal az adatbázisszerverhez fordulunk, de ha a feladat jellege olyan, hogy nem kell sokszor adatot beszúrni, ez a szisztéma is kitűnően működik.

A példaprogramban a 2. módszerre mutatunk példát. A beszúrás után lekérdezzük a @@identity mezőt, és a kapott érték alapján visszakeressük a rekordunkat. A program működéséhez hozzuk létre az alábbi tárolt eljárást az SQL Server pubs példaadatbázisában, mondjuk az SQL Server Enterprise Manager-ével:
CREATE PROCEDURE sp_add_job
    @job_desc  varchar(50),
    @min_lvl   tinyint,
    @max_lvl   tinyint
AS

insert into jobs (job_desc, min_lvl, max_lvl) 
          values (@job_desc, @min_lvl, @max_lvl)

select @@identity as result


Cikksorozat

#IDKategóriaCikk címeSorozat
2591WindowsTippek és trükkök - RAS - Modem csengetési szám állítása1. rész
2622WindowsTippek és trükkök - Program futtatása más felhasználóként2. rész
2640WindowsA Windows ikonméretének megváltoztatása és Windows 2000 Asztaltémák3. rész
2657WindowsTippek és trükkök - Internet Explorer4. rész
2667WindowsTippek és trükkök5. rész
2684WindowsAlapértelmezések állítása, telepítési fájlok helye, intéző nézetek6. rész
2696WindowsBiztonsági trükkök7. rész
2702WindowsWindows XP trükkök8. rész
2729WindowsWindows 2000 és XP tippek, trükkök9. rész
2757WindowsRegistry trükkök10. rész
2784WindowsTippek, trükkök11. rész
2829WindowsTippek, trükkök12. rész
2889WindowsWindows XP tippek13. rész
2909WindowsTippek Windows XP-hez14. rész
2919WindowsWindows tippek15. rész
2924WindowsWindows tippek16. rész
2963WindowsWindows tippek17. rész
2973WindowsWindows Tippek18. rész
2981WindowsWindows tippek19. rész
2990WindowsTippek-trükkök20. rész
3027WindowsIIS tippek21. rész
3034WindowsWindows XP tippek-trükkök22. rész
3088WindowsWindows 2000/XP tippek, trükkök23. rész
3133WindowsWindows XP tippcsokor24. rész
3140WindowsWindows XP tippek, trükkök25. rész
3152WindowsXP és IIS tippek - trükkök26. rész
3158WindowsWindows XP tippek, trükkök27. rész
3168WindowsTippek, trükkök28. rész
3170WindowsRegistry trükkök29. rész
3179WindowsTippek, trükkök30. rész
3197WindowsWindows XP tippek, trükkök31. rész
3205WindowsTippek, trükkök32. rész
3214WindowsTippek, trükkök33. rész
3223WindowsTippek, trükkök34. rész
3233WindowsTippek, trükkök35. rész
3271WindowsTippek, trükkök36. rész
3307WindowsTippek, trükkök37. rész
3370WindowsTippek, trükkök38. rész
3399WindowsTippek, trükkök39. rész
3510WindowsTippek, trükkök40. rész
3611WindowsHardverrel kapcsolatos tippek, trükkök41. rész
3668WindowsRegistry trükkök42. rész
3711WindowsTippek, trükkök43. rész
3771WindowsTippek, trükkök44. rész
3801WindowsTippek, trükkök45. rész
3831WindowsTippek, trükkök46. rész
3891WindowsTippek, trükkök47. rész
3921WindowsTippek, trükkök48. rész
3981WindowsTippek, trükkök49. rész
4041WindowsTippek, trükkök50. rész
4071WindowsTippek, trükkök51. rész
4151WindowsTippek, trükkök52. rész
4171C#Tippek, trükkök53. rész
4211WindowsTippek, trükkök54. rész
4251WindowsTippek, trükkök55. rész
4281WindowsTippek, trükkök56. rész
3589DelphiTippek, trükkök57. rész
3718DelphiTippek, trükkök58. rész


Könyv
Ez a cikk megtalálható ebben a könyvben: Delphi Software Offline 2000 évkönyv 527. oldal

Felhasználási feltételek
A Software Online szoftverfejlesztői magazin mindegyik cikke, minden megjelent képe, és egyéb publikált anyaga szerzői jog védelme alatt áll! Bármilyen formában történő másodlagos terjesztésük, közzétételük vagy felhasználásuk kizárólag a kiadó előzetes írásbeli engedélyével történhet!

Copyright © 1999-2012 Animare Software Kft. Minden jog fenntartva!
| Készült: Animare Stúdió | Adatvédelem | Kapcsolat |