Dlaczego dziennik transakcji nadal rośnie w trybie prostego odzyskiwania z nocnymi kopiami zapasowymi


24

Przed natychmiastowym oznaczeniem jako duplikat przeczytałem artykuł Mike'a Walsha: Dlaczego dziennik transakcji wciąż rośnie lub brakuje miejsca? , ale nie sądzę, że dało to odpowiedź na moją sytuację. Przejrzałem kilkanaście podobnych pytań, ale najważniejsze z nich po prostu powiedziały „duplikuj” i wskazały na pytanie Mike'a.

Szczegóły: Mam kilka ~ 500 MB baz danych na SQL Server 2008 R2, wszystkie w trybie odzyskiwania SIMPLE (nie mój wybór), nocne kopie zapasowe z ~ 200 MB plików danych i ~ 300 MB plików dziennika. Dziennik nie rośnie do 300 MB od razu, ale raczej powoli w ciągu kilku miesięcy. Żadna z nich nie zawiera otwartych transakcji, przynajmniej według sp_who2 i monitora aktywności. Jeśli kliknę bazę danych prawym przyciskiem myszy i wybiorę właściwości, wyświetli się informacja, że ​​~ 50 MB jest za darmo. Czy cały dziennik powinien być wolny, szczególnie tuż po utworzeniu kopii zapasowej? Czy w trybie PROSTYM dziennik nie powinien być wolny, dopóki nie ma otwartej transakcji?

log_reuse_wait_descfrom sys.databasesmówi mówi „NIC”, co na podstawie powyższego pytania i odpowiedzi mówi, że nie powinno czekać na nic, aby ponownie wykorzystać przestrzeń.

Jeśli zrobię „DBCC SHRINKFILE”, plik dziennika zmniejszy się do 1 MB, więc jest gotowy odzyskać miejsce. Mogę skonfigurować coś, co zmniejsza dzienniki co tydzień i nie pozwala mi wymknąć się spod kontroli, ale jestem zdezorientowany, dlaczego SQL Server kazałby mi to zrobić.

Rozumiem, czy była jakaś szalona transakcja, która wymagała 300 MB, aby ją zalogować, ale nie robimy nic ekstremalnego, tylko podstawowy OLTP. Z pytania / odpowiedzi Mike'a:

Prosty model odzyskiwania - więc dzięki powyższemu wstępowi najłatwiej jest mówić o prostym modelu odzyskiwania. W tym modelu mówisz SQL Serverowi - nic mi nie jest, gdy używasz pliku dziennika transakcji do przywracania po awarii i wznawiasz odzyskiwanie (naprawdę nie masz wyboru. Poszukaj właściwości ACID, a to powinno mieć sens szybko), ale kiedy już nie dłużej potrzebujesz go do tego celu odzyskiwania po awarii / zrestartowaniu, śmiało i ponownie użyj pliku dziennika.

SQL Server nasłuchuje tego żądania w prostym odzyskiwaniu i zachowuje tylko informacje potrzebne do przywrócenia awarii / ponownego uruchomienia. Po upewnieniu się, że SQL Server może odzyskać, ponieważ dane są zahartowane w pliku danych (mniej więcej), dane, które zostały zahartowane, nie są już potrzebne w dzienniku i są oznaczone do obcięcia - co oznacza, że ​​zostaną ponownie użyte.

Ciągle mówi, że przestrzeń na dzienniki powinna zostać ponownie wykorzystana, ale przy tak powolnym wzroście w ciągu miesięcy nie wydaje się, aby tak było.

czego mi brakuje? Czy coś powstrzymuje SQL Servera od rozpoznania danych jako „zahartowanych” i zwolnienia dziennika?

(edytuj) Raport After Action - AKA trochę wiedzy jest niebezpieczna

Po odkryciu, że jest to „popularne pytanie”, poczułem, że jestem winien wyjaśnienie tego, co wydarzyło się 7 miesięcy temu i czego nauczyłem się, jak mam nadzieję, ocalić innym ludziom smutek.

Po pierwsze, przestrzeń dostępna w SSMS podczas przeglądania właściwości w bazie danych to przestrzeń dostępna w pliku danych. Możesz to wyświetlić, uruchamiając następujące polecenie w bazie danych, a przekonasz się, że dostępne miejsce zgłoszone przez SSMS stanowi różnicę między FileSizeMB a UsedSpaceMB:

SELECT
    DB.name,
    MF.physical_name,
    MF.type_desc AS FileType,
    MF.size * 8 / 1024 AS FileSizeMB,
    fileproperty(MF.name, 'SpaceUsed') * 8/ 1024 AS UsedSpaceMB,
    mf.name LogicalName
