Löschen nicht verwendeter SqlDependency-Objekte in SQL Server, um Speicher freizugeben
Die Speicherverwaltung von SQL Server kann durch eine Ansammlung abgelaufener SqlDependency
Objekte beeinträchtigt werden, was zu einem hohen Speicherverbrauch durch den SQL Server-Prozess führt. Dies kann die Leistung erheblich beeinträchtigen, insbesondere in SQL Server Express, und möglicherweise zu Speichererschöpfung und Anwendungsabstürzen führen.
Das Problem angehen
Das Kernproblem liegt in der Funktionsweise von Microsoft SqlDependency
. Auch nach dem Aufruf von SqlDependency.Stop()
und der Freigabe von Verweisen auf SqlCommand
und SqlConnection
bleiben zugehörige Konversationsgruppen und Endpunkte in der Datenbank bestehen und verbrauchen weiterhin Speicher.
Bereinigungsschritte
Um diese ungenutzten Konversationsendpunkte zu entfernen und Speicher freizugeben, führen Sie dieses SQL-Skript für Ihre Datenbank aus:
<code class="language-sql">-- Disable conversation groups without index to prepare for endpoint deletion ALTER DATABASE <database_name> SET ALLOW_CONVERSATIONS_WITHOUT_INDEX = OFF; GO -- Remove disconnected conversation endpoints DELETE CEP FROM sys.conversation_endpoints CEP WHERE CEP.state = 'DI' OR CEP.state = 'CD'; GO -- Re-enable conversation groups ALTER DATABASE <database_name> SET ALLOW_CONVERSATIONS_WITHOUT_INDEX = ON; GO</code>
SqlDependency-Einschränkungen
Es ist wichtig zu beachten, dass SqlDependency
Einschränkungen unterliegt. Es werden nicht immer alle Tabellenänderungen erkannt, insbesondere beim erneuten Abonnieren.
Ein besserer Ansatz: SqlDependencyEx
Für eine robustere Änderungsverfolgung sollten Sie die Open-Source-Alternative SqlDependencyEx
in Betracht ziehen. Diese Bibliothek verwendet Datenbank-Trigger und Service Broker-Benachrichtigungen für eine zuverlässigere Überwachung von Tabellenänderungen. Hier ist ein Beispiel:
<code class="language-csharp">int changesReceived = 0; using (SqlDependencyEx sqlDependency = new SqlDependencyEx( TEST_CONNECTION_STRING, TEST_DATABASE_NAME, TEST_TABLE_NAME)) { sqlDependency.TableChanged += (o, e) => changesReceived++; sqlDependency.Start(); // Simulate table changes MakeTableInsertDeleteChanges(changesCount); // Allow time for changes to be processed Thread.Sleep(1000); } Assert.AreEqual(changesCount, changesReceived);</code>
Das obige ist der detaillierte Inhalt vonWie kann ich abgelaufene SqlDependency-Objekte bereinigen, um SQL Server-Speicher freizugeben?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!