🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

SQL Server

SQL Serverはマイクロソフトのリレーショナルデータベース管理システムです。データマイニングや多次元解析など、ビジネスインテリジェンスのための機能が備わっています。

並列処理

複数の計算が同時に実行される手法

イベントハンドラ

マウスのクリックなどの特定の事象(イベント)が発生した時に実行される処理のことをイベントハンドラと呼びます。

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

Q&A

解決済

3回答

1721閲覧

[C#]椅子取りゲーム(複数人が同一イベントを同時に発火し、先着1名のみtrueになる)イベントの実装

woria

総合スコア36

C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

SQL Server

SQL Serverはマイクロソフトのリレーショナルデータベース管理システムです。データマイニングや多次元解析など、ビジネスインテリジェンスのための機能が備わっています。

並列処理

複数の計算が同時に実行される手法

イベントハンドラ

マウスのクリックなどの特定の事象(イベント)が発生した時に実行される処理のことをイベントハンドラと呼びます。

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

0グッド

0クリップ

投稿2021/01/13 04:45

編集2021/01/13 07:35

前提・実現したいこと

FileSystemWatcherによってファイルが変更されたことを感知した時、そのデータをDBに書き込んだ後、DBからデータを取得してDataGridViewで表示する処理を作成しています。
本プログラムは1人以上10人以下が同時に起動しています。
その際、起動している全員が同時にファイルの変更を感知し、DBに書き込むことで、同一のレコードが起動している人数分だけ複製されてしまいます。

そこで、変更の感知後、先着1名がDBを更新し、残りの人はDBの更新を待機する処理を実装したいです。

該当のソースコード

C#

1private FileSystemWatcher _watcher = new FileSystemWatcher(); 2public WatchForm() 3{ 4 InitializeComponent(); 5 6 // フォームが呼び出されたとき、フォルダの監視を開始する 7 _watcher.Path = Properties.Resources.Folder; 8 _watcher.Filter = "*.csv" 9 _watcher.IncludeSubdirectories = true; 10 _watcher.NotifyFilter = (NotifyFilters.FileName | NotifyFilters.CreationTime | NotifyFilters.LastWrite); 11 _watcher.Created += new FileSystemEventHandler(OnResultChanged); 12 _watcher.Changed += new FileSystemEventHandler(OnResultChanged); 13 _watcher.Deleted += new FileSystemEventHandler(OnResultChanged); 14 _watcher.EnableRaisingEvents = true; 15} 16 17private void OnResultChanged(object sender, FileSystemEventArgs eventArgs) 18{ 19 // 更新権限を持つ人なら 20 if(HasReloadAuthority()) 21 { 22 // ファイルを読み取り、DBに書き込む 23 Program.DataBaseReload(); 24 } 25 else 26 { 27 // 更新権限を持つ人が更新を完了するまで待つ 28 Wait() 29 } 30 31 // フォームをリロードする(DBからデータを取得し、DataGridViewを更新する) 32 WatchForm_Load(sender, eventArgs); 33} 34 35private bool HasReloadAuthority() 36{ 37 // 先着1名のみ更新権限を持たせる処理 38}

補足情報(FW/ツールのバージョンなど)

.NET Framework 4.6.1
Windows アプリケーション

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答3

0

その際、起動している全員が同時にファイルの変更を感知し、DBに書き込むことで、同一のレコードが起動している人数分だけ複製されてしまいます。

そこで、変更の感知後、先着1名がDBを更新し、残りの人はDBの更新を待機する処理を実装したいです。

待機させたい理由が同一レコードになってしまう、という点だけであれば、DBに何を使用しているかは判りませんが、シーケンス等で自動採番するカラムを追加すれば良いと思います。

投稿2021/01/13 06:11

編集2021/01/13 06:24
退会済みユーザー

退会済みユーザー

総合スコア0

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

woria

2021/01/13 07:27

DBにはSQL Serverを使用しており、TableAdapter().Update()を使って更新処理を実装しています。 設計段階でシーケンス番号は付与していませんでしたが、手詰まりになったら採用してみます。
退会済みユーザー

退会済みユーザー

2021/01/13 07:29 編集

SQL Serverを使用しているのであれば、タグと補足情報に追記した方が良いと思われます。SQL Server詳しい人からコメントが付くかもしれませんし。
woria

2021/01/13 07:34

ありがとうございます。タグにSQLを追加します。
guest

0

System.Threading.Mutex を使えばプロセスをまたいだ排他制御が可能なので、
これを使って先着一名のみが更新/二番目以降は待機できるのでは。

投稿2021/01/13 05:08

episteme

総合スコア16612

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

woria

2021/01/13 05:44

System.Threading.Mutex確認してみます。
woria

2021/01/13 07:29

更新処理を非同期で記載しており、asyncを利用する際はSemaphoreSlimを利用すると記載されているため、そちらを確認しています。
dodox86

2021/01/13 07:41

>@質問者さん > asyncを利用する際はSemaphoreSlimを利用すると記載されているため、そちらを確認しています。 質問者さんのアプリの構成がいまいち分からないのですけど、こちらのリファレンスを見ると、 https://docs.microsoft.com/ja-jp/dotnet/api/system.threading.semaphoreslim?view=netframework-4.6.1 > SemaphoreSlimクラスは、単一のアプリ内で同期するために推奨されるセマフォです。 とあるので、複数プロセス間での排他制御には使えないのではないでしょうか。 (外れた指摘でしたらすみません)
woria

2021/01/13 07:52

SemaphoreSlimだと「先着1名のみ更新作業、残りは更新作業を行わない」という分岐処理ができない?みたいなので、更新作業は同期処理で書き直してMutex試してみます。
woria

2021/01/13 07:53

SemaphoreSlimだと複数アプリ間での排他制御には使えないんですね。ありがとうございます。
guest

0

自己解決

DBの更新処理にトランザクション処理を加えることで、同時起動した時に排他制御するように処理を変更しました。回答いただいた方々、ありがとうございました。

投稿2021/01/14 01:40

woria

総合スコア36

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問