開始之前
關於本教程
Ruby on Rails (Rails) 是用 Ruby 編 寫的一個 full-stack Web 應用程序框架,而 Ruby 是一種功能豐富的、免費的 、可擴展的、可移植的、面向對象的腳本編制語言。Rails 在 Web 應用程序開 發人員之間非常流行。通過它,可以快速有效地開發 Web 應用程序,並將其部 署到任何 Web 容器中,例如 IBM® WebSphere® 或 Apache Tomcat。
在 Rails 和類似的 Web 應用程序開發框架出現之前,用於 Web 應用程序開 發的標准工具是 Java 語言,因為 Java 語言是獨立於平台的,並且有完整的 API 集。很多 Java Web 應用程序仍然在運行,這導致很多非常有用的、編寫良 好的 Java 代碼(在本教程中統稱遺留 代碼)具有良好的可用性。遺留 Java 代碼通常被打包在一組 JAR 文件。
如果將 Web 應用程序開發平台改為 Rails,那麼可以重用遺留 Java 代碼。 Ruby Java Bridge (RJB) 是一個工具包,通過它可以將 JAR 文件裝載到 Rails 應用程序中,還可以在 Rail 應用程序中訪問其中的方法和變量。本教程解釋如 何在 Rails 應用程序中配置和使用 RJB。
目標
在本教程中,您 將學習如何:
下載、編譯和安裝 RJB
設置 RJB 以訪問共享 Java 庫
將遺留 Java 代碼裝載到 Rails 應用程序中並進行訪問
本教 程並不深入研究 Rails 的功能。與其他 Web 框架相比,Rails 有很多優點,其 中一個優點就是用於該平台的文檔的數量和質量都很高。
先決條件
本教程假設讀者基本熟悉 Java 語言、Ruby 和 Ruby on Rails。
系統需求
本教程假設您使用 Linux® 系統(但是,在 Windows® 上的步驟基本上是相同的)。本教程假設您有一個可以工作的 Ruby on Rails。
RJB 要求系統上安裝有 Java SDK。如果需要一個 Java SDK,可以針對您的平台下載最新的 Java SE SDK,並馬上安裝它。
RJB 安裝和設置
本節帶您親歷 RJB 的下載、安裝、編譯和設置。
下 載 RJB
可以下載標准 Ruby Gem 包或自己編譯的源代碼歸檔文件形式的 RJB。為了進行演示,我推薦下載源代碼歸檔文件,所以我將使用這種方法。閒 話少說,現在就 下載 RJB 1.1.3 source .zip 文件(在撰寫本教程之際,已經 有了最新的 RJB 版本)。
確保設置或更新了以下環境變量,它們是安裝 RJB 所必需的:
JAVA_HOME 必須指向 Java SDK 安裝目錄。
PATH 必須包括 $JAVA_HOME/bin。
例如,在 bash(僅用於 Linux 系統)中,假設已經將 Java SDK 安裝到 /usr/local/jdk60,則執行以 下命令:
[root@san]# export JAVA_HOME=/usr/local/jdk60
[root@san]# export PATH=$PATH:$JAVA_HOME/bin
編譯和安裝 RJB
下一步是通 過執行以下命令編譯和安裝 RJB:
[root@san]# unzip rjb- 1.1.3.zip
[root@san]# cd rjb-1.1.3
[root@san]# ruby setup.rb config
[root@san]# ruby setup.rb setup
[root@san]# ruby setup.rb install
確 認安裝成功
為了確認 RJB 安裝成功,首先調用 Ruby 的交互式控制台 irb:
[root@san]# irb
然後輸入 require 'rjb':
irb(main):001:0> require 'rjb'
=> true
irb(main):002:0>exit
如果 require 'rjb' 命令返回 true,則意味著 Ruby 安裝識別 出新安裝的 rjb 庫。現在可以在應用程序中開始使用 RJB。
通過 RJB 使用遺留代碼
在本節中,您將在 Rails 應用程序中裝載和訪問遺留 Java 代碼。
示例項目
Java Tar package from ICE Engineering 是用 Java 語言編寫的一個很好的工具包,用於處理歸檔文件。它提供了 tar 歸檔實用程序的本地 Java 實現,當與 java.util.zip 包相結合時,它可以處 理 .tar.gz 文件。它還利用 Java 語言的平台獨立性,可以不作修改地在所有 UNIX® 變體和 Windows 上運行。作為練習,您將使用它來解壓一個樣例 tar 文件的內容。通過類似的方法,可以在 Ruby on Rails 應用程序中使用任 何遺留 Java 代碼。
練習的目標是:
將 tar.jar 文件裝載到一個 Rails 應用程序中。
將 JAR 文件所需的類裝載到應用程序中。
解壓使用這些類的樣 例 test.tar 文件的內容。
入門
獲取樣例文件
首先,需 要為系統獲取樣例 tar 文件(test.tar)和 Java Tar 包:
將 test.tar 下載並保存 到一個方便的位置。
下載和保存 javatar- 2.5.tar.gz。
將 javatar-2.5.tar.gz 的內容解壓到一個方便的位置。 這個練習中,這個包中惟一需要用到的文件是 tar.jar,它在 jars 目錄中。
訪問共享庫
RJB 使用 Java Native Interface (JNI) 實現它的 功能。因此,它需要訪問 JDK 安裝中附帶的一些共享對象文件(共享庫)。您 必須使用以下命令將這些文件的位置添加到 LD_LIBRARY_PATH 環境變量中:
[root@san]# export JAVA_HOME=/usr/local/jdk60
[root@san]# export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$JAVA_HOME/jre/lib/i386
[root@san]# export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$JAVA_HOME/jre/lib/i386/client
如果打算在獨立的 Ruby 腳本中使用 RJB,那麼只需在正在處理的 shell 中 設置這些環境變量。對於您的 Ruby on Rails 應用程序,還必須在 Rails 應用 程序的 environment.rb 文件中設置這些變量。
將 RJB 裝載到 Rails 應用程序
要將 RJB 裝載到 Rails 應用程序並將它設置為可以調用 Java 類,需要執 行兩個步驟:
告訴 Ruby 將 rjb 庫包含到代碼中。
裝載 JVM,設置類路徑和其他可選的 JVM 參數。
首先,使用下面的命令初始化 RJB:
require 'rjb'
接著,將 Rails 應用程序中將要使用的所有 legacy .jar 文件 — 本例中 為 tar.jar — 添加到 classpath 變量:
Rjb::load(classpath = '.:/path/to/tar.jar', jvmargs=[])
可以將 jvmargs 留空,除非希望為 JVM 指定額外的參數。
現在,可以將准備使用的 Java 類導入到 Ruby 中,實例化這些類,並調用 所需的方法。
將 Java 類導入到 Ruby 並實例化
清單 1 中的 Ruby 代碼從 tar.jar 包中導入了需要的 Java 類,並從導入 的類中創建了 Ruby 對象:
清單 1. 將 Java 類導入到 Ruby 並實例化
tararchive = Rjb::import('com.ice.tar.TarArchive')
fileinputstream = Rjb::import('java.io.FileInputStream')
file = Rjb::import('java.io.File')
file_instance = file.new_with_sig('Ljava.lang.String;','.')
fileinputstream_instance =
fileinputstream.new_with_sig('Ljava.lang.String;','test.tar')
tararchive_instance =
tararchive.new_with_sig ('Ljava.io.InputStream;',fileinputstream_instance)
p "Let's verify that the objects created are indeed of the classes we
wanted..."
p "------------------------------"
p "For the File instance...."
p "Expecting: java.io.File , it is: " + file_instance._classname
p "------------------------------"
p "For the FileInputStream instance...."
p "Expecting: java.io.FileInputStream , it is: " +
fileinputstream_instance._classname
p "------------------------------"
p "For the TarArchive instance...."
p "Expecting: com.ice.tar.TarArchive , it is: " +
tararchive_instance._classname
導入 Java 類
清單 1 中的前三行將調用 RJB 的 import 方法,將所需的類分別導入到 Ruby 變量 tararchive、fileinputstream 和 file 中。必須指定類的完整包路 徑 — 例如,TarArchive 類為 com.ice.tar.TarArchive,而 FileInputStream 類為 java.io.FileInputStream — 就像使用 java 命令運行應用程序那樣。
實例化導入的類
接著,清單 1 創建導入類的對象。可以通過調用每個類的 new 方法創建類 ,就像創建任何 Ruby 對象一樣(例如,tararchive.new)。但是這樣做會調用 TarArchive 類的默認的構造函數(沒有參數),而您並不希望這樣做。
當重載了類的構造函數後,需要對上面的對象創建方法進行一些修改。在這 種情況下,必須按照下面的方式創建對象:
object = Classname.new_with_sig('signature', parameter[,more parameters])
第一個參數定義構造函數所需的參數的簽名類型。它告訴 RJB 調用其輸入參 數匹配指定簽名的構造函數。
清單 1 中的第 4 個和第 5 個語句分別創建 file 和 fileinputstream 類 的對象,它們調用相應的構造函數,參數類型為 String。
在清單 1 的第 6 個語句中,TarArchive 類的其中一個構造函數接受 InputStream 類型的對象作為參數。該語句的簽名類型是一個單獨的 InputStream 輸入參數。這些類型簽名的詳細細節在 getName API 的 Java SDK 文檔中做了很好的描述。第二個參數是創建的 InputStream 類型對象。
檢驗對象創建
清單 1 中的其余內容將檢驗 RJB 創建的對象是否是指定類的對象,方法是 調用添加到每個對象的 _classname 方法。例如,調用 tararchive_instance._classname 將返回 com.ice.tar.TarArchive,這意味著 類被正確裝載,並成功創建了該類的對象。
調用方法並捕獲結果
將類裝載到 Ruby 並從中創建了對象後,下一步是調用需要的方法並查看結 果。例如,您希望使用 TarArchive 類的 extractContents 方法,將樣例文件 (test.tar)的內容提取到當前目錄中。
和構造函數一樣,可以使用兩種方式調用方法。一種方式是直接調用方法, 例如:
tararchive_instance.extractContents(file_instance)
當方法重載後,使用 _invoke 調用指定方法的每個參數的類型簽名:
tararchive_instance._invoke('extractContents', 'Ljava.io.File;', file_instance)
這一步可以使 RJB 知道在方法重載時應該調用哪些方法。
和對待普通 Ruby 代碼一樣,您將捕獲對象方法返回的結果(如果有的話) ,並在自己的應用程序中使用結果。方法調用返回的結果被自動轉換為相應的對 象類型。您只需在對象內直接調用方法。
Java TarArchive 類中實現的功能現在可以用於您的 Ruby 代碼。通過使用 相同的方法,Java 代碼中已實現的任何功能可以不加修改地在您的 Ruby 和 Rails 應用程序中重用。
完整的代碼
清單 2 展示了本教程示例的完整 Ruby 代碼(也可以通過 下載 獲得):
清單 2. 完整的示例 Ruby 代碼
# Include 'rjb' library into your application
require 'rjb'
# Load the JVM specifying the jar files to include and any other optional JVM arguments
Rjb::load(classpath = '.:/path/to/tar.jar', jvmargs=[])
# Import the classes you want to use into a Ruby variable
# specify the full package path to the classes.
tararchive = Rjb::import('com.ice.tar.TarArchive')
fileinputstream = Rjb::import('java.io.FileInputStream')
file = Rjb::import('java.io.File')
# Create objects of the classes. Use the new method directly or use
# the 'new_with_sig' call to invoke overloaded constructors with arguments
# Directory you want to extract the files in this case, the current directory
file_instance = file.new_with_sig('Ljava.lang.String;','.')
# inputstream instance of the file to extract
fileinputstream_instance = fileinputstream.new_with_sig ('Ljava.lang.String;','test.tar')
# tararchive instance of the file to be extracted.
tararchive_instance = tararchive.new_with_sig ('Ljava.io.InputStream;'\
,fileinputstream_instance)
# Invoke the method from the class and capture the results.
# Use either the direct call of the method,
# or the '_invoke' call to invoke overloaded methods.
p 'Extracting file.....'
tararchive_instance.extractContents(file_instance)
p 'Done...'
嘗試這些代碼,將清單 2 中的代碼保存到一個文件中,並且擴展名為 .rb( 或使用 下載 中的 rjb-javatar.rb),然後在 Ruby 解釋程序中運行。
結束語
在全新的 Rails 應用程序中重用已有的遺留 Java 代碼其實非常簡單,方式 如下:
安裝 Java SDK 和 RJB。
將 JAVA_HOME 和 LD_LIBRARY_PATH 環境變量導出到您的 Rails 應用程序的 environment.rb 文件中。
在應用程序中包括 rjb 庫。
通過指定希望使用的 JAR 文件,裝載 RJB 和 JVM。
從希望使用的 JAR 文件中將類導入到 Ruby 變量並創建類的對象。
開始在 Rails 應用程序中使用剛剛創建的類,就像使用任何 Ruby 對象一樣 。
如果希望在 Rails 應用程序中重用已經使用 Java 代碼實現的業務邏輯, RJB 非常有用,並且不需要使用 Ruby 重新實現。它同時提供了 Ruby on Rails 和 Java 編程的優點。
考慮替代方法
還可以使用一種稱為 JRuby 的替代方法,它可以實現與 RJB 相同的目標。 JRuby 是使用 Java 語言實現的完整的 Ruby 包,使 Ruby 能夠運行在 JVM 之 上。使用 JRuby,您可以訪問所有 Java 庫。JRuby 要求安裝特定於 JRuby 的 Ruby Gems,因為針對非 Java 的 Ruby 的普通 Ruby Gems 不能與 JRuby 兼容 。
RJB 和 JRuby 各有優缺點。對於 JRuby,Ruby 全部在 JVM 之上運行,每個 Ruby 調用將經過 JVM,這將使執行變得非常緩慢。同樣,如果已經設置了一個 Rails 應用程序,需要從頭開始設置,以便 JRuby 訪問 Java 代碼。作為原生 Ruby 包,RJB 易於安裝,並且可以在已有的 Rails 設置中使用。如果需要在您 的 Rails 應用程序中快速調用一些 Java 代碼片段,那麼 RJB 是最好的選擇。
整體而言,在 Rails 應用程序中重用遺留 Java 代碼的能力非常有用。使用 Java 語言實現的設計和編寫都非常良好的業務邏輯不會擱置不用,相反,可以 在新的 Web 應用程序中繼續發揮有用的功能。
本文配套源碼