Visual Studio 2010 を使用して .NET 4.0 クライアント プロファイルをターゲットにしています。特定のプロセスの開始/終了を検出する C# クラスがあります。このために、クラスは以下のように初期化される ManagementEventWatcher を使用します。query
、scope
およびwatcher
クラス フィールドです。
query = new WqlEventQuery();
query.EventClassName = "__InstanceOperationEvent";
query.WithinInterval = new TimeSpan(0, 0, 1);
query.Condition = "TargetInstance ISA 'Win32_Process' AND TargetInstance.Name = 'notepad.exe'";
scope = new ManagementScope(@"\\.\root\CIMV2");
watcher = new ManagementEventWatcher(scope, query);
watcher.EventArrived += WatcherEventArrived;
watcher.Start();
イベント EventArrived のハンドラーは次のようになります。
private void WatcherEventArrived(object sender, EventArrivedEventArgs e)
{
string eventName;
var mbo = e.NewEvent;
eventName = mbo.ClassPath.ClassName;
mbo.Dispose();
if (eventName.CompareTo("__InstanceCreationEvent") == 0)
{
Console.WriteLine("Started");
}
else if (eventName.CompareTo("__InstanceDeletionEvent") == 0)
{
Console.WriteLine("Terminated");
}
}
このコードは、CodeProject の記事に基づいています。メモリ リークが発生したため、呼び出しを追加しましたmbo.Dispose()
: EventArrived が発生するたびに約 32 KB、1 秒に 1 回。リークは、WinXP と Win7 (64 ビット) の両方で明らかです。
ここまでは順調ですね。良心的になろうとしてtry-finally
、次のような句を追加しました。
var mbo = e.NewEvent;
try
{
eventName = mbo.ClassPath.ClassName;
}
finally
{
mbo.Dispose();
}
問題ありません。さらに良いことに、C#using
句はよりコンパクトですが同等です。
using (var mbo = e.NewEvent)
{
eventName = mbo.ClassPath.ClassName;
}
メモリ リークが再発しました。どうしたの?
まあ、私は知りません。しかし、ILDASM で 2 つのバージョンを逆アセンブルしてみましたが、ほとんど同じではありません。
イリノイからtry-finally
:
.try
{
IL_0030: nop
IL_0031: ldloc.s mbo
IL_0033: callvirt instance class [System.Management]System.Management.ManagementPath [System.Management]System.Management.ManagementBaseObject::get_ClassPath()
IL_0038: callvirt instance string [System.Management]System.Management.ManagementPath::get_ClassName()
IL_003d: stloc.3
IL_003e: nop
IL_003f: leave.s IL_004f
} // end .try
finally
{
IL_0041: nop
IL_0042: ldloc.s mbo
IL_0044: callvirt instance void [System.Management]System.Management.ManagementBaseObject::Dispose()
IL_0049: nop
IL_004a: ldnull
IL_004b: stloc.s mbo
IL_004d: nop
IL_004e: endfinally
} // end handler
IL_004f: nop
イリノイからusing
:
.try
{
IL_002d: ldloc.2
IL_002e: callvirt instance class [System.Management]System.Management.ManagementPath [System.Management]System.Management.ManagementBaseObject::get_ClassPath()
IL_0033: callvirt instance string [System.Management]System.Management.ManagementPath::get_ClassName()
IL_0038: stloc.1
IL_0039: leave.s IL_0045
} // end .try
finally
{
IL_003b: ldloc.2
IL_003c: brfalse.s IL_0044
IL_003e: ldloc.2
IL_003f: callvirt instance void [mscorlib]System.IDisposable::Dispose()
IL_0044: endfinally
} // end handler
IL_0045: ldloc.1
どうやら問題は次の行です。
IL_003c: brfalse.s IL_0044
これは と同等であるためif (mbo != null)
、mbo.Dispose()
呼び出されることはありません。しかし、mbo がアクセスできた場合、mbo が null になる可能性はあります.ClassPath.ClassName
か?
これについて何か考えはありますか?
また、この動作がここで未解決の議論を説明するのに役立つかどうか疑問に思っています:イベント ログを照会するときの WMI でのメモリ リーク。