- 浏览: 143799 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
EclipseEye:
fair_jm 写道不错 蛮详细的 谢谢分享
SWT/JFace专题 --- SWT中Display和多线程 -
fair_jm:
不错 蛮详细的 谢谢分享
SWT/JFace专题 --- SWT中Display和多线程
在 [url]SWT中org.eclipse.swt.SWTError: No more handles[/url] 一文中有介绍,但最近又遇到这个问题。
其实,在那篇文章中提到的用Map 模拟注册表 缓存的方法是很凑效的,之所以又出现这个异常是因为一时疏漏 在一个方法中new Image后没有及时dispose该image(由于该image在Graphics中调用属于临时image 没必要放到map中,正确的方法是用完后dispose),而该方法又被频繁调用,这就又导致了该异常。
虽然最后的解决方案很简单,就添加一行image.dispose就ok了,不过由于项目比较大,想快速定位到错误的位置并不容易。
所以,如何跟踪软件(这里针对Eclipse RCP程序进行说明)的handler消耗呢?
其实打开Windows的任务管理器,可以在“性能”子页下的 句柄总数找到句柄的描述信息,但是如果我们想要知道RCP程序中具体的handler消耗,这个显然是不够的。
用Sleak工具可以快速定位句柄的消耗情况,来快速查出被过度消耗的handler出现在何处:
Sleak工具主要就是 org.eclipse.swt.sleak.jar 这个jar包,可以在http://www.eclipse.org/swt/tools.php 下载到。
http://www.eclipse.org/articles/swt-design-2/sleak.htm 官网上sleak的介绍
该包就有两个有效类Sleak.java和SleakView.java
================
可以把以上两个类copy到自己的工程下面,扩展一个view 结点,指定实现类为SleakView 就可以了。
这样在你运行你的项目的时候就可看到Sleak View了,能够很容易跟踪handler的消耗。
其实,在那篇文章中提到的用Map 模拟注册表 缓存的方法是很凑效的,之所以又出现这个异常是因为一时疏漏 在一个方法中new Image后没有及时dispose该image(由于该image在Graphics中调用属于临时image 没必要放到map中,正确的方法是用完后dispose),而该方法又被频繁调用,这就又导致了该异常。
虽然最后的解决方案很简单,就添加一行image.dispose就ok了,不过由于项目比较大,想快速定位到错误的位置并不容易。
所以,如何跟踪软件(这里针对Eclipse RCP程序进行说明)的handler消耗呢?
其实打开Windows的任务管理器,可以在“性能”子页下的 句柄总数找到句柄的描述信息,但是如果我们想要知道RCP程序中具体的handler消耗,这个显然是不够的。
用Sleak工具可以快速定位句柄的消耗情况,来快速查出被过度消耗的handler出现在何处:
Sleak工具主要就是 org.eclipse.swt.sleak.jar 这个jar包,可以在http://www.eclipse.org/swt/tools.php 下载到。
http://www.eclipse.org/articles/swt-design-2/sleak.htm 官网上sleak的介绍
该包就有两个有效类Sleak.java和SleakView.java
import org.eclipse.swt.*; import org.eclipse.swt.graphics.*; import org.eclipse.swt.widgets.*; import java.io.*; /** * Instructions on how to use the Sleak tool. * * A) If you have access to the place where the display is created (most standalone applications): * * Modify the creation of the Display as follows: * * DeviceData data = new DeviceData(); * data.tracking = true; * Display display = new Display(data); * Sleak sleak = new Sleak(); * sleak.open(); * Shell shell = new Shell(display); * * B) If you do not have access to the place where the display is created (Eclipse): * * 1) This tool will be referenced from the Device class so it needs to be copied into the same project. * Copy the package org.eclipse.swt.internal.tools to a source folder in the org.eclipse.swt project. * * 2) This tool relies on debug information collected by the Device class. Modify the Device class to * turn DEBUG on: * * public static boolean DEBUG = true; * * 3) The Sleak tool must be launched from the constructor of the Device class. * Add the following lines to the end of the Device constructor: * * if (this.getClass().getName().equals("org.eclipse.swt.widgets.Display")) { * org.eclipse.swt.internal.tools.Sleak sleak = new org.eclipse.swt.internal.tools.Sleak(); * sleak.open(); * } * * 4) Create a new swt.jar for the platform you wish to test. This is done by running the build.xml script * in the appropriate windowing system fragment. Select the ws/$ws/swt.jar target. * * 5) Copy the new swt.jar to the appropriate location. When using Sleak with Eclipse, this is * eclipse/org.eclipse.swt.<ws>_<version>/ws/<ws> * * 6) Launch your application and the Sleak GUI will be displayed. Click on the Snap button to * capture the current resource state. Click on the Diff button to compare the current state to * the previous snapshot state. * */ public class Sleak { Display display; Composite parent; List list; Canvas canvas; Button start, stop, check; Text text; Label label; Object [] oldObjects = new Object [0]; Error [] oldErrors = new Error [0]; Object [] objects = new Object [0]; Error [] errors = new Error [0]; public void open () { Display display = Display.getCurrent (); Shell shell = new Shell (display); shell.setText ("S-Leak"); open (shell); } public void open (Composite parent2) { Composite composite = new Composite(parent2, SWT.NONE); parent = composite; parent.addListener(SWT.Resize, new Listener() { public void handleEvent (Event e) { layout (); } }); this.display = Display.getCurrent (); list = new List (parent, SWT.BORDER | SWT.V_SCROLL); list.addListener (SWT.Selection, new Listener () { public void handleEvent (Event event) { refreshObject (); } }); text = new Text (parent, SWT.BORDER | SWT.H_SCROLL | SWT.V_SCROLL); canvas = new Canvas (parent, SWT.BORDER); canvas.addListener (SWT.Paint, new Listener () { public void handleEvent (Event event) { paintCanvas (event); } }); check = new Button (parent, SWT.CHECK); check.setText ("Stack"); check.addListener (SWT.Selection, new Listener () { public void handleEvent (Event e) { toggleStackTrace (); } }); start = new Button (parent, SWT.PUSH); start.setText ("Snap"); start.addListener (SWT.Selection, new Listener () { public void handleEvent (Event event) { refreshAll (); } }); stop = new Button (parent, SWT.PUSH); stop.setText ("Diff"); stop.addListener (SWT.Selection, new Listener () { public void handleEvent (Event event) { refreshDifference (); } }); label = new Label (parent, SWT.BORDER); label.setText ("0 object(s)"); parent.addListener (SWT.Resize, new Listener () { public void handleEvent (Event e) { layout (); } }); check.setSelection (false); text.setVisible (false); Point size = parent.getSize (); parent.setSize (size.x / 2, size.y / 2); parent.setVisible (true); } void refreshLabel () { int colors = 0, cursors = 0, fonts = 0, gcs = 0, images = 0; int paths = 0, patterns = 0, regions = 0, textLayouts = 0, transforms= 0; for (int i=0; i<objects.length; i++) { Object object = objects [i]; if (object instanceof Color) colors++; if (object instanceof Cursor) cursors++; if (object instanceof Font) fonts++; if (object instanceof GC) gcs++; if (object instanceof Image) images++; if (object instanceof Path) paths++; if (object instanceof Pattern) patterns++; if (object instanceof Region) regions++; if (object instanceof TextLayout) textLayouts++; if (object instanceof Transform) transforms++; } String string = ""; if (colors != 0) string += colors + " Color(s)\n"; if (cursors != 0) string += cursors + " Cursor(s)\n"; if (fonts != 0) string += fonts + " Font(s)\n"; if (gcs != 0) string += gcs + " GC(s)\n"; if (images != 0) string += images + " Image(s)\n"; if (paths != 0) string += paths + " Paths(s)\n"; if (patterns != 0) string += patterns + " Pattern(s)\n"; if (regions != 0) string += regions + " Region(s)\n"; if (textLayouts != 0) string += textLayouts + " TextLayout(s)\n"; if (transforms != 0) string += transforms + " Transform(s)\n"; if (string.length () != 0) { string = string.substring (0, string.length () - 1); } label.setText (string); } void refreshDifference () { DeviceData info = display.getDeviceData (); if (!info.tracking) { MessageBox dialog = new MessageBox (parent.getShell(), SWT.ICON_WARNING | SWT.OK); dialog.setText (parent.getShell().getText ()); dialog.setMessage ("Warning: Device is not tracking resource allocation"); dialog.open (); } Object [] newObjects = info.objects; Error [] newErrors = info.errors; Object [] diffObjects = new Object [newObjects.length]; Error [] diffErrors = new Error [newErrors.length]; int count = 0; for (int i=0; i<newObjects.length; i++) { int index = 0; while (index < oldObjects.length) { if (newObjects [i] == oldObjects [index]) break; index++; } if (index == oldObjects.length) { diffObjects [count] = newObjects [i]; diffErrors [count] = newErrors [i]; count++; } } objects = new Object [count]; errors = new Error [count]; System.arraycopy (diffObjects, 0, objects, 0, count); System.arraycopy (diffErrors, 0, errors, 0, count); list.removeAll (); text.setText (""); canvas.redraw (); for (int i=0; i<objects.length; i++) { list.add (objects [i].toString()); } refreshLabel (); layout (); } void toggleStackTrace () { refreshObject (); layout (); } void paintCanvas (Event event) { canvas.setCursor (null); int index = list.getSelectionIndex (); if (index == -1) return; GC gc = event.gc; Object object = objects [index]; if (object instanceof Color) { if (((Color)object).isDisposed ()) return; gc.setBackground ((Color) object); gc.fillRectangle (canvas.getClientArea()); return; } if (object instanceof Cursor) { if (((Cursor)object).isDisposed ()) return; canvas.setCursor ((Cursor) object); return; } if (object instanceof Font) { if (((Font)object).isDisposed ()) return; gc.setFont ((Font) object); FontData [] array = gc.getFont ().getFontData (); String string = ""; String lf = text.getLineDelimiter (); for (int i=0; i<array.length; i++) { FontData data = array [i]; String style = "NORMAL"; int bits = data.getStyle (); if (bits != 0) { if ((bits & SWT.BOLD) != 0) style = "BOLD "; if ((bits & SWT.ITALIC) != 0) style += "ITALIC"; } string += data.getName () + " " + data.getHeight () + " " + style + lf; } gc.drawString (string, 0, 0); return; } //NOTHING TO DRAW FOR GC // if (object instanceof GC) { // return; // } if (object instanceof Image) { if (((Image)object).isDisposed ()) return; gc.drawImage ((Image) object, 0, 0); return; } if (object instanceof Path) { if (((Path)object).isDisposed ()) return; gc.drawPath ((Path) object); return; } if (object instanceof Pattern) { if (((Pattern)object).isDisposed ()) return; gc.setBackgroundPattern ((Pattern)object); gc.fillRectangle (canvas.getClientArea ()); gc.setBackgroundPattern (null); return; } if (object instanceof Region) { if (((Region)object).isDisposed ()) return; String string = ((Region)object).getBounds().toString(); gc.drawString (string, 0, 0); return; } if (object instanceof TextLayout) { if (((TextLayout)object).isDisposed ()) return; ((TextLayout)object).draw (gc, 0, 0); return; } if (object instanceof Transform) { if (((Transform)object).isDisposed ()) return; String string = ((Transform)object).toString(); gc.drawString (string, 0, 0); return; } } void refreshObject () { int index = list.getSelectionIndex (); if (index == -1) return; if (check.getSelection ()) { ByteArrayOutputStream stream = new ByteArrayOutputStream (); PrintStream s = new PrintStream (stream); errors [index].printStackTrace (s); text.setText (stream.toString ()); text.setVisible (true); canvas.setVisible (false); } else { canvas.setVisible (true); text.setVisible (false); canvas.redraw (); } } void refreshAll () { oldObjects = new Object [0]; oldErrors = new Error [0]; refreshDifference (); oldObjects = objects; oldErrors = errors; } void layout () { Rectangle rect = parent.getClientArea (); int width = 0; String [] items = list.getItems (); GC gc = new GC (list); for (int i=0; i<objects.length; i++) { width = Math.max (width, gc.stringExtent (items [i]).x); } gc.dispose (); Point size1 = start.computeSize (SWT.DEFAULT, SWT.DEFAULT); Point size2 = stop.computeSize (SWT.DEFAULT, SWT.DEFAULT); Point size3 = check.computeSize (SWT.DEFAULT, SWT.DEFAULT); Point size4 = label.computeSize (SWT.DEFAULT, SWT.DEFAULT); width = Math.max (size1.x, Math.max (size2.x, Math.max (size3.x, width))); width = Math.max (64, Math.max (size4.x, list.computeSize (width, SWT.DEFAULT).x)); start.setBounds (0, 0, width, size1.y); stop.setBounds (0, size1.y, width, size2.y); check.setBounds (0, size1.y + size2.y, width, size3.y); label.setBounds (0, rect.height - size4.y, width, size4.y); int height = size1.y + size2.y + size3.y; list.setBounds (0, height, width, rect.height - height - size4.y); text.setBounds (width, 0, rect.width - width, rect.height); canvas.setBounds (width, 0, rect.width - width, rect.height); } }==============
import org.eclipse.swt.widgets.Composite; import org.eclipse.ui.part.*; /** * This sample class demonstrates how to plug-in a new * workbench view. The view shows data obtained from the * model. The sample creates a dummy model on the fly, * but a real implementation would connect to the model * available either in this or another plug-in (e.g. the workspace). * The view is connected to the model using a content provider. * <p> * The view uses a label provider to define how model * objects should be presented in the view. Each * view can present the same model objects using * different labels and icons, if needed. Alternatively, * a single label provider can be shared between views * in order to ensure that objects of the same type are * presented in the same way everywhere. * <p> */ public class SleakView extends ViewPart { Composite parent = null; Sleak sleak = null; /** * The constructor. */ public SleakView() { } /** * This is a callback that will allow us * to create the viewer and initialize it. */ public void createPartControl(Composite parent) { this.parent = parent; sleak = new Sleak (); sleak.open(parent); sleak.layout(); } /** * Passing the focus request to the viewer's control. */ public void setFocus() { } }
================
可以把以上两个类copy到自己的工程下面,扩展一个view 结点,指定实现类为SleakView 就可以了。
<extension point="org.eclipse.ui.views"> <category id="org.xxx.xxx.rcp.category" name="Sleak 工具"> </category> <view category="org.xxx.xxx.rcp.category" class="org.xxx.xxx.rcp.sleak.SleakView" id="org.eclipse.swt.sleak.views.SleakView" name="Sleak00kaelS" restorable="true"> </view> </extension>
这样在你运行你的项目的时候就可看到Sleak View了,能够很容易跟踪handler的消耗。
发表评论
-
再说SWT中的滚动面板ScrolledComposite实现
2013-06-19 15:43 2255记得以前写过一篇关于滚动面板的文章 SWT中 Scrolle ... -
OSGi参考资料
2013-04-18 01:11 651基于 OSGi 的面向服务的组件编程 探索 OSGi 框架的组 ... -
CDT(编辑、调试)参考资料
2013-04-17 02:15 1072CDT编辑器 --------- 构建基于 CDT 的编辑器, ... -
Workspace Resource框架专题(3)处理工作空间资源更改事件
2013-04-17 01:44 13253 处理工作空间资源更改事件 工作空间API允许工具对它 ... -
Workspace Resource框架专题(2)workspace 框架API
2013-04-17 01:27 14192 工作空间API 本 ... -
Workspace Resource框架专题(1)Resource的概念
2013-04-17 01:12 13341 Resource的概念 如 ... -
如何恢复断点及Marker
2013-03-05 00:41 0如何恢复断点及Marker -
深入Workbench框架
2013-03-01 02:10 1628深入Workbench框架(结合UIPersistent) 1 ... -
Eclipse插件开发中的Action
2013-02-24 23:10 1960插入点用来定义菜单出 ... -
Eclipse开发中编辑器(Editors)和视图(View)总结
2013-02-24 22:58 28171.视图(Views) 视图( ... -
SWT/JFace专题 --- 对话框向导(Dialogs Wizards)
2013-02-24 22:42 2103对话框向导(Dialogs Wizar ... -
SWT/JFace专题 --- JFace
2013-02-24 22:37 1568JFace JFace是基于SWT的一套图形工具包,它没有为 ... -
SWT/JFace专题 --- SWT中Display和多线程
2013-02-24 15:25 3155Display 1.Display 的创建 一个SWT程序 ... -
SWT/JFace专题 --- SWT API 结构
2013-02-23 18:31 1039SWT API 结构 1.布局类(l ... -
Eclipse启动过程(源码级剖析)
2013-02-20 03:24 3357双击eclipse安装目录下的eclipse.exe运行后,会 ... -
SWT/JFace专题 --- SWT结构浅析
2013-02-23 17:02 983SWT技术是一套基于Java的 ... -
Eclipse平台体系结构
2013-02-21 23:56 17681.Eclipse平台体系结构 1 ... -
RCP平台架构
2013-02-23 14:11 1399RCP 富客户端通常是指具有独立用户界面的客户端程序。富客户 ... -
Ant构建脚本相关
2013-02-18 01:26 0Ant构建脚本相关 -
CDT源码架构研究
2013-02-18 01:24 0CDT源码架构研究
相关推荐
org.eclipse.swt.SWTError: No more handles [MOZILLA_FIVE_HOME=''] (java.lang.UnsatisfiedLinkError: Could not load SWT library. Reasons: no swt-mozilla-gtk-4335 in java.library.path no swt-mozilla-gtk ...
解决org.eclipse.swt.SWTError: No more handles [MOZILLA_FIVE_HOME=''] (java.lang.UnsatisfiedLinkError: Could not load SWT library. Reasons: no swt-mozilla-gtk-4335 in java.library.path
有时候会弹出一个方框,里面的内容是Unhandled event loop exception No more handles,当你点击OK的时候,又会出来一个方框,里面第一句话是 An SWT error has occurred,并且诱导你关闭eclipse,严重影响了我们对...
Matab GUI 的源代码,经过这个源代码的使用就可以很好掌握Matlab GUI的编写。
Contents About the Author...............................................................................................xix About the Technical Reviewer and Contributing Author.................xxi ...
文章目录源代码报错原因解决 源代码 # 需求:再添加一个城市的温度变化 #导入工具 import matplotlib.pyplot as plt import random #创建画布 plt.figure(figsize=(20,8),dpi = 80) #绘制图像,画出安徽省宣城市泾县...
This plugin does an in-memory scanning of process loaded modules checking if they were compiled with /SafeSEH, if so it can list the registered ... Error : There was an error reading module structure :-
no of handles 1 timer handle (`setTimeout(any, 1000)`) timer handle leaked at one of: at Test.t (/home/raynos/uber/leaked-handles/test/leak-timer.js:10:17) timer listener function any() {} no of ...
handles.figure1, handles.slider2. This | structure is created at GUI startup using GUIHANDLES and stored in | the figure s application data using GUIDATA. A copy of the structure | is passed to ...
:white_check_mark: Configuration change: Handles configuration changes :white_check_mark: Material Design: Not a fully Material Design App, but I am trying my best. :white_check_mark: Some custom ...
* Handles requests for the application home page. */ @Controller public class HomeController { private static final Logger logger = LoggerFactory.getLogger(HomeController.class); /** * ...
开源项目-libeclipse-memguard.zip,MemGuard: A library that handles sensitive values in memory.
kettle-linux环境下部署kettle,执行kitchen.sh文件报错后安装 libwebkitgtk,提示没有可用软件包libwebkitgtk 详细流程:https://blog.csdn.net/m0_37618809/article/details/81015492#commentBox
ocl.dll各个版本资源文件,有效解决Navicat is not able to create OCI handles的问题
Mozilla缺陷报告仓库特征分析,潘兴亮,胡燕,缺陷报告仓库管理一直是大型开源软件维护工作的关键问题之一。以Mozilla项目为例,通过收集和分析缺陷管理系统Bugzilla中的历史活动信
flex-object-handles flex中对象的移动 编译 放大 缩小的功能 早期的版本