FROM
    sys.master_files MF
    JOIN sys.databases DB ON DB.database_id = MF.database_id
WHERE   DB.name = 'yourdatabasename'

To potwierdziło, że w normalnych okolicznościach używaliśmy bardzo mało miejsca w dzienniku (20 MB lub mniej), ale prowadzi to do drugiego elementu ...

Po drugie, moje postrzeganie kłód rośnie powoli. Jednak w rzeczywistości dzienniki szybko rosły w nocy, gdy facet odpowiedzialny za stosowanie poprawek do tej aplikacji innej firmy nakładał łatki. Łatka została wykonana jako pojedyncza transakcja, więc w zależności od łaty 200 MB danych wymagało 300 MB dziennika. Kluczem do śledzenia tego było zapytanie Aarona Bertranda z https://sqlblog.org/2007/01/11/reviewing-autogrow-events-from-the-default-trace

DECLARE @path NVARCHAR(260);

SELECT 
    @path = REVERSE(SUBSTRING(REVERSE([path]), 
    CHARINDEX('\', REVERSE([path])), 260)) + N'log.trc'
FROM    sys.traces
WHERE   is_default = 1;

SELECT 
   DatabaseName,
   [FileName],
    SPID,
    Duration,
    StartTime,
    EndTime,
    FileType = CASE EventClass 
       WHEN 92 THEN 'Data'
       WHEN 93 THEN 'Log'
    END
FROM sys.fn_trace_gettable(@path, DEFAULT)
WHERE
    EventClass IN (92,93)
ORDER BY
    StartTime DESC;

To pokazało, że log rośnie w niektóre wieczory, kiedy klient nie korzysta z bazy danych. Doprowadziło to do rozmowy z facetem nakładającym łatki i odpowiedzi na tajemnicę.

Jeszcze raz dziękuję za osoby, które udzieliły mi pomocy w uzyskaniu odpowiedzi.


Właściwie sam zaobserwowałem podobne zjawisko w jednej z baz danych witryn moich klientów z modelem odzyskiwania SIMPLE. Dzienniki nie rosną (jeszcze, w każdym razie), ale używane przestrzeń w plikach dziennika ma wzrosnąć o odrobinę każdej nocy. I dzieje się tak, gdy działa kopia zapasowa bazy danych. Jeszcze nie zorientowałem się, co go powoduje, ani czy jest to problem, czy nie.
RBarryYoung

Odpowiedzi:


20

Nie możemy zgadnąć, co go powoduje, ale SQL Server nie tylko powiększa plik dziennika do 300 MB, ale do 300 MB, ponieważ w pewnym momencie od ostatniej operacji zmniejszania potrzebował tego dużo miejsca w dzienniku (czy to z powodu jakiejś dużej pojedynczej transakcji, czy też wielu mniejszych jednoczesnych transakcji). Musiałbyś śledzić zdarzenia wzrostu pliku dziennika (mówiłem o tym tutaj i tutaj ), aby spróbować zawęzić czas i dlaczego tak się dzieje (również jeśli ustawienie wzrostu pliku dziennika wynosi 300 MB lub coś, to wzrośnie o 300 MB gdy tylko zajmie więcej niż 1 MB miejsca, aby pomieścić aktywne transakcje).

W każdym razie, dlaczego uważasz, że musisz zmniejszyć plik dziennika, gdy osiągnie on 300 MB? Czy dokładnie przeczytałeś wszystkie odpowiedzi na pytanie Mike'a ? Plik dziennika NIE będzie sam się kurczył, ponieważ zmniejszenie pliku dziennika do 1 MB - aby mógł on ponownie wzrosnąć podczas największych transakcji - jest całkowitą stratą czasu. Co zamierzasz zrobić w międzyczasie z całym tym wolnym miejscem?


Nie sądzę, abyśmy robili coś, co wymagałoby 300 MB dziennika, ale nawet gdybyśmy to zrobili, to czy nie byłoby to widoczne jako wolne miejsce w bazie danych? Patrząc na właściwości bazy danych w SQL Server Management Studio, rozmiar jest rozmiarem danych i dziennika, i oczekiwałbym, że wolne miejsce będzie wolnym miejscem na dane i dziennik. Czy w dzienniku nie ma wolnego miejsca? To, że nie wyświetlało się jako wolne, wydawało się, że wciąż jest w użyciu, ale w bazie danych nie było żadnej aktywności.
DerekCate,

1
Nie, gdy to zrobisz, automatycznie nie stanie się wolnym miejscem. Zatwierdzone transakcje nie są zerowane, ich miejsce jest po prostu oznaczone jako dostępne do ponownego użycia.
Aaron Bertrand

1
@DerekCate „Nie sądzę, abyśmy robili coś, co wymagałoby 300 MB dziennika” ... Byłbyś zaskoczony, nie trzeba wiele wiązać z ponownym użyciem dziennika transakcji. Nie myśl o tym pod względem ilości pracy, ponieważ nie zawsze jest to przyczyną.
Thomas Stringer,

Ok, więc tylko dla potwierdzenia, że ​​rozumiem to poprawnie, dziennik 300 MB, nawet jeśli obecnie nie jest potrzebny, nie będzie wyświetlany jako wolne miejsce, ale zostanie ponownie wykorzystany. W pewnym momencie potrzebował 300 MB do obsługi niektórych transakcji. Ok, nowe rzeczy do obejrzenia. Dziękuję Ci!
DerekCate,

1
Kolejna rzecz do rozważenia: automatyczny punkt kontrolny dla prostej bazy danych odzyskiwania znajduje się w kolejce dopiero, gdy dziennik jest już w 70% zapełniony. W związku z tym możesz nie potrzebować tyle aktywności dziennika, aby spowodować wzrost, w zależności od czasu.
Paul White mówi GoFundMonica

6

Wszystkie twoje bieżące testy ( DBCC SHRINKFILE, log_reuse_wait_desc) po prostu dowodzą, że teraz ponownie odpowiednio używasz plików dziennika wirtualnego dziennika transakcji. Ale kiedy zdarzają się zdarzenia automatycznego wzrostu w pliku dziennika transakcji, jest to reakcja na niemożność ponownego wykorzystania dziennika.

Często nie jest to stan ciągły (dokładnie tak, jak ci się teraz wydaje, z braku objawów, które obecnie widzisz). Istnieje kilka rzeczy, które mogą powodować takie zachowanie, nawet w prostym modelu odzyskiwania.

Najlepszym rozwiązaniem byłoby skonfigurowanie zadania gromadzenia danych w celu rutynowego pobierania log_reuse_wait_descbazy danych i rutynowego rejestrowania tego gdzieś. Następnie powinieneś być w stanie dokonać inżynierii wstecznej, co powoduje brak ponownego użycia dziennika.

Ciągle mówi, że przestrzeń na dzienniki powinna zostać ponownie wykorzystana, ale przy tak powolnym wzroście w ciągu miesięcy nie wydaje się, aby tak było.

Jak wskazałem powyżej, zwykle nie jest to ciągły stan, który powoduje brak ponownego użycia dziennika transakcji (zapisz kilka przypadków narożnych, takich jak źle skonstruowane transakcje), więc musisz wskazać moment, w którym to się dzieje. Lekkie zbieranie danych diagnostycznych powinno być dobrym początkiem.


Jeśli widzi 50 MB wolnego miejsca i dziennik 300 MB, fn_DBLOG () dałby mu wgląd w to, co zwiększało rozmiar jego dziennika?
Kenneth Fisher

0

Czy masz włączone automatyczne zmniejszanie w bazach danych? I / lub czy masz zaplanowane plany konserwacji wykonujące przebudowy indeksu?

Automatyczne zmniejszanie, po którym następuje utrzymanie indeksu, spowoduje znaczną objętość dziennika.

Utrzymanie indeksu samo w sobie spowoduje także znaczną objętość dziennika, jeśli wystąpią problemy z projektem DB, takie jak indeksy klastrowe w identyfikatorach GUID.


Automatyczne zmniejszanie nie jest włączone w bazach danych, staramy się uniknąć fragmentacji indeksu brentozar.com/archive/2009/08/… . Mamy cotygodniowe kontrole integralności, ale nie sądzę, aby w ramach tego dochodziło do przebudowy indeksu, muszę to przeanalizować. Poza tym, bez identyfikatorów GUID, każda tabela ma kolumnę tożsamości, która jest kluczem podstawowym.
DerekCate

-3
 create table #dblog (Databasename varchar(100),
                     logsize float,
                     logspace%] float,
                     [Status] int)

 insert into #dblog
 EXEC ('DBCC sqlperf(logspace)') 

 select * from #dblog

 alter table #dblog 
 add [lgspace used GB] float;

 update #dblog 
 set [lgspace used GB ] = (logsize*[logspace%]/1024)

 update #dblog 
 set [logsize] =([logsize]/1024)

 alter table #dblog 
 drop column [Status];

 select * from #dblog

 drop table #dblog
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.