Zmiana klucza podstawowego z TOŻSAMOŚCI na utrwalony Kolumna obliczana za pomocą COALESCE


10

Próbując oddzielić aplikację od naszej monolitycznej bazy danych, próbowaliśmy zmienić kolumny INT IDENTITY w różnych tabelach na kolumnę obliczeniową PERSISTED, która używa COALESCE. Zasadniczo potrzebujemy odsprzęgniętej aplikacji możliwości ciągłej aktualizacji bazy danych wspólnych danych dla wielu aplikacji, jednocześnie umożliwiając istniejącym aplikacjom tworzenie danych w tych tabelach bez potrzeby modyfikacji kodu lub procedury.

Zasadniczo przeszliśmy od definicji kolumny;

PkId INT IDENTITY(1,1) PRIMARY KEY

do;

PkId AS AS COALESCE(old_id, external_id, new_id) PERSISTED NOT NULL,
old_id INT NULL, -- Values here are from existing records of PkId before table change
external_id INT NULL,
new_id INT IDENTITY(2000000,1) NOT NULL

We wszystkich przypadkach PkId jest również KLUCZEM PODSTAWOWYM, a we wszystkich przypadkach oprócz jednego jest KLASTROWANY. Wszystkie tabele mają takie same klucze obce i indeksy jak poprzednio. Zasadniczo nowy format pozwala na dostarczenie PkId przez oddzieloną aplikację (jako external_id), ale także pozwala, aby PkId był wartością kolumny TOŻSAMOŚCI, umożliwiając w ten sposób istniejący kod, który opiera się na kolumnie TOŻSAMOŚĆ dzięki użyciu SCOPE_IDENTITY i @@ IDENTITY pracować tak jak kiedyś.

Problem, który mieliśmy, polegał na tym, że natrafiliśmy na kilka zapytań, które kiedyś działały w akceptowalnym czasie, aby teraz całkowicie wysadzić w powietrze. Wygenerowane plany zapytań używane przez te zapytania nie przypominają tego, czym były wcześniej.

Biorąc pod uwagę, że nowa kolumna jest KLUCZEM PIERWOTNYM, tym samym typem danych co poprzednio, i TRWAŁĄ, oczekiwałbym, że zapytania i plany zapytań będą zachowywać się tak samo jak wcześniej. Czy ZLICZONY PERSISTED INT PkId powinien zasadniczo zachowywać się tak samo, jak jawna definicja INT pod względem sposobu, w jaki SQL Server wygeneruje plan wykonania? Czy są inne prawdopodobne problemy z tym podejściem, które możesz zobaczyć?

Celem tej zmiany było umożliwienie nam zmiany definicji tabeli bez potrzeby modyfikowania istniejących procedur i kodu. Biorąc pod uwagę te problemy, nie wydaje mi się, abyśmy mogli zastosować to podejście.


Komentarze nie są przeznaczone do rozszerzonej dyskusji; ta rozmowa została przeniesiona do czatu .
Paul White 9

Odpowiedzi:


4

PIERWSZY

Prawdopodobnie nie trzeba wszystkie trzy kolumny: old_id, external_id, new_id. new_idKolumna, bycia IDENTITY, będzie mieć nową wartość wygenerowaną dla każdego wiersza, nawet po włożeniu do external_id. Ale pomiędzy old_idi external_idsą one prawie wzajemnie wykluczające: albo istnieje już old_idwartość, albo ta kolumna, w obecnej koncepcji, będzie tylko, NULLjeśli użyjesz external_idlub new_id. Ponieważ nie dodasz nowego „zewnętrznego” identyfikatora do już istniejącego wiersza (tj. Takiego, który ma old_idwartość), i nie będzie żadnych nowych wartości old_id, więc może być używana jedna kolumna do obu celów.

Pozbądź się external_idkolumny i zmień nazwę, old_idaby była czymś podobnym old_or_external_idlub czymkolwiek. Nie powinno to wymagać żadnych rzeczywistych zmian w niczym, ale zmniejsza pewne komplikacje. Konieczne może być wywołanie kolumny external_id, nawet jeśli zawiera ona „stare” wartości, jeśli kod aplikacji jest już zapisany do wstawienia external_id.

Dzięki temu nowa struktura jest po prostu:

PkId AS AS COALESCE(old_or_external_id, new_id, -1) PERSISTED NOT NULL,
old_or_external_id INT NULL, -- values from existing record OR passed in from app
new_id INT IDENTITY(2000000, 1) NOT NULL

Teraz dodano tylko 8 bajtów na wiersz zamiast 12 bajtów (zakładając, że nie używasz SPARSEopcji ani kompresji danych). I nie trzeba było zmieniać żadnego kodu, T-SQL ani kodu aplikacji.

DRUGA

