我有一个具有并行线程计时器的程序。基本上这些计时器设置为每10秒运行一次方法。在该方法中,线程获得对列表的锁定以制作副本以完成它的工作。
private object access = new object();
PeriodTask(){
lock(access){
copy = list.Where(e => e.valid).toList();
}
foreach (var element in copy)
do some not so quick work.
}
我还有办法获取那些也要求锁定的元素的信息。并获得一份副本
informationMethod(){
lock(access){
return list.Select(element => new serializableElement(element)).toList()
}
}
我需要锁定的原因是因为有一些方法可以添加和删除此列表中的项目以锁定它。但我还没有打电话给他们。
我遇到的问题是信息检索方法在这个类的实例之间长时间阻塞或睡眠。可能有大约10个,它似乎一次可以访问2或3,但这个过程可能需要一分半钟到两分钟才能获得所有这些。我提到的第一个在计时器上执行的方法在应该访问时就可以正常访问并立即释放锁,但是另一个方法很难执行,即使它是执行的主线程。
我有几个测试实例化这些类的管理器。
//sets up updaters test
var manager = new manager(config);
Assert.IsNotNull(manager);
Assert.IsFalse(manager.UpdatersRunning);
Assert.AreNotEqual(0, manager.Updaters.Count());//calls the informationMethod on each updater
manager.Dispose();
//This test doesn't even run the period task
//Waits to start updaters test
var manager = new StatusObjectManager(config);
Assert.IsNotNull(manager);
Assert.AreNotEqual(0, manager.Updaters.Count());
Assert.IsFalse(manager.UpdatersRunning);
Assert.AreEqual(0, manager.Updaters.Count(u => u.Running));
manager.StartUpdaters();
Assert.IsTrue(manager.UpdatersRunning);
Assert.AreNotEqual(0, manager.Updaters.Count(u => u.Running));
manager.Dispose();
//This test runs the period tasks and checks to make sure their running. Or at least 1.
//THis test takes a long time for the first informationMethod bout but not the other 2.
The code in the manager looks like this
return updaters.select(u => new serialUpdater(su));
此时不需要锁定,因为该列表未被修改。
有没有人知道会发生什么。我不知道如何判断线程是否在锁定时阻塞或者只是在异常时间内睡眠。
答案 0 :(得分:0)
这里的第一步是添加日志记录。在锁定之前和之内简单Trace.WriteLine
,您还可以在创建时获取/设置Thread.CurrentThread.Name
,阅读Thread
,以便了解方法中的Thread
。< / p>
下一步是实现一些不那么干净但速度更快的事情,System.Threading.ReaderWriterLockSlim。关于这一点的好处是,它将允许所有线程同时读取,但阻止写入的所有内容。所以基本上你只是在写一个阻止。
另外,在执行ToList
或ToArray
等时请注意参考类型。
void Main()
{
var slist = new List<Chunk>();
var chunk = new Chunk{Name="asdf"};
slist.Add(chunk);
var slist2 = slist.ToList();
slist[0].Name = "asdf2";
Console.WriteLine(slist[0].Name); // outputs: asdf2
Console.WriteLine(slist2[0].Name); // outputs: asdf2
}
public class Chunk
{
public string Name {get;set;}
}