Office の 32 ビット バージョンと 64 ビット バージョン間の互換性

32 ビット バージョンの Office と 64 ビット バージョンの Office の互換性についてご確認ください。

Office アプリケーションは、32 ビット バージョンと 64 ビット バージョンでご利用いただけます。

64 ビット バージョンの Office により、Microsoft Excel 2010 で大量のデータを扱う場合など、機能の強化に伴ってより多くのデータを移動させることができるようになりました。 32 ビット コードを記述する場合は、変更なしで 64 ビット バージョンの Office を使うことができます。 しかし、64 ビット コードを記述する場合は、そのコードに特定のキーワードと条件付きコンパイル定数を含めることで、以前のバーションの Office との下位互換性が確保されるようにし、32 ビット コードと 64 ビット コードを組み合わせて使う場合に正しいコードが実行されるようにする必要があります。

Visual Basic for Applications 7.0 (VBA 7) は 64 ビット バージョンの Office でリリースされ、32 ビット バージョンと 64 ビット バージョンのどちらのアプリケーションでも動作します。 この記事で説明されている変更は、64 ビット バージョンの Office にのみ適用されます。 32 ビット バージョンの Microsoft Office を使うと、追加の変更を加えることなく、以前のバージョンの Office で作成したソリューションを使うことができます。

注:

既定では、64 ビット バージョンの Office をインストールすると、32 ビット バージョンをインストールすることはできません。 Microsoft Office 64 ビット バージョンのインストール オプションを明示的に選択する必要があります。

VBA 7 では、既存の Windows API ステートメント (Declare ステートメント) を更新して、64 ビット バージョンで動作するようにしなければなりません。 さらに、これらのステートメントで使用されるアドレス ポインターとディスプレイ ウィンドウ ハンドルをユーザー定義型で更新する必要があります。 この点については、32 ビット バージョンと 64 ビット バージョン間の互換性に関する問題と推奨される解決案と共に、この記事で詳しく説明します。

32 ビット システムと 64 ビット システムの比較

64 ビット バージョンの Office で作成されたアプリケーションは、32 ビットバージョンよりも大きいアドレス空間を参照することができます。 つまり、以前と比べると、データに対してより多くの物理メモリを使うことができるため、物理メモリでのデータの出入りにかかるオーバーヘッドを軽減できる可能性があります。

物理メモリ内の特定の場所への参照 ("ポインター" と呼ばれる) に加えて、表示ウィンドウ識別子 ("ハンドル" と呼ばれる) を参照するアドレスを使うこともできます。 32 ビット システムと 64 ビット システムのどちらを使うかに応じて、ポインターやハンドルのサイズ (バイト数) が決まります。

既存のソリューションを 64 ビット バージョンの Office を使って実行する場合は、次の点に注意してください。

  • Office のネイティブの 64 ビット プロセスは、32 ビット バイナリを読み込むことができません。 既存の Microsoft ActiveX コントロールと既存のアドインがある場合、これはよく起こる問題です。

  • これまで VBA にはポインター データ型がなかったため、ポインターとハンドルを格納する 32 ビットの変数を使う必要がありました。 Declare ステートメントを使う場合、API 呼び出しで返される 64 ビット値は、これらの変数で切り詰められるようになりました。

VBA 7 コード ベース

VBA 7 は、Office 2007 以前のバージョンでの VBA コード ベースを置き換えます。 VBA 7 は、Office の 32 ビットと 64 ビットのどちらのバージョンでもご利用いただけます。 次の 2 つの条件付きコンパイル定数があります。

  • VBA7 - アプリケーションが VBA 7 を使っているか、以前のバージョンの VBA を使っているかをテストすることによって、コードの下位互換性が確保されます。

  • Win64 32 ビットと 64 ビットのどちらでコードが実行されているかをテストします。

特定の例外を除き、32 ビット バージョンのアプリケーションで機能しているドキュメント内のマクロは 64 ビット バージョンでも機能します。

ActiveX コントロールと COM アドインの互換性

既存の 32 ビット ActiveX コントロールは Office の 64 ビット バージョンと互換性がありません。 ActiveX コントロールと COM オブジェクトについては、次のようにしてください。

  • ソース コードがあれば、64 ビット バージョンを自分で生成します。
  • ソース コードがなければ、販売元に更新されたバージョンを請求してください。

