苏飞论坛

 找回密码
 马上注册

QQ登录

只需一步,快速开始

分布式系统框架(V2.0) 轻松承载百亿数据,千万流量!讨论专区 - 源码下载 - 官方教程

HttpHelper爬虫框架(V2.7-含.netcore) HttpHelper官方出品,爬虫框架讨论区 - 源码下载 - 在线测试和代码生成

HttpHelper爬虫类(V2.0) 开源的爬虫类,支持多种模式和属性 源码 - 代码生成器 - 讨论区 - 教程- 例子

查看: 15593|回复: 11

[学生提问] NET初级面试题求解--多线程

[复制链接]
发表于 2014-9-7 23:14:04 | 显示全部楼层 |阅读模式
1金钱
下面是一段源程序,请帮忙分析为什么输出的是 "BruceBruce" 而不是 "MagicBruce"
    class Program
    {
        static string outString = "";
        static void Main(string[] args)
        {
            List<Stu> stuList = new List<Stu> { new Stu { StuName = "Magic" }, new Stu { StuName = "Bruce" } };
            foreach (Stu name in stuList)
            {
                Thread thread = new Thread(delegate() { CreateLoginProcess(name); });
                thread.IsBackground = true;
                thread.Start();
            }
            Thread.Sleep(5000);
            Console.WriteLine(outString);
            Console.ReadKey();
        }
        public static void CreateLoginProcess(Stu stu)
        {
            outString += stu.StuName;
        }
    }
    public class Stu
    {
        public string StuName { get; set; }
    }



1. 开通SVIP会员,免费下载本站所有源码,不限次数据,不限时间
2. 加官方QQ群,加官方微信群获取更多资源和帮助
3. 找站长苏飞做网站、商城、CRM、小程序、App、爬虫相关、项目外包等点这里
发表于 2014-9-7 23:47:18 | 显示全部楼层
把  Thread.Sleep(2000);放到循环里面就是MagicBruce了     

这个问题其实是这样的,是因为线程是从最后一个开始执行的,

只有你设置了间隔时间才是从第一个开始执行的。
因为For太快了,远远超过了线程的启动速度。
回复

使用道具 举报

 楼主| 发表于 2014-9-9 08:55:54 | 显示全部楼层
站长苏飞 发表于 2014-9-7 23:47
把  Thread.Sleep(2000);放到循环里面就是MagicBruce了     

这个问题其实是这样的,是因为线程是从最后 ...

for循环外的sleep函数能保证两个后台线程都能执行完毕,如果有快慢之分,那么应该输出 MagicBruce或者BruceMagic,这里输出了两遍 "BruceBruce"存在变量覆盖(即第二次传递的name值Bruce把第一次传递的值Magic覆盖了),这个该怎么解释呢?
回复

使用道具 举报

发表于 2014-9-10 10:57:25 | 显示全部楼层
测试了你的代码,没发现你的状况,但我遇到新的状况:
1.+=操作的非原子性,导致输出结果大多是 "Bruce"或者"Magic",而非两个的合体
2.我加了lock锁,后状况正常
[C#] 纯文本查看 复制代码
internal class Program
    {
        private static  string outString = "";

        static void Main(string[] args)
        {
            var stuList = new List<Stu> {new Stu {StuName = "Magic"}, new Stu {StuName = "Bruce"}};
            foreach (Stu name in stuList)
            {
                var thread = new Thread(delegate() { CreateLoginProcess(name); });
                thread.IsBackground = true;
                thread.Start();
            }
            Thread.Sleep(500);
            Console.WriteLine(outString);
            Console.ReadKey();
        }

        public static void CreateLoginProcess(Stu stu)
        {
            lock (outString)
            {
                outString += stu.StuName;
            }
            
        }
    }

    public class Stu
    {
        public string StuName { get; set; }
    }
回复

使用道具 举报

发表于 2014-9-10 11:07:57 | 显示全部楼层
强烈支持楼主ing……站长飞,
回复

使用道具 举报

 楼主| 发表于 2014-9-13 10:12:34 | 显示全部楼层
守望幸福 发表于 2014-9-10 10:57
测试了你的代码,没发现你的状况,但我遇到新的状况:
1.+=操作的非原子性,导致输出结果大多是 "Bruce"或 ...

经测试你的源码输出的还是 "BruceBruce",加Lock后结果依旧,请查证。
回复

使用道具 举报

发表于 2014-9-14 11:00:23 | 显示全部楼层
MagicBruce
MagicBruce
MagicBruce
MagicBruce
MagicBruce
MagicBruce
MagicBruce
MagicBruce
MagicBruce
MagicBruce
结束
QQ图片20140914105921.png
回复

使用道具 举报

 楼主| 发表于 2014-9-15 08:54:41 | 显示全部楼层
守望幸福 发表于 2014-9-14 11:00
MagicBruce
MagicBruce
MagicBruce

老兄,帮忙看看我哪里出错了,怎么输出结果跟你的不一致呢?

回复

使用道具 举报

发表于 2014-9-16 10:50:45 | 显示全部楼层
我出现了三种情况,第一种“MagicBruce”,第二种“BruceBruce”,第三种“Bruce”,这是在没lock锁的情况下,两个线程争抢引起了数据混乱,加了lock锁后只出现了一种“BruceBruce”,但是这也是我比较疑惑的地方,按理说应该跟不加lock锁的情况一样的,因为lock锁值类型永远返回false,也就是不能起到lock锁的作用,两个线程还是能同时访问到数据,达不到同步的效果,但是一直输出“BruceBruce”这点让我很疑惑,可能是我试验的次数还是少,我参考的资料是:http://www.cnblogs.com/promise-7/articles/2354077.html,你可以自己研究下
回复

使用道具 举报

发表于 2014-9-16 10:55:10 | 显示全部楼层
哦,对了,lock锁定字符串是会锁定该字符串所有实例,可能是这个原因吧,第一个进去的"Magic",被锁定了,其他的线程都访问不到,只能访问“Bruce”,还是参考上面的资料,lock字符串
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 马上注册

本版积分规则

QQ|手机版|小黑屋|手机版|联系我们|关于我们|广告合作|苏飞论坛 ( 豫ICP备18043678号-2)

GMT+8, 2025-1-11 00:06

© 2014-2021

快速回复 返回顶部 返回列表