- 我是一个线程(修订版)
- 我是一个 Java class
- Javascript:一个屌丝的逆袭
- Java : 一个帝国的诞生
- JSP 一个装配工的没落
- TCP/IP 之 大明王朝邮差
- TCP/IP 之大明内阁
- TCP/IP 之蓟辽督师
- CPU 阿甘
- CPU 阿甘之烦恼
- CPU 阿甘:函数调用的秘密
- 我是一个网卡
- 我是一个路由器
- 我是一个进程
- 我是一块硬盘(上)
- 我是一块硬盘(下)
- 我是一个键盘
- 张大胖的 socket
- 张大胖学递归
- 学习面向对象的令狐冲
- 张大胖学数据库
- 数据库村的旺财和小强
- 小李的数据库之旅(上)
- 小李的数据库之旅(下)
- 漫画:什么是机器学习?
- 那些烦人的同步和互斥问题
- IE 为什么把火狐和 Chrome 给打伤了?
- 对浏览器村的第二次采访
- 节约标兵 IE 的自述
- EMail 诞生记
- Email 诞生记(下)
- Http 历险记(上)
- Http 历险记(下)-- Struts 的秘密
- 动物王国的面向对象
- 冯·诺伊曼计算机的诞生
- Http Server : 一个差生的逆袭
- 张大胖的加法器
- 从 1 加到 100:一道简单的数学题挑战下你的大脑
- 编程语言
- Javascript:一个屌丝的逆袭
- 计算机语言之战
- 我和编程语言的爱恨情仇(上)
- 我和编程语言的爱恨情仇(下)
- Android 为什么选择了 Java
- iOS 为什么选择了 Object-C?
- Basic : 一个老兵的自述
- Node.js : 我只需要一个店小二
- 命令式编程 vs 声明式编程
- 编译还是解释?
- 程序人生
- “架构师"小赵
- 师兄说
- 师姐说
- 小王的架构师之路
- 小李的版本管理系统
- 小超穿越记
- 小李的 Build 之路(上)
- 小李的 Build 之路(下)
- 张大胖改 Bug
- 我的编程之路--大学趣事
- 码农小王的一天
- 小李在外企
- 张大胖的需求估算
- 从厨师到码农
- 聊一聊那些神一样的程序员们(上)
- 聊一聊那些神一样的程序员们(中)
- 聊一聊那些神一样的程序员们(下)
- 谁是互联网之父?
- 一个价值百万的创业教训
- 让自己与众不同 - 提升工作的价值
- 看看你的“易燃性”
- 从无聊的工作中寻找价值
- 什么样的学生适合报考计算机?
- 谈谈程序员的职业方向(上)
- 谈谈程序员的职业方向(中)
- 谈谈程序员的职业方向(下)
- 谈谈培训班的作用
- 码农需要知道的“潜规则”
- 学习编程的加速度
- 码农在工作中的必备能力
- 码农和英语
- 老司机经验
- 假如时光能够倒流, 我会这么学习 Java
- 假如我是计算机系老师
- 学会编程, 而不是学会 Java
- 从增删改查中突围
- 抽象:程序员必备的能力
- 懒就一个字
- 编程的自学方法
- 小王买房记
- 从一道面试题谈谈一线码农应该具备的基本素质
- 想写框架的看过来
- 苹果手机变砖头以后
- 如何快速的学习一门技术?
- 唯一不变的是变化: 谈谈微信应用号
- 什么是企业应用?
- 勿以浮沙筑高台
- 为什么敏捷开发难于成功?
- localhost vs 127.0.0.1
- GitHub/Stackoverflow 找工作时有什么用?
- 动词 or 名词 :这是一个问题
- 如何选择入行语言
- 有时候,沉默是金
- 零 Bug 的代码是怎么炼成的?
- 浮点数为什么不精确?
- 文章错误大全
- Open Source--不要为了开源而开源
- 一不留神,代码就腐化了
- 先做个“键盘侠”, 再来写程序
- 不加断点调试的程序员是好程序员
- 码农必备技能:烂代码的处理之道(上)
- 码农必备技能:烂代码的处理之道(下)
- 学习数据结构有用吗?
- 从现在开始,丰富你的简历
- 那些永不过时的书,你看过几本吗?
- 学好编程必备的一个品质你知道吗?
- 你最爱的 Java
- 搞懂了这几点,你就学会了 Web 编程
- Spring 的本质系列(1) -- 依赖注入
- Spring 本质系列(2)-AOP
- 三层架构和 MVC 那点事儿
- Java 帝国之拨云见日识回调
- 小张的 Duck Typing
- JDBC 的诞生
- JDBC 后传
- 一个不安分的 JDBC 驱动
- Java 帝国之 Java bean (上)
- Java 帝国之 Java bean(下)
- Java 帝国之函数式编程
- Java 帝国之函数式编程(下)
- 关于 Java 初学者需要知道的 10 件事
- JUnit 你不知道的那些事儿
- 圣诞礼物:Java EE 的历史
- Java EE 读书指南
- 给小白的 Java EE 指南
- 给小白的 Java EE 指南(2)
- 给小白的 Java EE 生存指南(3) : XML
- 给小白的 Java EE 生存指南(4) : 一只叫 Tom 的猫
- 给小白的 Java EE 指南(5) : AJAX
- 给小白的 Java EE 生存指南(6) :Java 反射
- 闲聊
- "饿了么"初体验
- 来自大脑的控诉
- 一个高中生是怎么玩自媒体的?
- 尝试 分答
- 到底应不应该上培训班?
- 自学编程中遇到问题怎么办?
- 据说 99%的初级程序员看完后都不迷茫了
- 一行代码引发的“血案”
- 对一个死锁问题的思考
- 通过外包进入名企
- 请开往十年前的今天
- 为什么自学中最好有个师傅指导一下?
- 这个网站值得你花时间投入
- 为什么你无法坚持自学编程?
那些烦人的同步和互斥问题
1 批处理和脱机打印 打印机程序, 准确的说是打印机进程, 在这个批处理系统中生活的非常自在, 它所在的机器叫做 IBM1401 , 除了打印之外什么也不干, 每天大部分时间都是歇着。
这个系统还有两台机器, 一台还是 IBM1401,它专门收集程序员写出来的穿孔卡片, 然后转成磁带。
然后操作员把磁带输入到 IBM7094 这个昂贵又强大的计算机上执行, 执行结果也会输出到磁带上。
最后磁带被拿到 1401 上进行打印, 这叫做 脱机打印 (不和 7094 相连接)。 3 冲突 但是这个队列可不是原来的磁带了, 它完全是个动态变化的东西,试运行还不到 20 秒, 冲突就出现了。
WPS 气冲冲的来着打印机进程:"打印机, 你怎么搞的, 我的 放假通知.wps 为什么没有打印?“ 打印机:“我没看到什么放假通知.wps 啊”
WPS: “我明明放在了编号为 3 的槽里, 怎么可能没有了?”
在操作系统老大的协助下, 大家查了半天,才知道是 Word 引起的:
当时 Word 插了一脚,也进来打印, 读到了 in = 3, 就是说队列中编号为 3 的槽是空着的, 他把 3 这个值放到了自己的局部变量 free_slot 中, 这时候发生了一次时钟中断 , 操作系统老大认为 Word 已经运行了足够长的时间,决定切换到 WPS 进程。
WPS 也读到了 in = 3, 把 3 也存到自己的局部变量 free_slot 中, 现在 Word, WPS 都认为 下一个空的槽是 3 !
WPS 接着干活, 他把文件放到了第 3 号槽里, 并且把 in 改为 4, 然后离开了。
接下来又轮到 Word 运行了, 它发现 free_slot 为 3 , 就把文件也放到了第 3 号槽里, 把 free_slot 加 1,得到 4, 存入 in 中。
可怜的 WPS , 他的文件被覆盖掉了。 但是打印机程序啥也察觉不出来, 照样打印不误。
4 临界区 很明显, Word 和 WPS 这两个进程甚至多个进程在读写 in 这个共享变量的时候, 最后的结果严重依赖于进程运行的精确次序, 这次是 WPS 的文件被覆盖掉了, 下次可能就是 Word 了。
这种对共享变量, 共享内存,共享资源进行访问的程序片段叫做临界区 , 代码在进入临界区之前一定要做好同步或者互斥的操作。
WPS 说: ”老大, 当时你切换 Word 的时候是不是发生了一次时钟中断 ?“
操作系统: “是啊, 有了时钟中断我才能计算时间, 然后做进程切换啊”
“那在访问这个 in 共享变量的时候,我们自己能不能把这个中断给屏蔽? 这样就不会有进程切换, 肯定没问题了。” Word 问到。
“你想的美,时钟中断是最基本的东西, 我把这个权限给了你们应用程序, 到时候那个家伙屏蔽以后忘记开中断, 我们整个系统就要完蛋了! ” 操作系统狠狠的瞪了 Word 一眼, Word 赶紧噤声。
“不过我听说有些机器提供了一个特别的指令, 这个指令能检查并且设置内存的值, 而不会被打断, 叫做 TestAndSet, 如果用 C 语言描述的话,类似这样:” WPS 说: “果然是抽象大法好, 这多简单啊。 “ 操作系统说: “是啊, 无论是什么东西,抽象以后用起来好多了, 但是还是要了解底层,这样出现了类似于 BlockingQueue 这样的新概念, 你能迅速搞明白。 ”
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论