程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> 關於JAVA >> 方法查找工具

方法查找工具

編輯:關於JAVA

第11章介紹了Java 1.1新的“反射”概念,並利用這個概念查詢一個特定類的方法——要麼是由所有方法構成的一個完整列表,要麼是這個列表的一個子集(名字與我們指定的關鍵字相符)。那個例子最大的好處就是能自動顯示出所有方法,不強迫我們在繼承結構中遍歷,檢查每一級的基礎類。所以,它實際是我們節省編程時間的一個有效工具:因為大多數Java方法的名字都規定得非常全面和詳盡,所以能有效地找出那些包含了一個特殊關鍵字的方法名。若找到符合標准的一個名字,便可根據它直接查閱聯機幫助文檔。
但第11的那個例子也有缺陷,它沒有使用AWT,僅是一個純命令行的應用。在這兒,我們准備制作一個改進的GUI版本,能在我們鍵入字符的時候自動刷新輸出,也允許我們在輸出結果中進行剪切和粘貼操作:

 

//: DisplayMethods.java
// Display the methods of any class inside
// a window. Dynamically narrows your search.
import java.awt.*;
import java.awt.event.*;
import java.applet.*;
import java.lang.reflect.*;
import java.io.*;

public class DisplayMethods extends Applet {
  Class cl;
  Method[] m;
  Constructor[] ctor;
  String[] n = new String[0];
  TextField 
    name = new TextField(40),
    searchFor = new TextField(30);
  Checkbox strip = 
    new Checkbox("Strip Qualifiers");
  TextArea results = new TextArea(40, 65);
  public void init() {
    strip.setState(true);
    name.addTextListener(new NameL());
    searchFor.addTextListener(new SearchForL());
    strip.addItemListener(new StripL());
    Panel 
      top = new Panel(),
      lower = new Panel(),
      p = new Panel();
    top.add(new Label("Qualified class name:"));
    top.add(name);
    lower.add(
      new Label("String to search for:"));
    lower.add(searchFor);
    lower.add(strip);
    p.setLayout(new BorderLayout());
    p.add(top, BorderLayout.NORTH);
    p.add(lower, BorderLayout.SOUTH);
    setLayout(new BorderLayout());
    add(p, BorderLayout.NORTH);
    add(results, BorderLayout.CENTER);
  }
  class NameL implements TextListener {
    public void textValueChanged(TextEvent e) {
      String nm = name.getText().trim();
      if(nm.length() == 0) {
        results.setText("No match");
        n = new String[0];
        return;
      }
      try {
        cl = Class.forName(nm);
      } catch (ClassNotFoundException ex) {
        results.setText("No match");
        return;
      }
      m = cl.getMethods();
      ctor = cl.getConstructors();
      // Convert to an array of Strings:
      n = new String[m.length + ctor.length];
      for(int i = 0; i < m.length; i++)
        n[i] = m[i].toString();
      for(int i = 0; i < ctor.length; i++)
        n[i + m.length] = ctor[i].toString();
      reDisplay();
    }
  }
  void reDisplay() {
    // Create the result set:
    String[] rs = new String[n.length];
    String find = searchFor.getText();
    int j = 0;
    // Select from the list if find exists:
    for (int i = 0; i < n.length; i++) {
      if(find == null)
        rs[j++] = n[i];
      else if(n[i].indexOf(find) != -1)
          rs[j++] = n[i];
    }
    results.setText("");
    if(strip.getState() == true)
      for (int i = 0; i < j; i++)
        results.append(
          StripQualifiers.strip(rs[i]) + "\n");
    else // Leave qualifiers on
      for (int i = 0; i < j; i++)
        results.append(rs[i] + "\n");
  }
  class StripL implements ItemListener {
    public void itemStateChanged(ItemEvent e) {
      reDisplay();
    }
  }
  class SearchForL implements TextListener {
    public void textValueChanged(TextEvent e) {
      reDisplay();
    }
  }
  public static void main(String[] args) {
    DisplayMethods applet = new DisplayMethods();
    Frame aFrame = new Frame("Display Methods");
    aFrame.addWindowListener(
      new WindowAdapter() {
        public void windowClosing(WindowEvent e) {
          System.exit(0);
        }
      });
    aFrame.add(applet, BorderLayout.CENTER);
    aFrame.setSize(500,750);
    applet.init();
    applet.start();
    aFrame.setVisible(true);
  }
}

