- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我曾尝试使用文件 IO 操作实现读写操作,并将这些操作封装到 TransformBlock
中,从而使这些操作线程安全,而不是使用锁定机制。
但问题是,当我尝试并行写入 5 个文件时,会出现内存异常,并且在使用此实现时它会阻塞 UI 线程。实现是在 Windows Phone 项目中完成的。请提出此实现中的错误。
文件IO操作
public static readonly IsolatedStorageFile _isolatedStore = IsolatedStorageFile.GetUserStoreForApplication();
public static readonly FileIO _file = new FileIO();
public static readonly ConcurrentExclusiveSchedulerPair taskSchedulerPair = new ConcurrentExclusiveSchedulerPair();
public static readonly ExecutionDataflowBlockOptions exclusiveExecutionDataFlow
= new ExecutionDataflowBlockOptions
{
TaskScheduler = taskSchedulerPair.ExclusiveScheduler,
BoundedCapacity = 1
};
public static readonly ExecutionDataflowBlockOptions concurrentExecutionDataFlow
= new ExecutionDataflowBlockOptions
{
TaskScheduler = taskSchedulerPair.ConcurrentScheduler,
BoundedCapacity = 1
};
public static async Task<T> LoadAsync<T>(string fileName)
{
T result = default(T);
var transBlock = new TransformBlock<string, T>
(async fName =>
{
return await LoadData<T>(fName);
}, concurrentExecutionDataFlow);
transBlock.Post(fileName);
result = await transBlock.ReceiveAsync();
return result;
}
public static async Task SaveAsync<T>(T obj, string fileName)
{
var transBlock = new TransformBlock<Tuple<T, string>, Task>
(async tupleData =>
{
await SaveData(tupleData.Item1, tupleData.Item2);
}, exclusiveExecutionDataFlow);
transBlock.Post(new Tuple<T, string>(obj, fileName));
await transBlock.ReceiveAsync();
}
MainPage.xaml.cs 用法
private static string data = "vjdsskjfhkjsdhvnvndjfhjvkhdfjkgd"
private static string fileName = string.Empty;
private List<string> DataLstSample = new List<string>();
private ObservableCollection<string> TestResults = new ObservableCollection<string>();
private static string data1 = "hjhkjhkhkjhjkhkhkjhkjhkhjkhjkh";
List<Task> allTsk = new List<Task>();
private Random rand = new Random();
private string fileNameRand
{
get
{
return rand.Next(100).ToString();
}
}
public MainPage()
{
InitializeComponent();
for (int i = 0; i < 5; i ++)
{
DataLstSample.Add((i % 2) == 0 ? data : data1);
}
}
private void Button_Click(object sender, RoutedEventArgs e)
{
AppIsolatedStore_TestInMultiThread_LstResultShouldBeEqual();
}
public async void AppIsolatedStore_TestInMultiThread_LstResultShouldBeEqual()
{
TstRst.Text = "InProgress..";
allTsk.Clear();
foreach(var data in DataLstSample)
{
var fName = fileNameRand;
var t = Task.Run(async () =>
{
await AppIsolatedStore.SaveAsync<string>(data, fName);
});
TestResults.Add(string.Format("Writing file name: {0}, data: {1}", fName, data));
allTsk.Add(t);
}
await Task.WhenAll(allTsk);
TstRst.Text = "Completed..";
}
异步保存和加载数据
/// <summary>
/// Load object from file
/// </summary>
private static async Task<T> LoadData<T>(string fileName)
{
T result = default(T);
try
{
if (!string.IsNullOrWhiteSpace(fileName))
{
using (var file = new IsolatedStorageFileStream(fileName, FileMode.OpenOrCreate, _isolatedStore))
{
var data = await _file.ReadTextAsync(file);
if (!string.IsNullOrWhiteSpace(data))
{
result = JsonConvert.DeserializeObject<T>(data);
}
}
}
}
catch (Exception ex)
{
//todo: log the megatron exception in a file
Debug.WriteLine("AppIsolatedStore: LoadAsync : An error occured while loading data : {0}", ex.Message);
}
finally
{
}
return result;
}
/// <summary>
/// Save object from file
/// </summary>
private static async Task SaveData<T>(T obj, string fileName)
{
try
{
if (obj != null && !string.IsNullOrWhiteSpace(fileName))
{
//Serialize object with JSON or XML serializer
string storageString = JsonConvert.SerializeObject(obj);
if (!string.IsNullOrWhiteSpace(storageString))
{
//Write content to file
await _file.WriteTextAsync(new IsolatedStorageFileStream(fileName, FileMode.Create, _isolatedStore), storageString);
}
}
}
catch (Exception ex)
{
//todo: log the megatron exception in a file
Debug.WriteLine("AppIsolatedStore: SaveAsync : An error occured while saving the data : {0}", ex.Message);
}
finally
{
}
}
编辑:
它出现内存异常的原因之一是我获取的数据字符串太大。字符串是链接:http://1drv.ms/1QWSAsc
但第二个问题是,如果我也添加小数据,那么它会阻塞 UI 线程。代码是否在 UI 界面上执行任何任务?
最佳答案
不,您使用并发对,它为它的任务使用默认线程池,并且您使用 Run
方法实例化任务,所以问题不在这里。但是你这里的代码有两个主要威胁:
var transBlock = new TransformBlock<string, T>
(async fName =>
{
// process file here
}, concurrentExecutionDataFlow);
你真的不应该每次都创建transBlock
。 TPL Dataflow
的主要思想是您创建一次 block 并在之后使用它们。因此,您应该重构您的应用程序以减少实例化的 block 数,否则不应使用 TPL Dataflow
。
代码中的另一个威胁是您显式地阻塞了线程!
// Right here
await Task.WhenAll(allTsk);
TstRst.Text = "Completed..";
从同步事件处理程序的 async void
方法调用任务的 await
会阻塞线程,默认情况下 it captures the synchronization context .首先,async void
should be avoided .其次,如果你是异步的,你 should be async all the way ,所以事件处理程序也应该是异步的。第三,你可以使用 continuation for your task用于更新您的 UI 或 use current synchronization context .
所以,你的代码应该是这样的:
// store the sync context in the field of your form
SynchronizationContext syncContext = SynchronizationContext.Current;
// avoid the async void :)
public async Task AppIsolatedStore_TestInMultiThread_LstResultShouldBeEqual()
// make event handler async - this is the only exception for the async void use rule from above
private async void Button_Click(object sender, RoutedEventArgs e)
// asynchronically wait the result without capturing the context
await Task.WhenAll(allTsk).ContinueWith(
t => {
// you can move out this logic to main method
syncContext.Post(new SendOrPostCallback(o =>
{
TstRst.Text = "Completed..";
}));
}
);
关于c# - IO 读写操作的 TPL 数据流实现中的内存问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34427707/
我用 chown 不行。 Bilals-MBP:~ $ sudo mkdir -p /data/db Password: mkdir: /data/db: Read-only file system
我陷入了一个非常简单的问题。 我正在尝试制作一个Qt GUI应用程序以从GUI控制我的Arduino(而不是从Arduino IDE的串行监视器控制它)。我能够使用QSerialPort write(
我正在尝试使用 Win32 的 CreateFile 函数打开一个 COM 端口。我已经在 MSDN 以及几个论坛上阅读了有关如何执行此操作的文档,但无论我做什么,我仍然收到错误代码 #2(端口不存在
我正在尝试使用系统调用 read() 和 write()。以下程序创建一个文件并将一些数据写入其中。这是代码.. int main() { int fd; open("stud
我对 Xcode 和 sqlite 有点陌生。现在我有一个名为“mydb.db”的数据库文件,它已经有一些表和数据。我把它放在我的 mac 文件夹中,然后将它拖到“支持文件”下的 Xcode 项目中。
背景:如果需要,请跳至问题部分 我正在研究测试设备的前端。前端的目的是为了更容易编写长测试脚本。几乎只是让它们更易读和可写。 设备将使用 Prologix GPIB-USB Controller 进行
本文实例讲述了python文件常见操作。分享给大家供大家参考,具体如下: 1.文件是什么? 文件是存储在外部介质上的数据或信息集合,程序中源程序、数据中保存的数据、图像中的像素数据等等; 文件
C++0x 指定 std::atomic线程安全原子访问变量的模板。这个模板有一个成员函数 std::atomic::exchange原子地在“this”中存储一个新值并检索“this”的现有值。 W
VBA 中是否有任何方法可以读取和写入 INI 文件?我知道我可以使用; Open "C:\test.ini" For Input As #1 ...并解析数据。相反,我试图查看已有哪些工具可用。 我
我最近在 GitHub 存储库 system-design-primer 上看到了系统设计示例,它显示了读/写 API。我正在尝试实现 this one 以进行练习。大纲是这样的。 它分离了读写API
我在使用 DEVMODE 结构的 dmColor 字段时遇到问题。 我的默认打印机是彩色打印机,如果我通过控制面板将打印机属性的颜色默认输出为黑白,则 DEVMODE.dmColor 字段始终返回 D
我知道套接字等如何与 java/android 配合使用,但是如何使用 java 或 python 连接到桌面上的 COM 端口?您想使用地址吗?或者查找您想要的端口是否可用或者什么? 我不知道该怎么
什么构成 DynamoDB 中的实际读取? 它是读取表格中的每一行还是返回什么数据? 这就是扫描如此昂贵的原因 - 您读取整个表格并为读取的每一行表格付费吗? 能否将 ElasticCache (Me
我想用Java编写一个程序来检查src是否存在(如果不抛出FileNoot的话) 并将src.txt的内容复制到des.txt 并在开头和结尾处打印两个文件的大小 输出为: src.txt is in
我在 C++ 中有一个 float 数组,我想将它保存到一个二进制文件中(以节省空间),以便以后能够再次读取它。为此,我编写了以下代码来编写数组: float *zbuffer = new float
我试图为 websocket 创建一个 Read\Write 函数,但我遇到了一个问题...... var inarrivo = 0; var risposta = ""; function RDW_
在我的应用程序中是用 Qt 编写的,我有一个 QGraphicsScene。在这个 QgraphicsScene 中有一个图像和一些由用户绘制的项目。我想保存这个包含所有内容的 QgraphicsSc
我正在编写一个程序,该程序应该无限期运行并保持变量的值。其他两个程序可以更改变量的值。我使用命名管道接收变量值并将其发送到外部程序。 这是我的变量管理器代码。 manager.c: #includ
我和一位老师谈过,他告诉我读写系统调用使用缓冲区,因为在你的系统规范中有一个变量控制你可以访问你想要读/写的设备的次数on,系统在他等待写入设备时使用buffer来存储数据。 我在另一篇 Stack
我运行一个方法,有三个部分,第 1 部分和第 3 部分都是“读取文本文件”, 第二部分是将字符串保存到文本文件, // The Save Path is the text file's Path, u
我是一名优秀的程序员,十分优秀!