Kontynuując tę ​​ścieżkę uproszczenia, spójrzmy na to, co nam pozostało:

  • old_or_external_idKolumna albo ma już wartości, lub będą miały nową wartość z aplikacji, lub będą lewo NULL.
  • new_idZawsze będzie mieć nową wartość wygenerowaną, ale wartość ta będzie używana tylko wtedy, gdy old_or_external_idkolumna jest NULL.

Nigdy nie ma czasu, kiedy potrzebujesz wartości zarówno w, jak old_or_external_idi w new_id. Tak, będą chwile, gdy obie kolumny będą miały wartości z powodu new_idbycia IDENTITY, ale te new_idwartości są ignorowane. Ponownie te dwa pola wzajemnie się wykluczają. Co teraz?

Teraz możemy zastanowić się, dlaczego tak naprawdę potrzebowaliśmy external_id. Biorąc pod uwagę, że można wstawić do IDENTITYkolumny za pomocą SET IDENTITY_INSERT {table_name} ON;, możesz w ogóle nie wprowadzać żadnych zmian schematu, a jedynie modyfikować kod aplikacji, aby zawijać INSERTinstrukcje / operacje SET IDENTITY_INSERT {table_name} ON;i SET IDENTITY_INSERT {table_name} OFF;instrukcje. Następnie należy określić, do jakiego zakresu początkowego należy zresetować IDENTITYkolumnę (dla nowo wygenerowanych wartości), ponieważ będzie ona musiała znajdować się znacznie powyżej wartości, które wstawi kod aplikacji, ponieważ wstawienie wyższej wartości spowoduje, że następna automatycznie wygenerowana wartość będzie być większa niż bieżąca wartość MAX. Ale zawsze możesz wstawić wartość poniżej wartości IDENT_CURRENT .

Łączenie kolumn old_or_external_idi new_idnie zwiększa również szansy na wystąpienie nakładającej się wartości między wartościami generowanymi automatycznie i wartościami generowanymi przez aplikację, ponieważ celem uzyskania 2, a nawet 3 kolumn jest połączenie ich w wartość klucza podstawowego, i są to zawsze unikalne wartości.

W tym podejściu wystarczy:

  • Pozostaw tabele jako:

    PkId INT IDENTITY(1,1) PRIMARY KEY

    Dodaje to 0 bajtów do każdego wiersza zamiast 8, a nawet 12.

  • Określ zakres początkowy dla wartości generowanych przez aplikację. Będą one wyższe niż bieżąca wartość MAX w każdej tabeli, ale mniejsza niż to, co stanie się wartością minimalną dla automatycznie generowanych wartości.
  • Określ wartość, od której powinien zacząć się automatycznie wygenerowany zakres. Pomiędzy obecną wartością MAX a miejscem na wzrost powinno być dużo miejsca , wiedząc, że górna granica wynosi nieco ponad 2,14 miliarda. Następnie możesz ustawić tę nową minimalną wartość początkową za pomocą DBCC CHECKIDENT .
  • Zawiń kod aplikacji WSTAWIANIE SET IDENTITY_INSERT {table_name} ON;i SET IDENTITY_INSERT {table_name} OFF;instrukcje.

DRUGA, część B

Odmianą podejścia opisanego bezpośrednio powyżej byłoby wprowadzenie wartości wstawiania kodu aplikacji zaczynających się od -1, a następnie zmniejszających się . To pozostawia IDENTITYwartości jako jedyne idące w górę . Korzyścią jest to, że nie tylko nie komplikujesz schematu, ale także nie musisz się martwić, że natrafisz na nakładające się identyfikatory (jeśli wartości wygenerowane przez aplikację trafią do nowego automatycznie wygenerowanego zakresu). Jest to opcja tylko wtedy, gdy nie używasz już ujemnych wartości identyfikatora (i wydaje się, że ludzie rzadko używają wartości ujemnych w automatycznie generowanych kolumnach, więc w większości sytuacji powinno to być prawdopodobne).

W tym podejściu wystarczy:

  • Pozostaw tabele jako:

    PkId INT IDENTITY(1,1) PRIMARY KEY

    Dodaje to 0 bajtów do każdego wiersza zamiast 8, a nawet 12.

  • Zakres początkowy dla wartości generowanych przez aplikację będzie wynosił -1.
  • Zawiń kod aplikacji WSTAWIANIE SET IDENTITY_INSERT {table_name} ON;i SET IDENTITY_INSERT {table_name} OFF;instrukcje.

Tutaj nadal musisz to zrobić IDENTITY_INSERT, ale: nie dodajesz żadnych nowych kolumn, nie musisz „ponownie wysyłać” żadnych IDENTITYkolumn i nie ma w przyszłości ryzyka nakładania się.

DRUGA, część 3

