我們知道,在Metro C++中非同步作業都是需要開發人員利用task、then這樣的方式去實現的,不像C#裡通過一個await就會確保當前的非同步執行完才會執行下一句。
最近在檔案儲存、將一檔案內容寫入另一檔案中經常會遇到線程衝突的問題,如,將檔案A的內容寫入檔案B,再將A刪除,這樣的操作將會涉及以下幾個步驟:
(1) 開啟B檔案,通過檔案流初始化DataWriter對象
(2) 開啟A檔案,通過檔案流初始化DataReader對象
(3) 通過DataReader對象負載檔案A的內容
(4) 通過DataWriter將內容寫入
(5) DataWriter儲存
(6) DataWriter重新整理
以上每一步都必須一環套一環,若沒有確保當前對A、B檔案的操作完全結束,而在其它線程又開始操作這些檔案,就會經常出現線程衝突。
下面提供對以上步驟一個簡單樣本:task<IRandomAccessStream^>(StorageFileB->OpenAsync(FileAccessMode::ReadWrite))
.then([this](IRandomAccessStream^ raStream)
{
IOutputStream^ outStream = raStream->GetOutputStreamAt(0);
DataWriter^ bbrw = ref new DataWriter(outStream);
task<IRandomAccessStream^>(StorageFileA->OpenAsync(FileAccessMode::Read))
.then([bbrw,this](IRandomAccessStream^ readStream)
{
DataReader^ dataReader = ref new DataReader(readStream);
task<unsigned int>(dataReader->LoadAsync((unsigned int)readStream->Size)).then([this,bbrw,dataReader](unsigned int numBytesLoaded)
{
Array<unsigned char,1>^ fileContent =ref new Array<unsigned char,1>(numBytesLoaded);
dataReader->ReadBytes(fileContent);
bbrw->WriteBytes(fileContent);
task<unsigned int> store(bbrw->StoreAsync());
store.then([bbrw] (unsigned int)
{
return bbrw->FlushAsync();
}).then([=](bool flushSucceeded)
{
bbrw->DetachStream(); Do anything….
});
});
});
});
最後一個bbrw->DetachStream()非常重要哦,它是用來分離DataWriter關聯流的,如果我們不將之分離,再一次操作檔案流時也會容易發生衝突。
如果在項目中會多次同一個檔案,一定要確保不要同時操作同一個檔案,否則很容易發生線程衝突。若是在不同的模組要操作同一個檔案,可以通過訊號量來處理。