10

最初に、サイトで同様の質問に対するいくつかの回答を読みましたが、正直なところ、それらは少し混乱していると思います(回答ではなく経験が不足しているためです!)。FileSystemWatcher()クラスを使用して、作成/変更されているファイルのフォルダーを監視しています。イベントが発生したら、プロジェクトに別のフォームをロードします。フォームをロードする代わりに、新しいフォームのコンストラクターが実行しようとするとエラーが発生します。私は1つのスレッドのみを使用しています-別のスレッドでフォームをロードしようとはしていません。私のコードは次のとおりです

 //MainWindow
 public static void FolderWatcher()
  {
        FileSystemWatcher fsWatcher = new FileSystemWatcher();
        fsWatcher.Path = "C:\\dump";
        fsWatcher.Filter = "*";
        fsWatcher.NotifyFilter = NotifyFilters.LastAccess | NotifyFilters.LastWrite
        | NotifyFilters.FileName | NotifyFilters.DirectoryName;
        fsWatcher.Created += new FileSystemEventHandler(OnChanged);
        fsWatcher.EnableRaisingEvents = true;    
  }

  public static void OnChanged(object source, FileSystemEventArgs e)
  {
       var imagePreview = new ImagePreview();
       imagePreview.Show();
  }


  //SecondForm
  public partial class ImagePreview : Window
  {
        public ImagePreview()
        {
              InitializeComponent(); //error occurs here
        }
  }

よろしくお願いします。よろしくお願いします。

4

3 に答える 3

22

使用しているスレッドの数は関係ありません。ルールがあります。UIを作成するスレッドはすべてSTAである必要があります。

スレッドが1つしかない場合は、これがSTAである必要があります。:-)メインスレッドをSTAにするには、 :のSTAThread属性を使用する必要があります。Main

[STAThread]
static void Main(string[] args)
{
    // ...

標準のWPFアプリケーションを作成するだけの場合、メインスレッドはすでに必要な属性でマークされているため、変更する必要はありません。

FileSystemWatcherからのイベントは、フレームワークによって内部的に作成された他のスレッドから発生する可能性があることに注意してください。(でブレークポイントを設定することで確認できますOnChanged。)この場合、ウィンドウの作成をSTAスレッドに転送する必要があります。アプリケーションがWPFアプリケーションの場合、次のように実行されます。

public static void OnChanged(object source, FileSystemEventArgs e)
{
    var d = Application.Current.Dispatcher;
    if (d.CheckAccess())
        OnChangedInMainThread();
    else
        d.BeginInvoke((Action)OnChangedInMainThread);
}

void OnChangedInMainThread()
{
    var imagePreview = new ImagePreview();
    imagePreview.Show();
}
于 2012-10-03T19:38:11.053 に答える
5

非UIスレッドからUIのものを呼び出しています(つまり、FileSystemWatcher非UIスレッドからイベントを発生させています)。

唯一のWPFプロジェクトで使用した拡張メソッドを作成しました(ugh):

public static class DispatcherEx
{
    public static void InvokeOrExecute(this Dispatcher dispatcher, Action action)
    {
        if (dispatcher.CheckAccess())
        {
            action();
        }
        else
        {
            dispatcher.BeginInvoke(DispatcherPriority.Normal,
                                   action);
        }
    }
}

したがって、メソッドを非静的にしてMainWindow Dispatcherにアクセスできるようにした後、イベントコールバックでこれを行います。

public void OnChanged(object source, FileSystemEventArgs e)
{
    this.Dispatcher.InvokeOrExecute(()=>{
       var imagePreview = new ImagePreview();
       imagePreview.Show();
    });
}
于 2012-10-03T19:59:51.873 に答える
1

メイン ウィンドウのディスパッチャを使用します。通常、アプリケーションごとに 1 つの UI スレッドしかありません。それ以外の場合、WPF から「無効なクロススレッド アクセス」例外が発生する可能性があります。

ファイル システム ウォッチャーは別のスレッドでイベントを発生させ、ディスパッチャーを使用してそのコードを UI スレッドで実行できます。

そのイベント ハンドラで Dispatcher.BeginInvoke を使用すれば問題ありません。:)

于 2012-10-03T19:43:53.237 に答える