Ostatnią odmianą tego podejścia byłoby ewentualnie zamiana IDENTITYkolumn i zamiast tego użycie Sekwencji . Powodem takiego podejścia jest możliwość wprowadzenia wartości wstawiania kodu aplikacji, które są: dodatnie, powyżej zakresu generowanego automatycznie (nie poniżej) i nie ma takiej potrzeby SET IDENTITY_INSERT ON / OFF.

W tym podejściu wystarczy:

  • Twórz sekwencje za pomocą CREATE SEQUENCE
  • Skopiuj IDENTITYkolumnę do nowej kolumny, która nie ma IDENTITYwłaściwości, ale ma DEFAULTOgraniczenie za pomocą funkcji NASTĘPNA WARTOŚĆ DLA :

    PkId INT PRIMARY KEY CONSTRAINT [DF_TableName_NextID] DEFAULT (NEXT VALUE FOR...)

    Dodaje to 0 bajtów do każdego wiersza zamiast 8, a nawet 12.

  • Zakres początkowy dla wartości generowanych przez aplikację będzie znacznie powyżej tego, co Twoim zdaniem zbliży się do wartości wygenerowanych automatycznie.
  • Zawiń kod aplikacji WSTAWIANIE SET IDENTITY_INSERT {table_name} ON;i SET IDENTITY_INSERT {table_name} OFF;instrukcje.

JEDNAK , ze względu na wymaganie, aby kod z jednym SCOPE_IDENTITY()lub @@IDENTITYnadal działał poprawnie, przełączanie na Sekwencje nie jest obecnie opcją, ponieważ wydaje się, że nie ma odpowiednika tych funkcji dla Sekwencji :-(. Smutne!


Dziękuję bardzo za odpowiedź. Podnosisz kilka punktów, które zostały tu omówione wewnętrznie. Niestety niektóre z nich nie będą dla nas działać z kilku powodów. Nasza baza danych jest dość stara i nieco krucha i działa w trybie zgodności z 2005 r., Więc SEQUENCES jest wyłączony. Wypychanie danych z naszej aplikacji odbywa się za pomocą narzędzia do ładowania danych, które uzyskuje nowe rekordy z kolejek brokera usług i przepycha je przez wiele wątków. IDENTITY_INSERT może być użyty tylko dla jednego stołu na sesję, a obecne myślenie jest takie, że nasza architektura nie może tego zrobić bez znaczących zmian. Testuję teraz twoją propozycję pięści.
Pan Moose

@MrMoose Tak, zaktualizowałem swoją odpowiedź, aby na końcu zamieścić więcej informacji o Sekwencjach. W każdym razie to nie zadziałałoby. Zastanawiałem się nad potencjalnymi problemami z współbieżnością IDENTITY_INSERT, ale nie przetestowałem tego. Nie jestem pewien, czy opcja nr 1 rozwiąże ogólny problem, była to tylko obserwacja mająca na celu zmniejszenie niepotrzebnej złożoności. Jeśli jednak masz wiele wątków wstawiających nowe „zewnętrzne” identyfikatory, jak możesz zagwarantować, że są one unikalne?
Solomon Rutzky

@MrMoose Właściwie, w odniesieniu do „ IDENTITY_INSERT można użyć tylko dla jednej tabeli na sesję ”, jaki dokładnie jest tutaj problem? 1) możesz wstawiać tylko do jednej tabeli na raz, więc wyłączasz ją dla Tabeli A przed wstawieniem do Tabeli B, i 2) Właśnie przetestowałem i wbrew temu, co myślałem, nie ma problemów z współbieżnością - byłem w stanie mieć IDENTITY_INSERT ONna ten sam stół w dwóch sesjach i wkładał się do obu bez problemu.
Solomon Rutzky

1
Jak sugerowałeś, zmiana 1 nie miała większego znaczenia. Identyfikator, którego będziemy używać, zostanie przydzielony poza bieżącą bazą danych i użyty do powiązania zapisów. Może się zdarzyć, że moje rozumienie sesji nie jest w porządku, więc IDENTITY_INSERT może działać. Zajmie mi to trochę czasu, aby to zbadać, więc nie będę mógł przez chwilę się zgłosić. Jeszcze raz dziękuję za wkład. To jest bardzo cenione.
Mr Moose

1
Myślę, że Twoja sugestia użycia IDENTITY_INSERT (z wysoką wartością początkową dla istniejących aplikacji) będzie dobrze działać. Aaron Bertrand udzielił tutaj odpowiedzi z dobrym, dobrym przykładem na testowanie go z wykorzystaniem współbieżności. Zmodyfikowaliśmy nasze narzędzie do ładowania danych, aby móc obsługiwać tabele, które muszą określać wartości tożsamości, i przejdziemy do dalszych testów w nadchodzących tygodniach.
Pan Moose
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.