DBCCSHRINKFILE (Transact-SQL)
Сокращает размер указанного файла данных или журнала для текущей базы данных или освобождает файл, перемещая данные из указанного файла в другие файлы из той же файловой группы, разрешая удаление файла из базы данных. Можно сжать файл до размера, который будет меньше, чем размер, указанный во время его создания. В результате будет установлено новое значение минимального размера файла.
Синтаксис
DBCC SHRINKFILE
(
{ file_name | file_id }
{ [, EMPTYFILE ]
| [ [, target_size ] [, { NOTRUNCATE | TRUNCATEONLY } ] ]
}
)
[ WITH NO_INFOMSGS ]
Аргументы
file_name
Логическое имя файла, предназначенного для сжатия.
file_id
Идентификационный номер (идентификатор) файла, предназначенного для сжатия. Для получения идентификатора файла используйте функцию FILE_IDEX или выполните запрос к представлению каталога sys.database_files текущей базы данных.
target_size
Размер файла (в мегабайтах), выражаемый целым числом. Если он не указан, то инструкция DBCCSHRINKFILE уменьшает файл до размера файла по умолчанию. Размер по умолчанию представляет собой размер, указанный в момент создания файла.
Примечание
Можно уменьшить размер пустого файла, заданный по умолчанию, с помощью инструкции DBCCSHRINKFILEtarget_size. Например, при создании файла с размером 5 МБ и последующем уменьшении размера до 3 МБ, в то время как файл остается пустым, размер файла по умолчанию задается равным 3 МБ. Это правило применимо только к пустым файлам, в которых никогда не содержались данные.
Этот параметр не поддерживается для контейнеров файловых групп FILESTREAM.
Если аргумент target_size указан, то инструкция DBCCSHRINKFILE пытается сжать файл до заданного размера. Используемые страницы в освобождаемой части файла перемещаются в свободное место сохраняемой части файла. Например, если размер файла данных составляет 10 МБ, инструкция DBCCSHRINKFILE со значением аргумента target_size, равным 8, перемещает все страницы, используемые в последних 2 МБ файла, на место любых нераспределенных страниц в первых 8 МБ файла. Инструкция DBCCSHRINKFILE не сжимает файл до меньшего размера, чем требуется для хранения данных в файле. Например, если для файла данных, размер которого составляет 10 МБ, необходимо сжатие до 7 МБ, инструкция DBCCSHRINKFILE со значением аргумента target_size, равным 6, сжимает файл до размера в 7 МБ, а не 6 МБ.
|
EMPTYFILE
Выполняет миграцию всех данных из указанного файла в другие файлы в той же файловой группе. Поскольку компонент Компонент DatabaseEngine больше не разрешает размещать данные в пустом файле, этот файл может быть удален инструкцией ALTERDATABASE.
Для контейнеров файловых групп FILESTREAM файл нельзя удалить с помощью ALTERDATABASE до тех пор, пока сборщик мусора FILESTREAM не выполнит и не удалит все ненужные файлы контейнеров файловых групп, которые были скопированы в другой контейнер с помощью EMPTYFILE. Дополнительные сведения см. в разделе sp_filestream_force_garbage_collection (Transact-SQL).
Примечание |
Сведения об удалении контейнера FILESTREAM см. в соответствующем разделе в Параметры инструкции ALTERDATABASE для файлов и файловых групп (Transact-SQL). |
NOTRUNCATE
Перемещает распределенные страницы из конца файла на место нераспределенных страниц в начале файла данных с параметром target_percent или без него. Свободное место в конце файла операционной системе не возвращается, и физический размер файла не изменяется. Следовательно, если указан аргумент NOTRUNCATE, файл сжимается незначительно.
|
Аргумент NOTRUNCATE применим только к файлам данных. На файлы журнала он не влияет.
Этот параметр не поддерживается для контейнеров файловых групп FILESTREAM.
TRUNCATEONLY
Освобождает все свободное пространство в конце файла операционной системе, но не перемещает страницы внутри файла. Файл данных сокращается только до последнего выделенного экстента.
Аргумент target_size не обрабатывается, если указан аргумент TRUNCATEONLY.
Параметр TRUNCATEONLY не переносит сведения в журнале, но удаляет неактивные VLF в конце файла журнала. Этот параметр не поддерживается для контейнеров файловых групп FILESTREAM.
WITHNO_INFOMSGS
Подавляет вывод всех информационных сообщений.
Результирующие наборы
В следующей таблице отображены столбцы результирующего набора.
Имя столбца | Описание |
DbId | Идентификатор базы данных, файл которой компонент Компонент DatabaseEngine пытался сжать. |
FileId | Идентификационный номер файла, сжатие которого было предпринято компонентом Компонент DatabaseEngine. |
CurrentSize | Количество 8-килобайтных страниц, занятых файлом в настоящее время. |
MinimumSize | Минимальное количество 8-килобайтных страниц, которое может занимать файл. Оно соответствует минимальному размеру или размеру файла, указанному при создании. |
UsedPages | Количество 8-килобайтных страниц, используемых файлом в настоящее время. |
EstimatedPages | Количество 8-килобайтных страниц, до которого можно было бы сжать файл по оценке компонента Компонент DatabaseEngine. |
Замечания
Инструкция DBCCSHRINKFILE применяется к файлам в текущей базе данных. Дополнительные сведения об изменении текущей базы данных см. в разделе USE (Transact-SQL).
|
Операции DBCCSHRINKFILE могут быть остановлены на любом этапе процесса, при этом вся выполненная работа сохраняется.
В случае сбоя операции DBCCSHRINKFILE возникает ошибка.
Сжимаемая база данных необязательно должна находиться в однопользовательском режиме; при выполнении сжатия файла в базе данных могут работать другие пользователи. Для сжатия системных баз данных также не обязательно запускать экземпляр SQLServer в однопользовательском режиме.
Сжатие файла журнала
Для файла журнала компонент DatabaseEngine использует аргумент target_size для вычисления целевого размера всего журнала. Поэтому аргумент target_size является количеством свободного пространства в журнале после операции сжатия. Затем по заданному размеру всего журнала рассчитываются заданные размеры каждого файла журнала. Инструкция DBCCSHRINKFILE сразу же пытается сжать каждый физический файл журнала до намеченного размера. Однако если часть логического журнала хранится в виртуальных журналах за пределами заданного размера, то компонент DatabaseEngine освобождает как можно больше места, а затем формирует информационное сообщение. Сообщение описывает действия, которые необходимо предпринять, чтобы переместить логический журнал из виртуальных журналов в конец файла. После выполнения всех действий инструкция DBCCSHRINKFILE может быть использована для освобождения оставшегося пространства.
Так как файл журнала может быть сжат только до границы виртуального файла журнала, сжатие файла журнала к размеру, меньшему, чем размер виртуального файла журнала, невозможно, даже если он не используется. Размер виртуального файла журнала динамически выбирается компонентом DatabaseEngine при создании или расширении файлов журнала.
Рекомендации
Примите во внимание следующие сведения при планировании сжатия файла.
- Наибольший эффект от операции сжатия достигается при ее применении после операции, создающей много неиспользуемого пространства, например после усечения таблицы или удаления таблицы.
- Большинству баз данных требуется некоторое свободное пространство для выполнения обычных ежедневных операций. Если сжатие базы данных производится регулярно, но она снова увеличивается в размерах, это означает, что место, освобожденное при сжатии, необходимо для нормальной работы. В таких случаях повторное сжатие базы данных бессмысленно.
- Операция сжатия не избавляет от фрагментации индексов в базе данных и обычно приводит к еще более сильной фрагментации. Это еще одна причина, по которой не стоит выполнять регулярное сжатие базы данных.
- Сжимайте несколько файлов в одной базе данных последовательно, а не одновременно. Состязание в системных таблицах может привести к задержке из-за блокировки.
Устранение неполадок
Этот раздел описывает методы диагностики и устранения проблем, которые могут произойти при выполнении команды DBCCSHRINKFILE:
Файл не удалось сжать
Если операция сжатия выполняется без ошибки, но файл не изменяется в размере, убедитесь, что он имеет свободное пространство для удаления, выполнив одну из следующих операций.
- Выполните следующий запрос.
· SELECT name,size/128.0 - CAST(FILEPROPERTY(name, 'SpaceUsed') AS int)/128.0 AS AvailableSpaceInMB
· FROM sys.database_files;
- Выполните команду DBCCSQLPERF, чтобы освободить пространство, используемое журналом транзакций.
Если свободного пространства недостаточно, операция сжатия не сможет уменьшить размер файла в дальнейшем.
Обычно это файл журнала, который сжимается незначительно. Это характерно для файла журнала, который не был усечен. Можно усечь файл журнала, установив значение SIMPLE для модели восстановления базы данных или создав резервную копию журнала, а затем выполнив операцию DBCCSHRINKFILE снова.
Операция сжатия заблокирована
Операции сжатия могут быть блокированы транзакцией, запущенной с уровнем изоляции, основанном на управлении версиями строк. Например, если при выполнении масштабной операции удаления с уровнем изоляции, основанном на управлении версиями строк, выполнить инструкцию DBCCSHRINKDATABASE, то, прежде чем приступить к сжатию файлов, она будет ожидать завершения операции удаления. В этом случае операции DBCCSHRINKFILE и DBCCSHRINKDATABASE выводят информационное сообщение (5202 для SHRINKDATABASE и 5203 для SHRINKFILE) в журнал ошибок SQLServer каждые 5 минут в течение первого часа, а затем по одному сообщению каждый час. Например, если журнал ошибок содержит следующее сообщение об ошибке, произойдет следующая ошибка.
DBCC SHRINKFILE for file ID 1 is waiting for the snapshot
transaction with timestamp 15 and other snapshot transactions linked to
timestamp 15 or with timestamps older than 109 to finish.
Это означает, что операция сжатия блокируется транзакциями моментального снимка, которые имеют отметки времени старше, чем метка 109, представляющая последнюю транзакцию, завершающую операцию сжатия. Это также показывает, что столбцы transaction_sequence_num или first_snapshot_sequence_num в динамическом представлении управления sys.dm_tran_active_snapshot_database_transactions содержат значение 15. Если столбцы transaction_sequence_num или first_snapshot_sequence_num в представлении содержат меньшее число, чем последняя транзакция, выполненная операцией сжатия (109), то операция сжатия будет ждать завершения этих транзакций.
Разрешить эту проблему можно одним из следующих способов.
- Прервите выполнение транзакции, блокирующей операцию сжатия.
- Прервите операцию сжатия. Если операция сжатия прервана, любая завершенная работа сохранена.
- Пока операция сжатия ожидает завершения блокирующей транзакции, ничего делать не нужно.
Разрешения
Необходимо быть членом предопределенной роли сервера sysadmin или предопределенной роли базы данных db_owner.
Примеры