勇哥注:
《多线程安全》这个系列会持续写下去,它是我的一个弱点,有兴趣的朋友可以选择性看看。
运行源码后,执行button1
for (int i = 0; i < 5; i++) { var k = i; //加上下面这句效果又不一样 Task.Run(() => { Console.WriteLine($"{i},{k},start...[{Thread.CurrentThread.ManagedThreadId}]"); Thread.Sleep(1000); Console.WriteLine($"{i},{k},end...[{Thread.CurrentThread.ManagedThreadId}]"); }); }
程序的目的是:在for循环中,打印出i
结果是:
(1)其中i都是5
(2)如果设置局部变量k,则结果是正确的。
这个原因是因为循环中启动task后是不会阻塞的,当你5次循环完成后,task这个时候才开始执行,此时读到的i都是5了。
task内部是由调度器执行,对于多多线程,我们是无法知道哪个先运行哪个后运行的。
局部变量k在循环的作用域内,5次循环就相当于有5个k
因此可以避免上面这种问题。
另一种常见的错误是OutOfRange。
它的原因是因为在集合中启动task,并且还企图下标访问集合。
i怎么可能为10呢?其道理跟button1那个例 子所说是一样的。
测试程序 :
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; namespace WindowsFormsApplication1 { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void button1_Click(object sender, EventArgs e) { //outofrange的问题也跟这是一样的。 //此问题是在循环集合的时候启动线程 for (int i = 0; i < 5; i++) { var k = i; //加上下面这句效果又不一样 //Thread.sleep(5); Task.Run(() => { Console.WriteLine($"{i},{k},start...[{Thread.CurrentThread.ManagedThreadId}]"); Thread.Sleep(1000); Console.WriteLine($"{i},{k},end...[{Thread.CurrentThread.ManagedThreadId}]"); }); } } private void button2_Click(object sender, EventArgs e) { var list1 = new List<int>(); list1=Enumerable.Repeat(1,10).ToList(); for(int i=0;i<list1.Count;i++) { Task.Run(() => { Console.WriteLine($"{list1[i]}"); }); } } } }
---------------------
作者:hackpig
来源:www.skcircle.com
版权声明:本文为博主原创文章,转载请附上博文链接!

