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

          java 事件機(jī)制

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

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

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

          2.event source:具體的接受事件的實(shí)體,比如說,你點(diǎn)擊一個(gè)button,那么button就是event source,這樣你必須使button對(duì)某些事件進(jìn)行相應(yīng),你就需要注冊(cè)特定的listener,比如說MouseEvent之中的MouseClicked方法,這是他就必須有了add方法

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

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

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

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

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

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

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

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

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

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

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

          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;該接口包含對(duì)DemeEvent事件的處理程序:

          DemoListener.java

          package demo.listener;

           

          import java.util.EventListener;

           

          public interface DemoListener extends EventListener

          {

                 public void demoEvent(DemoEvent dm);

          }

           

          通過上面的接口我們?cè)俣x事件監(jiān)聽類,這些類具體實(shí)現(xiàn)了監(jiān)聽功能和事件處理功能。回想一下上文中那四種實(shí)現(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)造一個(gè)事件源類,它用一個(gè)java.utile.Vector對(duì)象來存儲(chǔ)所有的事件監(jiān)聽器對(duì)象,存儲(chǔ)方式是通過addListener(..)這樣的方法。notifyDemeEvent(..)是觸發(fā)事件的方法,用來通知系統(tǒng):事件發(fā)生了,你調(diào)用相應(yīng)的處理函數(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)       好了,最后寫一個(gè)測試程序測試一下我們自定義的事件吧,這段程序應(yīng)該不難理解吧:)

          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();

                 }

          }



          評(píng)論

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

          2008-12-12 17:20 by 鮑鮑
          不錯(cuò)。謝謝樓主。

          # re: java 事件機(jī)制[未登錄]  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制[未登錄]  回復(fù)  更多評(píng)論   

          2009-05-11 17:53 by neo
          恩 很好 贊一個(gè)

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制[未登錄]  回復(fù)  更多評(píng)論   

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

          # re: java 事件機(jī)制  回復(fù)  更多評(píng)論   

          2015-04-29 15:17 by 莾s
          多虧看評(píng)論了。

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


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 大宁县| 孝昌县| 中卫市| 庆元县| 常德市| 新巴尔虎左旗| 西峡县| 闸北区| 平凉市| 双辽市| 方城县| 色达县| 马关县| 马龙县| 毕节市| 南投市| 伊吾县| 从江县| 广安市| 万宁市| 佛冈县| 枞阳县| 三台县| 辽阳市| 丽江市| 多伦县| 呼图壁县| 天祝| 三亚市| 家居| 九寨沟县| 东至县| 麻栗坡县| 漠河县| 得荣县| 德令哈市| 宝山区| 三原县| 永丰县| 公安县| 千阳县|