博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Swing Threading的限制
阅读量:6673 次
发布时间:2019-06-25

本文共 1250 字,大约阅读时间需要 4 分钟。

一个GUI程序是具有数个thread的,这些thread之中有一个被称为时间派发(event-dispatching) thread。这个thread运行程序中所有与事件有关的回调(callback),如打字游戏中的actionPerformed()与keyPressed()方法。对所有Swing对象的访问都必须发生在此thread中。

 

之所以需要这样的原因是Swing并没有用同步来访问本身对象的复杂内在状态。以JSlider对象为例,它有个用来指示slider位置的单一值。如果使用者正在改变此slider的位置,该值可能正位于中间或尚未确定的状态,这样的处理全部都是发生在事件派发thread中。另一个尝试要读取该slider值的thread将无法直接读取,因为此thread可能会因此读到中间状态的值。所以,另一个thread必须安排事件派发thread去读取该值并将值返回给此thread。

 

注意,只靠另一个thread同步访问JSlider对象是不够的。Swing的内部机制并不是同步访问,所以两个thread还是同时地在访问slider的内部状态。要记得lock是协同运作的:如果所有的thread都没有尝试取得lock,race condition还是会发生。

 

感觉上这样的限制太过强烈:JSlider的值只是单一的变量且可以被轻易地做成volatile。实际上并不是这样。在Swing组件内各种事物的值可能是非常复杂的。许多Swing组件都遵循着mode-view-controller这种design pattern,且在此model被时间派发thread更新的时候从某个thread去访问这些组件是非常危险的。即使是最简单的SWing组件也带有复杂的状态,决不能接受从不是事件派发的thread中调用它们的任何一个method。

 

因此,所有对Swing对象的调用都必须从事件派发thread发出。它是Swing用来在内部改变对象状态的thread,只要你对Swing对象的调用都是来自该thread,就不会有race condition。这个规则有以下四个例外:

  1. 还没有显示出的Swing对象可以被任何的thread创建于操作。这代表你可以在任何thread中创建GUI对象,若一旦对象已经显示,它们就仅能由事件派发thread来访问。GUI对象是在它的父frame的show()方法被调用的时候显示的。
  2. repaint()方法可以从任何的thread调用。
  3. invokeLater()方法可以从任何的thread调用。
  4. invokeAndWait()方法可以从任何不是事件派发的thread调用

-----------------------------------------------------------------------------------------------------

Swing class已经确保所有的回调都会发生在事件派发thread上。

转载地址:http://zprxo.baihongyu.com/

你可能感兴趣的文章
openoffice启动8100端口
查看>>
cnetos 6.0下Chage的使用方法来提升系统安全级别
查看>>
tomcat启动没有8080端口
查看>>
ubuntu 16.04 安装lamp
查看>>
Javascript的匿名函数
查看>>
OC中类的属性与成员变量的区别
查看>>
SMTP命令邮件投递(无身份认证)
查看>>
Nginx + MySQL + PHP + Xcache + Memcached
查看>>
使用Windows live movie maker轻松与朋友分享视频
查看>>
我的友情链接
查看>>
数据库死锁的类型
查看>>
找水王
查看>>
grep及正则表达式
查看>>
MongoDB常用命令大全
查看>>
Python程序的执行过程
查看>>
Proxmox-VE搭配Ceph存储组建高可用虚拟化平台
查看>>
前端基础---JavaScript
查看>>
Linux关于大于2T的磁盘分区格式化
查看>>
lamp系列-MySQL主从复制原理视频(老男孩出品)
查看>>
堆和栈的区别
查看>>