Office のネイティブの 64 ビット プロセスでは 32 ビットのバイナリを読み込むことができません。 これには、 MSComCtl の一般的なコントロール (TabStrip、Toolbar、StatusBar、ProgressBar、TreeView、ListViews、ImageList、Slider、ImageComboBox) と、 MSComCt2 のコントロール (Animation、UpDown、MonthView、DateTimePicker、FlatScrollBar) が含まれます。 これらのコントロールは、Office 2010 より前の 32 ビット バージョンの Office でインストールされました。 64 ビット バージョンの Office にコードを移行するときにこれらのコントロールを使う既存の VBA ソリューションの代替策を見つける必要があります。

API の互換性

VBA とタイプ ライブラリの組み合わせにより、Office アプリケーションを作成するための多くの機能が提供されます。 ただし、メモリやプロセスを管理する場合、UI 要素を使用してウィンドウとコントロールをリンクする場合、または Windows レジストリを変更する場合など、コンピューターのオペレーティング システムやその他のコンポーネントと直接通信する必要がある場合があります。 このようなシナリオでは、DLL ファイルに埋め込まれている外部関数のいずれかを使用することをお勧めします。 これを VBA で行うには、 Declare ステートメントを使用して API 呼び出しを行います。

注:

Microsoft は、1,500 個の Declare ステートメントが含まれている Win32API.txt ファイルと、コードに含める Declare ステートメントをコピーするツールを提供しています。 ただし、これらのステートメントは 32 ビット システム用であるため、この記事で後ほど説明する情報に従って 64 ビットに変換する必要があります。 既存の Declare ステートメントは、PtrSafe 属性を使って 64 ビットに対して安全なステートメントとしてマークするまでは、64 ビットの VBA にコンパイルされません。 この種類の変換の例については、Excel MVP Jan Karel Pieterse の Web サイト (https://www.jkp-ads.com/articles/apideclarations.asp) をご覧ください。 Office Code Compatibility Inspector ユーザーズ ガイドは、PtrSafe 属性と適切な戻り値の型 (必要に応じて) の API Declare ステートメントの構文を検査するのに役立ちます。

Declare ステートメントの形式は、サブルーチン (戻り値がない) と関数 (戻り値がある) のどちらを呼び出すのかに応じて、次のどちらかになります。

Public/Private Declare Sub SubName Lib "LibName" Alias "AliasName" (argument list)
Public/Private Declare Function FunctionName Lib "Libname" alias "aliasname" (argument list) As Type

SubName 関数または FunctionName 関数は、プロシージャを VBA コードから呼び出すときに使用する名前を表しており、DLL ファイル内のプロシージャの実際の名前に置き換えます。 プロシージャの名前として AliasName 引数を指定することもできます。 Lib キーワードの後ろには、呼び出すプロシージャが含まれている DLL ファイルの名前を指定します。 引数リストには、プロシージャに渡す必要のあるパラメーターとデータ型を指定します。

次の Declare ステートメントは、Windows レジストリ内のサブキーを開き、その値を置き換えます。

Declare Function RegOpenKeyA Lib "advapi32.dll" (ByVal Key As Long, ByVal SubKey As String, NewKey As Long) As Long

RegOpenKeyA 関数の Windows.h (ウィンドウ ハンドル) エントリは次のようになります。

LONG RegOpenKeyA ( HKEY hKey, LPCSTR lpSubKey, HKEY *phkResult );

Visual C と Microsoft Visual C++ では、上記の例は 32 ビットと 64 ビットのどちらについても正しくコンパイルされます。 なぜなら、HKEY がポインターとして定義されており、そのサイズはコードをコンパイルするプラットフォームのメモリ サイズを反映するからです。

以前のバージョンの VBA では、特定のポインター データ型がないため、 Long データ型が使用されていました。 また、Long データ型は常に 32 ビットであるため、上位 32 ビットが切り捨てられたり、他のメモリ アドレスが上書きされたりする可能性があるため、64 ビット メモリを持つシステムで使用すると、これは中断されます。 どちらの状況でも、予期しない動作やシステム クラッシュが発生する可能性があります。

これを解決するために、VBA には真の ポインター データ型 LongPtr が含まれています。 この新しいデータ型を使用すれば、次のように本来の Declare ステートメントを正しく書くことができます。

Declare PtrSafe Function RegOpenKeyA Lib "advapire32.dll" (ByVal hKey as LongPtr, ByVal lpSubKey As String, phkResult As LongPtr) As Long

このデータ型と新しい PtrSafe 属性を使うと、この Declare ステートメントを 32 ビット システムでも 64 ビット システムでも使うことができます。 PtrSafe 属性は、 Declare ステートメントの対象を 64 ビット バージョンの Office とすることを VBA コンパイラに指示するものです。 この属性を指定せずに Declare ステートメントを 64 ビット システムで使うと、コンパイル時エラーが発生します。 PtrSafe 属性は、32 ビット バージョンの Office ではオプションです。 これにより、既存の Declare ステートメントは従来どおりに動作します。

次の表では、新しい修飾子とデータ型に加え、1 つのデータ型、2 つの変換演算子、3 つの関数についても説明します。

種類 アイテム 説明
修飾子
PtrSafe
Declare ステートメントが 64 ビットと互換性があることを示します。 この属性は 64 ビット システムでは必須です。
データ型
LongPtr
32 ビット バージョンでは 4 バイトのデータ型、Microsoft Office の 64 ビット バージョンでは 8 バイトのデータ型である変数データ型。 これは、新しいコードのポインターまたはハンドルを宣言する推奨される方法ですが、64 ビット バージョンの Office で実行する必要がある場合は、レガシ コードにも使用できます。 これは、32 ビットと 64 ビットの VBA 7 ランタイムでのみサポートされます。 数値は割り当てることができますが、数値型は割り当てられないことに注意してください。
データ型
LongLong
これは 64 ビット バージョンの Microsoft Office でのみ使用できる 8 バイトのデータ型です。 数値を代入することはできますが、数値型を代入することはできません (切り詰められるのを避けるためです)。
変換演算子
CLngPtr
単純な式を LongPtr データ型に変換します。
変換演算子
CLngLng
単純な式を LongLong データ型に変換します。
職務
VarPtr
バリアント コンバーター。 64 ビット バージョンでは LongPtr を返し、32 ビット バージョンでは Long を返します (4 バイト)。
関数
ObjPtr
オブジェクト コンバーター。 64 ビット バージョンでは LongPtr を返し、32 ビット バージョンでは Long を返します (4 バイト)。
関数
StrPtr
文字列コンバーター。 64 ビット バージョンでは LongPtr を返し、32 ビット バージョンでは Long を返します (4 バイト)。

次の例は、これらのアイテムのいくつかについて、 Declare ステートメントでの使用方法を示したものです。

Declare PtrSafe Function RegOpenKeyA Lib "advapi32.dll" (ByVal Key As LongPtr, ByVal SubKey As String, NewKey As LongPtr) As Long

Declare ステートメントで PtrSafe を指定しないと、64 ビット バージョンの Office と互換性がないものと見なされます。

VBA7Win64 という 2 つの条件付きコンパイル定数があります。 以前のバージョンの Microsoft Office との下位互換性を確保するには、 VBA7 定数を使って (これがより一般的なやり方です)、以前のバージョンの Office で 64 ビット コードが実行されないようにします。 32 ビット バージョンと 64 ビット バージョンとでコードが異なる場合は (たとえば、64 ビット バージョンについては LongLong を使い、32 ビット バージョンについては Long を使う数値演算 API を呼び出す場合など)、 Win64 定数を使います。 次のコードで、この 2 つの定数の使用例を示します。

#if Win64 then
   Declare PtrSafe Function MyMathFunc Lib "User32" (ByVal N As LongLong) As LongLong
#else
   Declare Function MyMathFunc Lib "User32" (ByVal N As Long) As Long
#end if
#if VBA7 then
   Declare PtrSafe Sub MessageBeep Lib "User32" (ByVal N AS Long)
#else
   Declare Sub MessageBeep Lib "User32" (ByVal N AS Long)
#end if

要約すると、64 ビット コードを記述し、以前のバージョンの Office で使用する場合は、 VBA7 条件付きコンパイル定数を使用します。 ただし、Office で 32 ビット コードを記述した場合、そのコードは、コンパイル定数を必要とせずに、以前のバージョンの Office と同様に機能します。 32 ビット バージョンに 32 ビット ステートメントを使用し、64 ビット バージョンに 64 ビット ステートメントを使用する場合は、 Win64 条件付きコンパイル定数を使用することをお勧めします。

条件付きコンパイル属性の使用

次に、更新する必要がある 32 ビット用に記述された VBA コードの例を示します。 このレガシー コードのデータ型は、ハンドルやポインターを参照しているため、 LongPtr を使うように更新されていることに注意してください。

32 ビット バージョン用に記述された VBA コード

Declare Function SHBrowseForFolder Lib "shell32.dll" _
  Alias "SHBrowseForFolderA" (lpBrowseInfo As BROWSEINFO) As LongPtr
  
Public Type BROWSEINFO
  hOwner As Long
  pidlRoot As Long
  pszDisplayName As String
  lpszTitle As String
  ulFlags As Long
  lpfn As Long
  lParam As Long
  iImage As Long
End Type

64 ビット バージョン用に記述された VBA コード

#if VBA7 then    ' VBA7 
Declare PtrSafe Function SHBrowseForFolder Lib "shell32.dll" _
  Alias "SHBrowseForFolderA" (lpBrowseInfo As BROWSEINFO) As Long
Public Type BROWSEINFO
  hOwner As LongPtr
  pidlRoot As Long
  pszDisplayName As String
  lpszTitle As String
  ulFlags As Long
  lpfn As LongPtr
  lParam As LongPtr
  iImage As Long
End Type
 
#else    ' Downlevel when using previous version of VBA7
Declare Function SHBrowseForFolder Lib "shell32.dll" _
  Alias "SHBrowseForFolderA" (lpBrowseInfo As BROWSEINFO) As Long
Public Type BROWSEINFO
  hOwner As Long
  pidlRoot As Long
  pszDisplayName As String
  lpszTitle As String
  ulFlags As Long
  lpfn As Long
  lParam As Long
  iImage As Long
End Type
 
#end if
Sub TestSHBrowseForFolder ()
    Dim bInfo As BROWSEINFO
    Dim pidList As Long
    bInfo.pidlRoot = 0&
    bInfo.ulFlags = &H1
    pidList = SHBrowseForFolder(bInfo)
End Sub

よく寄せられる質問

どのような場合に 64 ビット バージョンの Office を使うべきですか。

使っているホスト アプリケーション (Excel、Word など) によって異なります。 たとえば、Excel を 64 ビット バージョンの Microsoft Office で使うと、より大きなワークシートを処理できます。

64 ビット バージョンと 32 ビット バージョンの Office を同時にインストールできますか。

いいえ。

どのような場合に Long パラメーターを LongPtr に変換する必要がありますか。

呼び出す関数については、Microsoft Developers Network の Windows API ドキュメントを確認する必要があります。 ハンドルとポインターは LongPtr に変換する必要があります。 例として、RegOpenKeyA 用のドキュメントには、次の署名があります。

LONG WINAPI RegOpenKeyEx(
  __in        HKEY hKey,
  __in_opt    LPCTSTR lpSubKey,
  __reserved  DWORD ulOptions,
  __in        REGSAM samDesired,
  __out       PHKEY phkResult
);

パラメーターは次のように定義されます。

パラメーター 説明
hKey [in]
レジストリ キーを開くハンドル
lpSubKey [in, optional]
開くレジストリ サブキーの名前。
ulOptions
このパラメーターは予約済みで、0 にする必要があります。
samDesired [in]
キーに対する必要なアクセス権を指定するマスク。
phkResult [out]
キーを開くハンドルを受け取る変数へのポインター

Win32API_PtrSafe.txt では、Declare ステートメントが次のように定義されています。

Declare PtrSafe Function RegOpenKeyEx Lib "advapi32.dll" Alias "RegOpenKeyExA" (ByVal hKey As LongPtr , ByVal lpSubKey As String, ByVal ulOptions As Long, ByVal samDesired As Long, phkResult As LongPtr ) As Long

構造体のポインターとハンドルを変換する必要がありますか。

はい。 Win32API_PtrSafe.txt の MSG 型をご覧ください。

Type MSG
    hwnd As LongPtr 
    message As Long
    wParam As LongPtr 
    lParam As LongPtr 
    time As Long
    pt As POINTAPI
End TypeF

strptr、varpt、objptr は、どのような場合に使用すべきですか。

これらの関数を使用して、それぞれ文字列、変数、オブジェクトへのポインターを取得する必要があります。 64 ビット バージョンの Office では、これらの関数は 64 ビット の LongPtr を返します。これは Declare ステートメントに渡すことができます。 これらの関数の使用は、以前のバージョンの VBA から変更されていません。 唯一の違いは、 LongPtr が返される点です。

関連項目