class StripQualifiers {
  private StreamTokenizer st;
  public StripQualifiers(String qualified) {
      st = new StreamTokenizer(
        new StringReader(qualified));
      st.ordinaryChar(' ');
  }
  public String getNext() {
    String s = null;
    try {
      if(st.nextToken() !=
            StreamTokenizer.TT_EOF) {
        switch(st.ttype) {
          case StreamTokenizer.TT_EOL:
            s = null;
            break;
          case StreamTokenizer.TT_NUMBER:
            s = Double.toString(st.nval);
            break;
          case StreamTokenizer.TT_WORD:
            s = new String(st.sval);
            break;
          default: // single character in ttype
            s = String.valueOf((char)st.ttype);
        }
      }
    } catch(IOException e) {
      System.out.println(e);
    }
    return s;
  }
  public static String strip(String qualified) {
    StripQualifiers sq = 
      new StripQualifiers(qualified);
    String s = "", si;
    while((si = sq.getNext()) != null) {
      int lastDot = si.lastIndexOf('.');
      if(lastDot != -1)
        si = si.substring(lastDot + 1);
      s += si;
    }
    return s;
  }
} ///:~


程序中的有些東西已在以前見識過了。和本書的許多GUI程序一樣,這既可作為一個獨立的應用程序使用,亦可作為一個程序片(Applet)使用。此外,StripQualifiers類與它在第11章的表現是完全一樣的。
GUI包含了一個名為name的“文本字段”(TextField),或在其中輸入想查找的類名;還包含了另一個文本字段,名為searchFor,可選擇性地在其中輸入一定的文字,希望在方法列表中查找那些文字。Checkbox(復選框)允許我們指出最終希望在輸出中使用完整的名字,還是將前面的各種限定信息刪去。最後,結果顯示於一個“文本區域”(TextArea)中。
大家會注意到這個程序未使用任何按鈕或其他組件,不能用它們開始一次搜索。這是由於無論文本字段還是復選框都會受到它們的“偵聽者(Listener)對象的監視。只要作出一項改變,結果列表便會立即更新。若改變了name字段中的文字,新的文字就會在NameL類中捕獲。若文字不為空,則在Class.forName()中用於嘗試查找類。當然,在文字鍵入期間,名字可能會變得不完整,而Class.forName()會失敗,這意味著它會“擲”出一個違例。該違例會被捕獲,TextArea會隨之設為“Nomatch”(沒有相符)。但只要鍵入了一個正確的名字(大小寫也算在內),Class.forName()就會成功,而getMethods()和getConstructors()會分別返回由Method和Constructor對象構成的一個數組。這些數組中的每個對象都會通過toString()轉變成一個字串(這樣便產生了完整的方法或構建器簽名),而且兩個列表都會合並到n中——一個獨立的字串數組。數組n屬於DisplayMethods類的一名成員,並在調用reDisplay()時用於顯示的更新。
若改變了Checkbox或searchFor組件,它們的“偵聽者”會簡單地調用reDisplay()。reDisplay()會創建一個臨時數組,其中包含了名為rs的字串(rs代表“結果集”——Result Set)。結果集要麼直接從n復制(沒有find關鍵字),要麼選擇性地從包含了find關鍵字的n中的字串復制。最後會檢查strip Checkbox,看看用戶是不是希望將名字中多余的部分刪除(默認為“是”)。若答案是肯定的,則用StripQualifiers.strip()做這件事情;反之,就將列表簡單地顯示出來。
在init()中,大家也許認為在設置布局時需要進行大量繁重的工作。事實上,組件的布置完全可能只需要極少的工作。但象這樣使用BorderLayout的好處是它允許用戶改變窗口的大小,並特別能使TextArea(文本區域)更大一些,這意味著我們可以改變大小,以便毋需滾動即可看到更長的名字。
編程時,大家會發現特別有必要讓這個工具處於運行狀態,因為在試圖判斷要調用什麼方法的時候,它提供了最好的方法之一。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved