posts - 51, comments - 17, trackbacks - 0, articles - 9
            BlogJava :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

          java 事件機制

          Posted on 2007-04-13 09:22 chenweicai 閱讀(10339) 評論(14)  編輯  收藏

          java中的事件機制的參與者有3種角色:

          1.event object:就是事件產(chǎn)生時具體的“事件”,用于listener的相應的方法之中,作為參數(shù),一般存在與listerner的方法之中

          2.event source:具體的接受事件的實體,比如說,你點擊一個button,那么button就是event source,這樣你必須使button對某些事件進行相應,你就需要注冊特定的listener,比如說MouseEvent之中的MouseClicked方法,這是他就必須有了add方法

          3.event listener:具體的對監(jiān)聽的事件類,當有其對應的event object產(chǎn)生的時候,它就調(diào)用相應的方法,進行處理。在windows程序設計里邊這種相應使用callback機制來實現(xiàn)的

          先看看jdk提供的event包:
          public interface EventListener:所有事件偵聽器接口必須擴展的標記接口。
          public class EventObject extends Object implements Serializable

          所有事件狀態(tài)對象都將從其派生的根類。 所有 Event 在構造時都引用了對象 "source",在邏輯上認為該對象是最初發(fā)生有關 Event 的對象。

                  在Java2處理事件時,沒有采用dispatchEvent()-postEvent()-handleEvent()方式,采用了監(jiān)聽器類,每個事件類都有相關聯(lián)的監(jiān)聽器接口。事件從事件源到監(jiān)聽者的傳遞是通過對目標監(jiān)聽者對象的Java方法調(diào)用進行的。

            對每個明確的事件的發(fā)生,都相應地定義一個明確的Java方法。這些方法都集中定義在事件監(jiān)聽者(EventListener)接口中,這個接口要繼承 java.util.EventListener。 實現(xiàn)了事件監(jiān)聽者接口中一些或全部方法的類就是事件監(jiān)聽者。

            伴隨著事件的發(fā)生,相應的狀態(tài)通常都封裝在事件狀態(tài)對象中,該對象必須繼承自java.util.EventObject。事件狀態(tài)對象作為單參傳遞給應響應該事件的監(jiān)聽者方法中。發(fā)出某種特定事件的事件源的標識是:遵從規(guī)定的設計格式為事件監(jiān)聽者定義注冊方法,并接受對指定事件監(jiān)聽者接口實例的引用。

          開始之前首先問個問題:您熟悉java.util.EventObject 和java.util.EventListener兩個類以及他們已有的子類嗎?

          如果你已經(jīng)能夠熟練使用jdk為我們提供的事件監(jiān)聽器,并且很熟悉MouseEvent, KeyEvent, WindowEvent等等這些jdk為我們準備好的事件,那么想必你對java的事件機制已經(jīng)有所理解。但是也許你還是覺得雖然用起來沒什么問題,但是原理還是有些糊涂,那么下面我們再進一步自己實現(xiàn)這些事件和監(jiān)聽器,我們把這個取名為自定義事件。

          其實自定義事件在java中很有用處,我們有的時候想讓自己的程序產(chǎn)生一個事件,但有不希望(或者不可能)用鼠標,鍵盤之類的輸入設備進行操作,比如你寫一個應用程序,在這個程序中一旦收到郵件就對郵件進行相關處理,對于“收到郵件”這個事件,jdk中就沒有定義。對于這樣的事件,以及對于這樣的事件的監(jiān)聽器,我們只能自己動手完成了。

          那么下面就以實例開始我們這個“創(chuàng)新”的過程:首先,我們要明確jdk中需要的資源:類EventObject作為父類用來生成我們自己的事件類,接口EventListener用來實現(xiàn)我們自己的監(jiān)聽器;剩下的事情就是如何注冊這些事件以及測試他們了。

          (1)       通過DemoEvent.java文件創(chuàng)建DemoEvent類,這個類繼承EventObject。這個類的構造函數(shù)的參數(shù)傳遞了產(chǎn)生這個事件的事件源(比如各種控件),方法getSource用來獲得這個事件源的引用。

          DemoEvent.java

          package demo.listener;

           

          import java.util.EventObject;

           

          public class DemoEvent extends EventObject

          {

                  Object obj;

                  public DemoEvent(Object source)

                  {

                         super(source);

                         obj = source;

                  }

                  public Object getSource()

                  {

                         return obj;

                  }

                  public void say()

                  {

                         System.out.println("This is say method...");

                  }

          }

           

          (2)       定義新的事件監(jiān)聽接口,該接口繼承自EventListener;該接口包含對DemeEvent事件的處理程序:

          DemoListener.java

          package demo.listener;

           

          import java.util.EventListener;

           

          public interface DemoListener extends EventListener

          {

                 public void demoEvent(DemoEvent dm);

          }

           

          通過上面的接口我們再定義事件監(jiān)聽類,這些類具體實現(xiàn)了監(jiān)聽功能和事件處理功能。回想一下上文中那四種實現(xiàn)方式,我們這里不正是使用了其中的第三種——外部類寫法的方式嗎?

          Listener1.java

          package demo.listener;

           

          public class Listener1 implements DemoListener

          {

                 public void demoEvent(DemoEvent de)

                 {

                        System.out.println("Inside listener1...");

                 }

          }



          Listener2.java

          package demo.listener;

           

          public class Listener2 implements DemoListener

          {

                 public void demoEvent(DemoEvent de)

                 {

                        System.out.println("Inside listener2...");

                 }

          }


          Listener3.java

          package demo.listener;

           

          public class Listener3 implements DemoListener

          {

                 public void demoEvent(DemoEvent de)

                 {

                        System.out.println("Inside listener3...");

                 }

          }

           

          (3)       通過DemeSource..ava文件創(chuàng)造一個事件源類,它用一個java.utile.Vector對象來存儲所有的事件監(jiān)聽器對象,存儲方式是通過addListener(..)這樣的方法。notifyDemeEvent(..)是觸發(fā)事件的方法,用來通知系統(tǒng):事件發(fā)生了,你調(diào)用相應的處理函數(shù)(回調(diào)函數(shù))吧。

          DemoSource.java

           

          package demo.listener;

          import java.util.*;

           

          public class DemoSource

          {

                 private Vector repository = new Vector();

                 DemoListener dl;

                 public DemoSource()

                 {

           

                 }

                 public void addDemoListener(DemoListener dl)

                 {

                        repository.addElement(dl);

                 }

                 public void notifyDemoEvent()

                 {

                        Enumeration enum = repository.elements();

                        while(enum.hasMoreElements())

                        {

                              dl = (DemoListener)enum.nextElement();

                              dl.demoEvent(new DemoEvent(this));

                        }

                 }

          }

           

           

                       

          (4)       好了,最后寫一個測試程序測試一下我們自定義的事件吧,這段程序應該不難理解吧:)

          TestDemo.java

           

          package demo.listener;

           

          public class TestDemo

          {

                 DemoSource ds;

           

                 public TestDemo()

                 {

                        try{

                              ds = new DemoSource();

                              Listener1 l1 = new Listener1();

                              Listener2 l2 = new Listener2();

                              Listener3 l3 = new Listener3();

           

                              ds.addDemoListener(l1);

                              ds.addDemoListener(l2);

                              ds.addDemoListener(l3);

                              ds.addDemoListener(new DemoListener(){
                                         public void demoEvent(DemoEvent event){
                                                   System.out.println("Method come from 匿名類...");
                                         }
                                 });

                              ds.notifyDemoEvent();

           

                        }catch(Exception ex)

                        {ex.printStackTrace();}

                 }

           

                 public static void main(String args[])

                 {

                        new TestDemo();

                 }

          }



          評論

          # re: java 事件機制  回復  更多評論   

          2007-09-29 15:33 by somesongs
          解惑啊,寫得非常好,謝謝。
          本來自己還想邊學邊寫,但無法超越了。呵呵。

          # re: java 事件機制  回復  更多評論   

          2007-12-18 13:40 by hondz
          還是有收獲的,但是例子倒是沒啥看明白,初學可能是水平太次了吧,DemoEvent對象在哪里創(chuàng)建的呢?如果在事件處理器中調(diào)用say方法會不會拋空指針異常?

          # re: java 事件機制  回復  更多評論   

          2007-12-18 14:03 by hondz
          不好意思了,在notify方法里面創(chuàng)建的,完全明白了,謝謝!!

          # re: java 事件機制  回復  更多評論   

          2008-01-21 11:33 by 盧劍鳴
          淺顯易懂!贊!

          # re: java 事件機制  回復  更多評論   

          2008-11-24 19:51 by 路段的
          鳳飛飛

          # re: java 事件機制  回復  更多評論   

          2008-12-12 17:20 by 鮑鮑
          不錯。謝謝樓主。

          # re: java 事件機制[未登錄]  回復  更多評論   

          2009-03-30 10:33 by abc
          Observer pattern

          # re: java 事件機制[未登錄]  回復  更多評論   

          2009-05-11 17:53 by neo
          恩 很好 贊一個

          # re: java 事件機制  回復  更多評論   

          2009-05-13 17:51 by 路人
          非常謝謝樓主!!!!

          # re: java 事件機制  回復  更多評論   

          2009-08-11 20:10 by hermit
          這個例子還是有些問題的。主要是事件何時觸發(fā)應該是由 DemoSource 自身決定的,所以,事件源 DemoSource 的通知方法 notifyDemoEvent 一般是私有的方法,或者若你想 DemoSource 可以有子類,那么設成protected。也就是 notifyDemoEvent 應該由 DemoSource 在能夠發(fā)生DemoEvent 時自己來調(diào)用,而何時發(fā)生DemoEvent事件,是DemoSource 自身特點決定的,外部對象是不知道的。比如我們可以在 DemoEvent 中加一個屬性 demoValue,
          private int demoValue;
          public int getDemoValue() {
          return demoValue;
          }
          public void setDemoValue(int val) {
          this.demoValue = val;
          notifyDemoEvent(new DemoEvent(this));
          }
          這時,外部對象 TestDemo 可以調(diào)用 ds.setDemoValue(1),就使DemoSource 觸發(fā)了事件,并通知各個監(jiān)聽器處理事件。外部對象 TestDemo 不應該調(diào)用 notifyDemoEvent 方法,它怎么知道什么時候應該發(fā)生 DemoEvent 事件呢?

          # re: java 事件機制  回復  更多評論   

          2009-09-04 10:24 by zzw
          嗯,樓上有理,樓主也寫的好

          # re: java 事件機制  回復  更多評論   

          2009-10-10 12:08 by 啊白
          受益了,感謝樓主

          # re: java 事件機制[未登錄]  回復  更多評論   

          2014-11-13 15:03 by scott
          樓主總體來說是不錯的,我就想問問你有沒有想過為什么要“DemoEvent extends EventObject”,不這樣寫行不?是運行不了還是性能損失了?或者處于別的什么考慮寫了這個?
          你為什么又要這樣寫“DemoListener extends EventListener”,這個也是必須的嗎?
          我只能說你在不求甚解的時候,已經(jīng)開始誤人子弟了(有很多人讀了你的這篇博文,而且留言了)。網(wǎng)上有很多類似的帖子,代碼幾乎都是一樣的。不知道是樓樓拷了別人的代碼,還是別人拷了樓樓的代碼,希望樓樓加油,寫出更好的帖子。

          # re: java 事件機制  回復  更多評論   

          2015-04-29 15:17 by 莾s
          多虧看評論了。

          只有注冊用戶登錄后才能發(fā)表評論。


          網(wǎng)站導航:
           
          主站蜘蛛池模板: 南澳县| 察雅县| 藁城市| 马鞍山市| 朔州市| 新蔡县| 宁夏| 织金县| 桑日县| 武宣县| 宿州市| 信丰县| 缙云县| 浦东新区| 中宁县| 夏邑县| 丹凤县| 彭泽县| 田林县| 光泽县| 永登县| 喀喇沁旗| 惠东县| 永靖县| 汾西县| 章丘市| 南雄市| 洛扎县| 日喀则市| 莎车县| 杨浦区| 东至县| 岳西县| 安徽省| 霍林郭勒市| 江门市| 芜湖县| 互助| 文昌市| 长海县| 类乌齐县|