VBA から Java のクラスを呼び出すようなプログラムを組みました。
呼び出す Java のクラス(jarファイル)は J2SE 5.0 で作成したものです。
ゆえに、J2SE Runtime Environment 5.0 の java.exe を起動しなければなりません。
しかし、パスを指定せずに java.exe を起動した場合、他のバージョンの java.exe が起動し、java.lang.UnsupportedClassVersionError などが発生し、処理が失敗してしまう可能性があります。
そのため、J2SE Runtime Environment 5.0 の java.exe を決め撃ちで起動したいのですが、そのインストール場所は環境によって異なります。
確実にインストール場所を突き止め、java.exe を補足するにはどうすればいいか...どうやらレジストリに頼るしかないようです。
J2SE Runtime Environment 5.0 のインストール場所は HKEY_LOCAL_MACHINE\SOFTWARE\JavaSoft\Java Runtime Environment\1.5 の JavaHome に格納されています。
このキーの値で示されるディレクトリの直下の BIN ディレクトリに java.exe は存在するはずです。
では、VB や VBA でレジストリから特定のキーの値を取得するにはどうすればいいのか...
どうやら そのための API(advapi32.dll)が提供されているようです。
Microsoft のナレッジベースにその方法についての解説がありました。
操作方法使用レジストリ設定を保存してそして取得する API
(自動翻訳のためか、意味不明な日本語タイトルになっています^^;)
上記の解説で挙げられているサンプルコードをほとんど丸々流用しただけですが、一応、値の取得だけに絞込み、使いやすいように若干のアレンジを加えたものが以下のコードになります。
Option Explicit Public Const REG_SZ As Long = 1 Public Const REG_DWORD As Long = 4 Public Const HKEY_CLASSES_ROOT = &H80000000 Public Const HKEY_CURRENT_USER = &H80000001 Public Const HKEY_LOCAL_MACHINE = &H80000002 Public Const HKEY_USERS = &H80000003 Public Const ERROR_NONE = 0 Public Const ERROR_BADDB = 1 Public Const ERROR_BADKEY = 2 Public Const ERROR_CANTOPEN = 3 Public Const ERROR_CANTREAD = 4 Public Const ERROR_CANTWRITE = 5 Public Const ERROR_OUTOFMEMORY = 6 Public Const ERROR_ARENA_TRASHED = 7 Public Const ERROR_ACCESS_DENIED = 8 Public Const ERROR_INVALID_PARAMETERS = 87 Public Const ERROR_NO_MORE_ITEMS = 259 Public Const KEY_QUERY_VALUE = &H1 Public Const KEY_SET_VALUE = &H2 Public Const KEY_ALL_ACCESS = &H3F Public Const REG_OPTION_NON_VOLATILE = 0 Declare Function RegCloseKey Lib "advapi32.dll" ( _ ByVal hKey As Long _ ) As Long Declare Function RegOpenKeyEx Lib "advapi32.dll" Alias "RegOpenKeyExA" ( _ ByVal hKey As Long, _ ByVal lpSubKey As String, _ ByVal ulOptions As Long, _ ByVal samDesired As Long, _ phkResult As Long _ ) As Long Declare Function RegQueryValueExString Lib "advapi32.dll" Alias "RegQueryValueExA" ( _ ByVal hKey As Long, _ ByVal lpValueName As String, _ ByVal lpReserved As Long, _ lpType As Long, _ ByVal lpData As String, _ lpcbData As Long _ ) As Long Declare Function RegQueryValueExLong Lib "advapi32.dll" Alias "RegQueryValueExA" ( _ ByVal hKey As Long, _ ByVal lpValueName As String, _ ByVal lpReserved As Long, _ lpType As Long, _ lpData As Long, _ lpcbData As Long _ ) As Long Declare Function RegQueryValueExNULL Lib "advapi32.dll" Alias "RegQueryValueExA" ( _ ByVal hKey As Long, _ ByVal lpValueName As String, _ ByVal lpReserved As Long, _ lpType As Long, _ ByVal lpData As Long, _ lpcbData As Long _ ) As Long Function QueryValue(hKey As Long, sKeyName As String, sValueName As String) As Variant Dim lRetVal As Long 'result of the API functions lRetVal = RegOpenKeyEx(hKey, sKeyName, 0, KEY_QUERY_VALUE, hKey) lRetVal = QueryValueEx(hKey, sValueName, QueryValue) RegCloseKey (hKey) End Function Function QueryValueEx(ByVal lhKey As Long, ByVal szValueName As String, vValue As Variant) As Long Dim cch As Long Dim lrc As Long Dim lType As Long Dim lValue As Long Dim sValue As String On Error GoTo QueryValueExError ' Determine the size and type of data to be read lrc = RegQueryValueExNULL(lhKey, szValueName, 0&, lType, 0&, cch) If lrc <> ERROR_NONE Then Error 5 Select Case lType ' For strings Case REG_SZ: sValue = String(cch, 0) lrc = RegQueryValueExString(lhKey, szValueName, 0&, lType, _ sValue, cch) If lrc = ERROR_NONE Then vValue = Left$(sValue, cch - 1) Else vValue = Empty End If ' For DWORDS Case REG_DWORD: lrc = RegQueryValueExLong(lhKey, szValueName, 0&, lType, _ lValue, cch) If lrc = ERROR_NONE Then vValue = lValue Case Else 'all other data types not supported lrc = -1 End Select QueryValueExExit: QueryValueEx = lrc Exit Function QueryValueExError: Resume QueryValueExExit End Function
このようなファンクションや定数を定義しておくと、あとは以下のサンプルコードのようにキーを引数にQueryValueというファンクションをコールして値を取得することができるというわけです。
Sub main() Dim javaHome As Variant javaHome = QueryValue(HKEY_LOCAL_MACHINE, "SOFTWARE\JavaSoft\Java Runtime Environment\1.5", "JavaHome") If javaHome = "" Then MsgBox "Java Runtime Environment 5.0 Not Found", vbCritical, "Error" Else MsgBox javaHome End If End Subレジストリを参照するなんて、できればやりたくはないですが、どうしてもやらざるを得ない場合はこんな感じで...