zoukankan      html  css  js  c++  java
  • 事件驱动模型。。。。有时间弄

    public class A{
    private Vector aListeners = new Vector();
    private int value;
    public int getValue(){
    return value;
    }
    public void setValue(int newValue){
    if(value!=newValue){
    value = newValue;
    AEvent evt= new AEvent(this,value);
    //如果值改变的话,就触发事件 
    fireAEvent(evt);
    }
    }
    public synchronized void addAListener(AListener a){
    aListeners.addElement(a);
    }
    public synchronized void removeAListener(Alistener a){
    aListeners.removeElement(a);
    }
    public void fireAEvent(AEvent evt){
    Vector currentListeners = null;
    synchronized(this){
    currentListeners = (Vector)aListeners.clone();
    }
    for(int i =0;i<currentListeners.size();i++){
    AListener listener = (AListener)currentListeners.elementAt(i);
    listener.performed(evt);
    }
    }
    }
    //定义接口,当事件触发时调用 
    public interface AListener extends java.util.EventListener{
    public abstract void performed(AEvent e);
    }
    //定义事件 
    public class AEvent extends EventObject{
    private int value;
    public AEvent(Object source){
    this(source,null);
    }
    public AEvent(Object source,int newValue){
    super(source);
    value = newValue;
    }
    }
    class B{
    public B(){
    A a = new A();
    a.addAListener(new AListener(){
    public performed(AEvent e){
    //要处理的 
    }
    });
    }
    }



































    事件驱动架构Event Driven Architecture (EDA) 是一种低耦合可分布式的架构,它通常处理异步信息流。

    通常可以把事件看作是业务领域中发生的一个变化。因此,在代码中我们需要定义事件,并且实例化该事件。在.net中,事件可以看作行为的结果。必然有事件发生者和接收者,触发事件的对象为发生者,响应事件的对象则为事件接收者。委托(delegate)则串联起发生者和接收者。C#中委托的概念不在此累述。下面代码是声明委托,初始化委托和调用委托的示例。

    声明一个委托

    1. public delegate int TestDelegate(object obj1, object obj2) ; 

    实例化一个委托

    1. TestDelegate TD = new TestDelegate(TestDelegateMethod) ; 

    调用一个委托

    1. TestDelegateMethod(" This is a Test."); 

    注意,其实在C#中,更多的时候是使用event修饰的delegate。使用event修饰后的委托是一个特殊的委托,它的特殊性体现在对象的封装性上。比如上面的代码可以写成

    event TestDelegate TD = TestDelegateMethod;关于event和delegate的更多细节可以参考博文

    http://www.tracefact.net/CSharp-Programming/Delegates-and-Events-in-CSharp.aspx

    http://blog.monstuff.com/archives/000040.html

    假设有如下一个应该场景,一个血压监测仪器,在病人每一次心跳的时候,监测血压,一旦发现病人血压小于80,则开始报警。

    那么此处,动作的发出者就是病人,一旦发出心跳这个动作后,(通知)血压检测仪立即检测血压,如果小于80,则发出警报。

    有一个BloodPressureMonitor类和Paient类。BloodPressureMonitor类只用一个方法,监测血压。病人类中主要有一个委托和一个HeartBeat()方法。其中每次调用HeartBeat()方法,都会调用委托方法。(此处,委托的意义类似于函数式编程的概念,将函数作为一个变量值,该变量赋予了哪个函数就调用哪个函数)。

    因此,在main方法中,主要就是对dosth委托的赋值,也就是对病人每次心跳的时候,要调用一下血压检测仪的CheckPresssure方法。

    代码如下:

       

    1. public delegate void DoAction(int s); 
    2.     class BloodPressureMonitor 
    3.     { 
    4.         public void CheckPresssure(int p) 
    5.         { 
    6.             if (p < 80) 
    7.             { 
    8.                 Console.WriteLine(String.Format("Alert!Bloodpressure is less than {0}", p)); 
    9.             } 
    10.             else 
    11.             { 
    12.                 Console.WriteLine(String.Format("Bloodpressure is {0}", p)); 
    13.             } 
    14.         } 
    15.     } 
    16.     class Patient { 
    17.         public event DoAction dosth; 
    18.         public String Name { get; set; } 
    19.         public int BloodPressure { get; set; } 
    20.         public void HeartBeat() 
    21.         { 
    22.             if (dosth != null) 
    23.                 dosth(BloodPressure); 
    24.         } 
    25.     } 
    26.     class Program 
    27.     { 
    28.         static void Main(string[] args) 
    29.         { 
    30.             Patient  p = new Patient(); 
    31.             p.Name = "Tom"; 
    32.             p.BloodPressure = 100; 
    33.             BloodPressureMonitor monitor = new BloodPressureMonitor(); 
    34.             p.dosth += monitor.CheckPresssure; 
    35.             for (int i = 0; i < 10; i++) 
    36.             { 
    37.                 p.BloodPressure -= 5; 
    38.                 p.HeartBeat(); 
    39.             } 
    40.         } 
    41.     } 

    运行结果如下:

    600A4D7F7AA747EBA23D5B30A8C0B3D2

    也许很多人会觉得,那何必这么麻烦,直接在 HeartBeat方法中调用BloodPressureMonitor的CheckPresssure方法不就行了嘛,何必用委托呢?例如直接写成

    1. public void HeartBeat() 
    2.  { 
    3.      new BloodPressureMonitor(). CheckPresssure( BloodPressure ); 
    4.  } 

    这样的写法不也是可以实现的吗?

    是的,但是这种写法不灵活,首先,使用委托的话,可以利用委托的一些特点,比如调用委托的BeginInvoke方法可以异步在线程池线程上执行。

    其次,如果此时再要加一个电话呼叫功能,一旦血压小于60,电话机自动呼叫医生,那么在现有的基础上,改起来十分容易。只需要加个电话类,然后在委托上再挂一个方法就行了。

    代码如下:

    1. using System; 
    2. using System.Collections.Generic; 
    3. using System.Linq; 
    4. using System.Text; 
    5. namespace EDA 
    6.     public delegate void DoAction(int s);//定义一个委托 
    7.     class BloodPressureMonitor 
    8.     { 
    9.         public void CheckPresssure(int p) 
    10.         { 
    11.             if (p < 80) 
    12.             { 
    13.                 Console.WriteLine(String.Format("Alert!Bloodpressure is less than {0}", p)); 
    14.             } 
    15.             else 
    16.             { 
    17.                 Console.WriteLine(String.Format("Bloodpressure is {0}", p)); 
    18.             } 
    19.         } 
    20.     } 
    21.     class Telephone 
    22.     { 
    23.         public void NeedCallDoctor(int p) 
    24.         { 
    25.             if (p < 60) 
    26.             { 
    27.                 Console.WriteLine(String.Format("Call doctor!", p)); 
    28.             } 
    29.         } 
    30.     } 
    31.     class Patient 
    32.     { 
    33.         public event DoAction dosth; 
    34.         public String Name { get; set; } 
    35.         public int BloodPressure { get; set; } 
    36.         public void HeartBeat()//心跳时,调用dosth委托 
    37.         { 
    38.             if (dosth != null) 
    39.                 dosth(BloodPressure); 
    40.         } 
    41.     } 
    42.     class Program 
    43.     { 
    44.         static void Main(string[] args) 
    45.         { 
    46.             Patient p = new Patient(); 
    47.             p.Name = "Tom"; 
    48.             p.BloodPressure = 100; 
    49.             BloodPressureMonitor monitor = new BloodPressureMonitor(); 
    50.             p.dosth += monitor.CheckPresssure;//具体设定dosth是哪些方法。用+=可以挂载多个方法 
    51.             Telephone Phone = new Telephone(); 
    52.             p.dosth += Phone.NeedCallDoctor; 
    53.             for (int i = 0; i < 10; i++) 
    54.             { 
    55.                 p.BloodPressure -= 5; 
    56.                 p.HeartBeat(); 
    57.             } 
    58.         } 
    59.     } 

    9C986808A9BF457CA9576779A0976D98

    本文出自 “一只博客” 博客,请务必保留此出处http://cnn237111.blog.51cto.com/2359144/1176649

    仿net事件委托的java事件模型实现

    学习过Java Swing的读者一定对于Swing中相对较为复杂的事件驱动模型比较困惑,虽然事件驱动模型在Java Swing中被完完全全的体现出来了,

    但是对于一个软件初学者而言这样的近乎“裸体”的事件驱动模型确实是很难理解的。
    Microsoft公司.Net框架与Java Swing的GUI编程相比要简单很多,同样是事件驱动模型.Net框架就进行了大量的封装处理,.Net把这种封装称之

    为委托器(Delegate)其代码如下:


     1 //当btnSubmit按钮被点击以后要求交给btnSubmit_Click方法处理
     2 // EventHandler在中间启到委托器的作用,
     3 //它负责将事件分发到指定的方法中进行处理
     4 this.btnSubmit.Click += new EventHandler(this.btnSubmit_Click);
     5 //事件处理方法
     6 // object sender:事件源,这里指btnSubmit对象
     7 // EventArgs e:事件处理参数,它保存了需要提供给程序员的必要信息
     8     private void btnSubmit_Click(object sender, EventArgs e)
     9     {
    10         //打印This is a button语句
    11         System.Diagnostics.Debug.WriteLine("This is button");
    12    }
    13 

    作为对比,我们来看看Java Swing的事件处理和委托就要复杂很多:代码如下

     


     1 //为btnSubmit增加侦听器SelectHandler,当btnSubmit被点击以后
     2 //有侦听器的actionPerformed负责处理该点击事件的业务
     3 //由于事件源btnSubmit和侦听器类SelectHandler处于两个不同的类中
     4 //为了让SelectHandler类取得页面的信息,我们需要将窗体对象(this)
     5 //传入到侦听器中
     6 btnSubmit.addActionListener(new SelectHandler(this));
     7 //侦听器SelectHandler,它必须实现动作事件ActionListener接口
     8 //以达到事件分发的作用
     9 class SelectHandler implements ActionListener {
    10     private CommonDialogDemo form = null;
    11     //将窗体对象CommonDialogDemo通过构造函数传入SelectHandler类中
    12     public SelectHandler(CommonDialogDemo form) {
    13         this.form = form;
    14     }
    15     //事件处理方法,当btnSubmit被点击,自动执行以下打印代码
    16     publicvoid actionPerformed(ActionEvent e) {
    17         System.out.println("This is button");
    18     }
    19 }
    20 

    根据以上代码,我们可以清晰的看到Java Swing要比.Net的麻烦的多,而且更不能让人忍受的就是,一个页面如果有多个按钮的话,我们必须针

    对每个按钮编写多个事件侦听类,而且这些类一般都会被设为内部类。学过软件建模的读者可能知道,内部在软件建模在软件工程中是不推荐使

    用的,所以这样的代码编写明显会增加设计冗余度和复杂度,因此我们可以考虑自己编写一个类似于.Net中EventHandler一样的事件委托类来处

    理事件分发。
           由于我们无权修改Java的编译器,所以我在这里将会借助于反射技术,利用一个事件委托类处理所有的点击事件,代码如下: 


     1 package cn.softworks.teachersearchsystem.support;
     2 
     3 import java.awt.event.ActionEvent;
     4 import java.awt.event.ActionListener;
     5 import java.lang.reflect.Method;
     6 
     7 /**
     8 *该类是用来处理所有的Swing按钮点击事件,并根据将处理权<br>
     9 *转交给使用者来处理
    10 *
    11 *@authorChen.yu
    12 *
    13 */
    14 publicclass EventHandlerimplements ActionListener {
    15     
    16     //组件所在的窗体对象
    17     private Object form = null;
    18     
    19     //受到委托的方法名
    20     private String methodName = null;
    21     
    22     /**
    23      *构造函数
    24      *
    25      *@paramform           组件所在的窗体对象
    26      *@parammethodName     受到委托的方法名
    27      */
    28     public EventHandler(Object form,String methodName) {
    29         this.form = form;
    30         this.methodName = methodName;
    31     }
    32     
    33     /**
    34      *事件处理委托方法
    35      */
    36     publicvoid actionPerformed(ActionEvent e) {
    37         
    38         //得到窗体对象的类型
    39         Class formType = this.form.getClass();
    40         
    41         try {
    42             //得到指定委托方法的类型 
    43             Method method = 
    44                 formType.getMethod(this.methodName, new Class[] {e.getClass()});
    45             //调用指定的方法
    46             method.invoke(this.form, new Object[] {e});
    47             
    48         }catch(Exception ex) {
    49             
    50             return;
    51         }       
    52     
    53     }
    54 
    55 }
    56 
    57 

    现在我们来编写一个测试程序,代码如下: 


    1 btnSearch.addActionListener(
    2 new EventHandler(this,"btnSearch_Click"));

    4 public void btnSearch_Click(ActionEvent e) {
    5     System.out.println("This is btnSearch");
    6 }


    从以上代码中我们可以清晰的看到,事件处理和事件委托处于同一窗体中了,.Net方便的Delegate处理被我们用反射实现了。

  • 相关阅读:
    U启动安装原版Win7系统教程
    U启动制作U盘启动盘详细教程
    大白菜装机版一键制作启动u盘教程
    git 学习笔记5--rm & mv,undo
    git 学习笔记4--.gitignore
    git 学习笔记3--status flow
    git 学习笔记2--How to create/clone a repository
    git 学习笔记1--config & help
    Ubuntu Mono 运行 Helloworld
    HttpClient 使用
  • 原文地址:https://www.cnblogs.com/liuxiaoke/p/3823000.html
Copyright © 2011-2022 走看看