`

理解Java的事件机制

阅读更多
Java中的事件机制的参与者有3种角色:

1.event object:事件状态对象,用于listener的相应的方法之中,作为参数,一般存在与listerner的方法之中

2.event source:具体的事件源,比如说,你点击一个button,那么button就是event source,要想使button对某些事件进行响应,你就需要注册特定的listener。

3.event listener:具体的对监听的事件类,当它监听到event object产生的时候,它就调用相应的方法,进行处理。


先看看jdk提供的event包:
public interface EventListener:所有事件侦听器接口必须扩展的标记接口。
public class EventObject extends Object implements Serializable
所有事件状态对象都将从其派生的根类。 所有 Event 在构造时都引用了对象 "source",在逻辑上认为该对象是最初发生有关 Event 的对象。

在Java2处理事件时,没有采用dispatchEvent()-postEvent()-handleEvent()方式,采用了监听器类,每个事件类都有相关联的监听器接口。事件从事件源到监听者的传递是通过对目标监听者对象的Java方法调用进行的。

对每个明确的事件的发生,都相应地定义一个明确的Java方法。这些方法都集中定义在事件监听者(EventListener)接口中,这个接口要继承 java.util.EventListener。 实现了事件监听者接口中一些或全部方法的类就是事件监听者。

伴随着事件的发生,相应的状态通常都封装在事件状态对象中,该对象必须继承自java.util.EventObject。事件状态对象作为单参传递给应响应该事件的监听者方法中。发出某种特定事件的事件源的标识是:遵从规定的设计格式为事件监听者定义注册方法,并接受对指定事件监听者接口实例的引用。

首先问个问题:您熟悉java.util.EventObject 和java.util.EventListener两个类以及他们已有的子类吗?

如果你已经能够熟练使用jdk为我们提供的事件监听器,并且很熟悉MouseEvent, KeyEvent, WindowEvent等等这些jdk为我们准备好的事件,那么想必你对java的事件机制已经有所理解。但是也许你还是觉得虽然用起来没什么问题,但是原理还是有些糊涂,那么下面我们再进一步自己实现这些事件和监听器,即自定义事件。

其实自定义事件在java中很有用处,我们有的时候想让自己的程序产生一个事件,但有不希望(或者不可能)用鼠标,键盘之类的输入设备进行操作,比如你写一个应用程序,在这个程序中一旦收到邮件就对邮件进行相关处理,对于“收到邮件”这个事件,jdk中就没有定义。对于这样的事件,以及对于这样的事件的监听器,我们只能自己动手完成了。

那么下面就以实例开始我们这个“创新”的过程:首先,类EventObject作为父类用来生成我们自己的事件类,接口EventListener用来实现我们自己的监听器;剩下的事情就是如何注册这些事件以及测试他们了。

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

(1)通过DoorEvent.java文件创建DoorEvent类,这个类继承EventObject。
 
/**
 * 定义事件对象,必须继承EventObject
 */
package test;
import java.util.EventObject;

public class DoorEvent extends EventObject {

	private static final long serialVersionUID = -1748946803753053501L;
	private String doorState = ""; // 表示门的状态,有“开”和“关”两种

	public DoorEvent(Object source, String doorState) {
		super(source);
		this.doorState = doorState;
	}

	public String getDoorState() {
		return doorState;
	}

	public void setDoorState(String doorState) {
		this.doorState = doorState;
	}

}
 


----------------------------------------------------------------------------------------------------------------------------
(2)定义新的事件监听接口,该接口继承自EventListener;该接口包含对doorEvent事件的处理程序:
/**
 * 定义监听接口,负责监听DoorEvent事件
 */
package test;
import java.util.EventListener;

public interface DoorListener extends EventListener {

	public void doorEvent(DoorEvent event);
}



通过上面的接口我们再定义事件监听类,这些类具体实现了监听功能和事件处理功能。
/**   
 * 该类为 门1监听接口的实现,做具体的开门,关门动作   
 */
package test;

public class DoorListener1 implements DoorListener {

	@Override
	public void doorEvent(DoorEvent event) {
		// TODO Auto-generated method stub
		if (event.getDoorState() != null && event.getDoorState().equals("open")) {
			System.out.println("门1打开");
		} else {
			System.out.println("门1关闭");
		}
	}

}


/**   
 * 该类为 门2监听接口的实现,做具体的开门,关门,以及开灯,关灯动作   
 */
package test;

public class DoorListener2 implements DoorListener {

	@Override
	public void doorEvent(DoorEvent event) {
		// TODO Auto-generated method stub
		if (event.getDoorState() != null && event.getDoorState().equals("open")) {
			System.out.println("门2打开,同时打开走廊的灯");
		} else {
			System.out.println("门2关闭,同时关闭走廊的灯");
		}
	}

}


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


(3)通过DoorManager.java创造一个事件源类,它用一个Collection listeners对象来存储所有的事件监听器对象,存储方式是通过addDoorListener(..)这样的方法。 notifyListeners(..)是触发事件的方法,用来通知系统:事件发生了,你调用相应的处理函数吧。
/**   
 * 事件源对象,在这里你可以把它想象成一个控制开门关门的遥控器,   
 * (如果是在swing中,就类似一个button)   
 */
package test;

import java.util.Collection;
import java.util.HashSet;
import java.util.Iterator;

public class DoorManager {
	private Collection listeners;

	/**
	 * 添加事件
	 * 
	 * @param listener
	 *            DoorListener
	 */
	public void addDoorListener(DoorListener listener) {
		if (listeners == null) {
			listeners = new HashSet();
		}
		listeners.add(listener);
	}

	/**
	 * 移除事件
	 * 
	 * @param listener
	 *            DoorListener
	 */
	public void removeDoorListener(DoorListener listener) {
		if (listeners == null)
			return;
		listeners.remove(listener);
	}

	/**
	 * 触发开门事件
	 */
	protected void fireWorkspaceOpened() {
		if (listeners == null)
			return;
		DoorEvent event = new DoorEvent(this, "open");
		notifyListeners(event);
	}

	/**
	 * 触发关门事件
	 */
	protected void fireWorkspaceClosed() {
		if (listeners == null)
			return;
		DoorEvent event = new DoorEvent(this, "close");
		notifyListeners(event);
	}

	/**
	 * 通知所有的DoorListener
	 */
	private void notifyListeners(DoorEvent event) {
		Iterator iter = listeners.iterator();
		while (iter.hasNext()) {
			DoorListener listener = (DoorListener) iter.next();
			listener.doorEvent(event);
		}
	}
}


(4)好了,最后写一个测试程序测试一下我们自定义的事件吧,这段程序应该不难理解吧:)
  /**  
 * 主程序,就想象成要开门的哪个人  
 */
package test;

public class DoorMain {
	public static void main(String[] args) {
		DoorManager manager = new DoorManager();
		manager.addDoorListener(new DoorListener1());// 给门1增加监听器
		manager.addDoorListener(new DoorListener2());// 给门2增加监听器
		// 开门
		manager.fireWorkspaceOpened();
		System.out.println("我已经进来了");
		// 关门
		manager.fireWorkspaceClosed();
	}
}
 

----------------------------------------------------------------------------------------------------------------------------
运行DoorMain

门1打开
门2打开,同时打开走廊的灯
我已经进来了
门1关闭
门2关闭,同时关闭走廊的灯

=================================================================================================

下面我们看一个jdk内部是如何处理事件机制的,你可以和上面的自定义事件做一个比较,你会高兴的发现机制是一样的。
   /**  
   * java swing的监听器,实现ActionListener接口,注意参数:(事件状态类:ActionEvent)  
   *   
   */  
   package test;   
      
   import java.awt.*;   
   import java.awt.event.*;   
   import javax.swing.*;   
  
  class SimpleListener implements ActionListener {   
       /*  
        * 利用该类来监听事件源产生的事件,利用响应机制  
        */  
       public void actionPerformed(ActionEvent e) {   
           String buttonName = e.getActionCommand();   
           if (buttonName.equals("按钮1"))   
               System.out.println("按钮1 被点击");   
     
       }   
   }   
     
     
   public class ActionTest {   
       private static JFrame frame; // 定义为静态变量以便main使用   
       private static JPanel myPanel; // 该面板用来放置按钮组件   
       private JButton button1; // 这里定义按钮组件   
     
     
       public ActionTest() { // 构造器, 建立图形界面   
           // 新建面板   
           myPanel = new JPanel();   
           // 新建按钮   
           button1 = new JButton("按钮1"); // 新建按钮1   
           // 建立一个actionlistener让按钮1注册,以便响应事件   
           SimpleListener ourListener = new SimpleListener();   
           button1.addActionListener(ourListener);   
           myPanel.add(button1); // 添加按钮到面板   
             
       }   
     
     
       public static void main(String s[]) {   
           ActionTest gui = new ActionTest(); // 新建Simple1组件   
     
           frame = new JFrame("Simple1"); // 新建JFrame   
           // 处理关闭事件的通常方法   
           frame.addWindowListener(new WindowAdapter() {   
               public void windowClosing(WindowEvent e) {   
                   System.exit(0);   
               }   
           });   
     
           frame.getContentPane().add(myPanel);   
           frame.pack();   
           frame.setVisible(true);   
       }   
   }  


在这里,我们再看一下java中的事件机制的参与者的3种角色:

我们定义了一个SimpleListener 实现ActionListener接口,


1.event object:事件状态对象,用于listener的相应的方法之中。用了jdk提供的ActionEvent,不需要我们自己定义。

2.event source:具体的事件源,就是哪个button,,注册特定的SimpleListener。

3.event listener:具体的对监听的事件类,当它监听到event

object产生的时候,它就调用相应的方法,进行处理。这里是我们自己定义的SimpleListener。


是不是和上面自定义的事件在机制上完全一致呢?Yes
---------------------------------------------------------------------------------------------------------------------------
这里你也许会问,为什么event object不需要我们自己定义呢?你可以想一下,这是一个表示“事件状态变化”的类,你能扑获“鼠标变化”

吗?这好象和平台有关的低层编码了,所有所不可能扑获,也没有必要去扑获,这些jdk已经给我们实现了。简单的看一下ActionEvent这个类

,它继承了java.awt.AWTEvent, 在这个类的构造方法源码如下:
static {
        /* ensure that the necessary native libraries are loaded */
Toolkit.loadLibraries();
        if (!GraphicsEnvironment.isHeadless()) {
            initIDs();
        }
    }

我们在看jdk官方的解释:

Toolkit是 Abstract Window Toolkit 的所有实际实现的抽象超类。Toolkit 的子类被用于将各种组件绑定到特定本机工具包实现。大多数应

用程序不应直接调用该类中的任何方法。Toolkit 定义的方法是“胶合剂”,将 java.awt 包中与平台无关的类与 java.awt.peer 中的对应物

连接起来。Toolkit 定义的一些方法能直接查询本机操作系统。
分享到:
评论

相关推荐

    Java事件机制例子

    在Java编程语言中,事件处理是图形用户界面(GUI)...理解这一机制对于开发具有用户友好界面的Java应用程序至关重要。通过实践"JavaEventDemo"这样的例子,新手可以更好地掌握事件处理的概念,并将其应用到实际项目中。

    JAVA事件处理机制

    Java事件处理机制是Java编程中一个关键的概念,它主要用于用户界面(UI)应用程序,使得程序能够响应用户的操作,如点击按钮、选择菜单项等。在Java中,事件处理主要涉及三个核心组件:事件源(Event Source)、事件...

    java中的事件机制

    Java中的事件机制是Java编程中一个重要的概念,它基于观察者模式(Observer Pattern)来实现对象之间的通信。观察者模式是一种行为设计模式,...理解并熟练运用这一机制,对于提高Java编程效率和代码质量具有重要意义。

    java事件处理机制Demo

    总结来说,"java事件处理机制Demo"是一个关于如何在Java项目中使用观察者模式和事件处理机制的实例,通过学习和实践这个示例,开发者可以更好地理解和掌握这两项核心技术,从而提升程序的交互性和可维护性。

    Java事件处理机制教学方法探讨.pdf

    为了让初学者更好理解Java事件处理机制及其编程技术,黄增喜、王晓明和于春三位作者提出了一个基于核心概念解析、事件处理流程分析以及角色分工清晰化的教学思路,并给出了相应的教学方法。以下是对这些知识点的详细...

    初步接触Java事件机制

    Java事件机制是Java编程中一个重要的概念...通过理解并熟练运用Java事件机制,开发者可以创建响应用户操作的动态应用程序,提升用户体验。在实际开发中,结合源码阅读和工具使用,可以更深入地理解和优化事件处理流程。

    java事件处理机制

    Java事件处理机制是Java GUI编程中的重要组成部分,它允许用户与图形界面进行交互。在Java中,事件处理涉及事件源...通过理解并熟练掌握这些概念和机制,开发者可以构建出丰富的、响应用户操作的Java GUI应用程序。

    JavaGUI事件处理机制

    Java GUI事件处理机制是Java应用程序开发中的重要组成部分,主要用于响应用户在图形用户界面中的各种操作。这一机制采用了委托事件模型,使得...理解和掌握这一机制,对于编写高效、易维护的Java GUI应用程序至关重要。

    用Java事件处理机制实现录制回放功能

    总结来说,实现Java GUI应用的录制回放功能,需要理解Java事件处理机制的原理,包括事件监听器、适配器和事件模型的演变。通过捕获和回放用户的操作,可以有效地进行自动化测试和应用演示,提高开发和调试效率。

    java事件处理机制详解

    Java事件处理机制是一种用于在组件之间传递和处理事件的机制,它是GUI编程中不可或缺的一部分。在Java中,事件处理涉及到三个关键角色:事件对象...理解和掌握这一机制,对于编写功能丰富的Java GUI应用程序至关重要。

    项目教学法的设计——以java事件处理机制为例.pdf

    项目教学法的设计——以java事件处理机制为例 在计算机科学和技术教育中,项目教学法是一种...通过java事件处理机制的教学,我们可以让学生更好地理解java事件处理机制的知识,并提高学生的实践能力和解决问题的能力。

    一张图让你通俗易懂得理解java反射机制!

    一张图让你通俗易懂得理解java反射机制!

    事件驱动模型实例详解(Java篇)

    ### 事件驱动模型实例详解(Java篇) #### 1. 事件驱动模型概念解析 事件驱动模型是现代软件开发中一种关键的...通过深入理解事件源、侦听器和事件处理程序的概念,开发者可以构建出更加灵活、交互性更强的软件系统。

    java反射机制简单理解

    Java 反射机制简单理解 Java 反射机制是一种强大的机制,它允许程序在运行时检查和修改自身的结构和行为。Java 反射机制简单理解,非常容易理解的 Java 反射机制。 Java 反射机制的主要应用场景有两种:一种是 ...

    java面试题--反射机制

    这是因为Java反射机制不仅是Java语言的一项重要特性,也是理解Java动态特性的关键所在。通过本文,我们将深入探讨Java反射机制的核心概念、基本原理及其应用场景。 #### 二、Java反射机制简介 Java反射机制允许...

    深入理解java异常处理机制Java开发Java经验技巧共

    这篇19页的PDF文档《深入理解java异常处理机制Java开发Java经验技巧共》可能涵盖了Java异常处理的基本概念、最佳实践以及高级用法。在Java中,异常处理通过五个关键字来实现:try、catch、finally、throw和throws。 ...

    Java GUI编程中的事件处理机制详细讲解 (1)

    ### Java GUI 编程中的事件处理机制详细讲解 在Java GUI编程中,事件处理机制是构建用户界面交互的核心部分。本文将围绕“Java GUI编程中的事件处理...希望本文能帮助读者更好地理解并应用Java GUI中的事件处理机制。

    Java事件处理机制教学方法探讨.zip

    Java事件处理机制是Java编程中一个重要的组成部分,它主要用于用户界面(UI)应用程序,使得程序能够响应用户的操作,如点击按钮、选择菜单项或者在文本框中输入文字等。本教学方法探讨旨在深入理解并有效地教授Java...

    JAVA事件处理指南

    本指南将深入探讨Java事件处理的原理、机制以及相关实践。 首先,Java事件处理基于Observer模式,即观察者模式。在这种模式下,事件源(通常是GUI组件,如按钮、文本框等)是被观察的对象,当它们的状态发生改变时...

Global site tag (gtag.js) - Google Analytics