`
wolfmaster
  • 浏览: 154393 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

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 String doorState = "";//表示门的状态,有“开”和“关”两种

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

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

    public String getDoorState() {
        return this.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 {
    public void doorEvent(DoorEvent event) {
        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 {
    public void doorEvent(DoorEvent event) {
        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.*;
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 定义的一些方法能直接查询本机操作系统。

========以上摘自http://hi.baidu.com/dobodo/blog/item/172e2c51657b2c8c8d54301a.html========


本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/xiaolang85/archive/2010/02/22/5316859.aspx

分享到:
评论

相关推荐

    Java事件处理机制(自定义事件)实例详解

    主要介绍了Java事件处理机制(自定义事件)实例详解的相关资料,需要的朋友可以参考下

    java异常处理机制

    异常的概念 异常的捕获与处理 自定义异常等

    springboot工程自定义response注解、自定义规范化返回数据结构

    springboot工程通过自定义response注解、java反射机制、自定义java拦截器、自定义功能类实现WebMvcConfigurer接口等功能,实现自定义规范化返回数据结构。

    Java编程实训 Java开发基础入门教程1-09 Java异常处理机制和调试(共32页).pptx

    【完整课程列表】 ...09 Java异常处理机制和调试(共32页).ppt 10 java集合框架(共27页).ppt 11 JAVA UI swing编程基础(共39页).ppt 12 AWT布局管理器(共14页).ppt 13 AWT事件处理模型(共29页).ppt

    java 自定义异常-编译异常和运行异常

    java 自定义异常--编译异常和运行...虽说Java类库已经提供很多可以直接处理异常的类,但是有时候为了更加地捕获和处理异常以呈现更好的用户体验,需要开发者自定义异常。本文是探讨如何自定义异常以及使用自定义的异常

    深入探索Spring事件监听机制:技术与应用

    除此之外,Spring也支持自定义事件,提供灵活性来处理特定业务需求。 事件监听器是对事件响应的处理器。它们可以通过实现ApplicationListener接口或使用@EventListener注解来定义。这些监听器关注特定类型的事件,...

    Java编程实训 Java开发基础入门教程1-13 AWT事件处理模型(共29页).ppt

    Java编程实训 Java开发基础入门教程1-09 Java异常处理机制和调试(共32页).ppt Java编程实训 Java开发基础入门教程1-10 java集合框架(共27页).ppt Java编程实训 Java开发基础入门教程1-11 JAVA UI swing编程基础...

    Java编程实训 Java开发基础入门教程1-03 自定义方法(共14页).ppt

    Java编程实训 Java开发基础入门教程1-09 Java异常处理机制和调试(共32页).ppt Java编程实训 Java开发基础入门教程1-10 java集合框架(共27页).ppt Java编程实训 Java开发基础入门教程1-11 JAVA UI swing编程基础...

    Java异常处理讲义.docx

    Java给我们提供了一套完善的异常处理机制来检查和解决可能出现的错误,以...在本文中,将带领大家来学习Java的异常处理机制,包括异常机制、异常类型、如何捕获异常、如何抛出异常以及如何创建自定义异常等核心内容。

    Java异常处理.md

    Java异常处理机制是一种用于有效管理程序运行时错误的方法。在Java中,通过try、catch和finally语句来实现异常捕获与处理: 1. **基础异常捕获**: - 当代码执行过程中出现如除数为零的`ArithmeticException`等...

    netty 数据分包、组包、粘包处理机制(部分)1

    netty 数据分包、组包、粘包处理机制(部分)1

    Java的异常处理

    异常处理的基础知识 异常处理机制 自定义异常类 异常处理应注意的问题

    深入理解java类加载机制

    此外,我们还会探讨Java程序的类加载器和双亲委派机制,以及自定义类加载器和类卸载的实现原理和应用方法。 总的来说,本资源将为Java程序员提供全面的Java字节码和类加载原理和实践经验。通过学习本资源,开发人员将...

    java源码包---java 源码 大量 实例

     Java波浪文字,一个利用Java处理字符的实例,可以设置运动方向参数,显示文本的字符数组,高速文本颜色,显示字体的 FontMetrics对象,得到Graphics实例,得到Image实例,填充颜色数组数据,初始化颜色数组。...

    Java异常是Java提供的一种识别及响应错误的一致性机制,Java异常机制可以使程序中异常处理代码和正常业务代码分离,保证程序

    Java异常处理 声明异常 抛出异常 捕获异常 如何选择异常类型 常见异常处理方式 直接抛出异常 封装异常再抛出 捕获异常 自定义异常 try-catch-finally try-with-resource Java异常常见面试题 1. Error 和 Exception ...

    Java实验报告——用户登录系统

    1、实验项目名称:用户登录系统 2、实验要求: (1)了解自定义异常的用法; (2)熟悉异常处理机制及创建自定义异常。 3、使用:在了解Java异常机制的基础上自己编写的实验报告。

    Java自定义异常

    虽说Java类库已经提供很多可以直接处理异常的类,但是有时候为了更加地捕获和处理异常以呈现更好的用户体验,需要开发者自定义异常。本文是探讨如何自定义异常以及使用自定义的异常。  在进行程序开发的过程中,...

    Java语言程序设计实验指导书

    Java语言程序设计实验指导书 理论教材:《Java程序设计标准教程》 第1版 邱加永 人民邮电...9 图形用户界面 用图形界面工具,结合事件处理机制,实现一个可视化的计算器。 10 JDBC基础 使用JDBC方式执行数据库基本操作

    Java Spring框架面试题大全

    开发者可以自定义事件类和事件监听器,并通过应用程序上下文的publishEvent方法来发布事件,Spring框架会自动分发事件给所有对该事件感兴趣的监听器进行处理。事件监听机制帮助开发者更好地解耦代码,提高应用程序的...

    JAVA上百实例源码以及开源项目源代码

     Java波浪文字,一个利用Java处理字符的实例,可以设置运动方向参数,显示文本的字符数组,高速文本颜色,显示字体的 FontMetrics对象,得到Graphics实例,得到Image实例,填充颜色数组数据,初始化颜色数组。...

Global site tag (gtag.js) - Google Analytics