Содержание |
DotNetNuke (далее DNN) по умолчанию ведёт несколько логов своей деятельности в БД:
К сожалению, это всё не проходит бесследно. Из негативных моментов такой работы следует отметить:
Также отметим, что в этих таблицах нет прямой необходимости. Статистика посещений сайта также доступна со стороны хостинга, посмотрите статью по статистике. Таблица ошибок не нужна, если ваш сайт работает исправно и не требует отладки или вмешательства программистов.
Против ведения таблицы SiteLog выступает и регламент хостинга, пункт 2.5.b как раз про это.
После отключения логов вам необходимо очистить таблицы от имеющихся в них данных. Не вздумайте исполнять запрос типа DELETE FROM SiteLog – в условиях удаления миллионов строк это создаст массу проблем, так как для каждой операции удаления строки будет создана запись в логе транзакций. Используйте метод очистки таблиц MSSQL без ведения лога транзакций – TRUNCATE TABLE, например:
TRUNCATE TABLE SiteLog
Запрос исполняется мгновенно и не вызывает проблем перегрузки.
После того, как в списке не остаётся ни одного события True, нажимаем Возврат, возвращаемся на входную страницу прсмотра журнала. Внизу списка появляется Очистить журнал (Clear Event Log) - пользоваться этой функцией не надо, смотрите выше пример использования TRUNCATE TABLE.
Внимание! Это решение зависит от версии DNN и может не дать эффекта. Проверяйте результат.
Суть проблемы в том, что внутри этой процедуры исполняется SELECT с двумя параметрами по множеству связанных таблиц. При это для таблиц проставлены внешние ключи и, предпологается, что это должно решить проблему. Однако это не так. Внутренний запрос примерно выглядит так (лишние поля убраны для ясности):
SELECT P.[ThreadID], P.[PostID], ... P.[DateApproved], F.PublicView FROM dbo.Forum_Posts P INNER JOIN dbo.Forum_Threads T ON P.ThreadID = T.ThreadID INNER JOIN dbo.Forum_Forums F ON T.ForumID = F.ForumID INNER JOIN dbo.Forum_Groups G ON F.GroupID = G.GroupID WHERE G.ModuleID = 123 AND P.[CreatedDate] >= '2008-12-05 16:41:39:467'
Для всех полей из FROM объединённых через INNER JOIN стоят внешние ключи. По полю Forum_Groups.ModuleID также есть индекс. По полю Forum_Posts.CreatedDate индекса нет (!). Казалось бы - достаточно создать индекс по полю CreatedDate для эффективной выборки по нему. К сожалению это не действует, так как основным индексом в данном случае SQL Server выбирает внешний ключ по полю Forum_Posts.ThreadID. Решение - создать внутренний (обычный) индекс по полю Forum_Posts.ThreadID и сделать его составным добавив вторым полем в индексе Forum_Posts.CreatedDate - в этом случае новый индекс сможет успешно заменить собой внешний ключ и SQL Server сможет использовать индекс пополю Forum_Posts.CreatedDate. Ниже код создания индекса:
CREATE NONCLUSTERED INDEX [IX_ThreadID_CreatedDate] ON [dbo].[Forum_Posts] ( [ThreadID] ASC, [CreatedDate] ASC ) WITH ( PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, SORT_IN_TEMPDB = OFF, IGNORE_DUP_KEY = OFF, DROP_EXISTING = OFF, ONLINE = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON ) ON [PRIMARY]
После этого потребление ресурсов на исполнение запроса снижается в 30 раз (на тестируемом DNN).
Если вы уже занимались проблемой оптимизации работы DNN с базой данных и выполняли действия из этой статьи или действия другого типа – напишите нам или прямо сюда сценарий решения проблем. Это поможет множеству людей и увеличит сумму счастья на планете.