###環境
Windows10 64bit / Visual Studio 2019
C++17
###経緯
以下コードの実行時に、
最初の IDispatch::Invoke() の引数用として準備した CComVariant 型の変数 params について、
2回目の IDispatch::Invoke() 実行時にその値が書き換わってしまう現象が発生しました。
該当の変数の初期化時には、取得するエクセルの Range オブジェクトのアドレスが格納されています。
これが、2回目の IDispatch::Invoke() 実行後に取得した result.parray の一番最初の要素に書き換わります。
掲題では値としましたが、参照先が result.parray[1][1] (最初の要素)と全く同じになります。
以下コードの通り、2回目の IDispatch::Invoke() の実行時には該当の変数 params を渡していません。
強いて言うなら、params を利用して取得した CComDispatchDriver 型の range を渡しています。
また、以下コードでは1回目と2回目の実行時に同じ変数 result が渡されていますが、
実際は1回目と2回目の実行でルーチンを分離しており、この事象の発生とは無関係であることを確認しています。
ルーチンを分離せず、以下の通りに実行しても同様の事象が発生することは確認済みです。
###知りたいこと
1.この事象の原因は何でしょうか。
2.この事象が発生しない為にはどの様にすれば良いでしょうか。
###コード
※値が格納されない変数については、コメントの通りの値が格納されているものと仮定して下さい。
C++
1int main() { 2 CComVariant params; 3 params.vt = VT_BSTR; 4 params.bstrVal = CComBSTR(L"A1:B2"); 5 6 DISPPARAMS disp_params = { nullptr,nullptr,NULL,NULL }; 7 disp_params.cArgs = 1; 8 disp_params.rgvarg = ¶ms; 9 10 CComDispatchDriver worksheet; //エクセルのワークシートの IDispatch*。 11 DISPID dispid; // Worksheet.Range の DISPID。 12 CComVariant result; 13 HRESULT hr = worksheet->Invoke(dispid, IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_PROPERTYGET, &disp_params, &result, nullptr, nullptr); 14 15 CComDispatchDriver range = result.pdispVal; 16 result.Clear(); 17 18 // dispid を Range.Value の DISPID へ変更。 19 hr = range->Invoke(dispid, IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_PROPERTYGET, nullptr, &result, nullptr, nullptr); 20 21 //この時点で、params の値が "A1:B2" から、セルA1の値に変更されている。 22 23 CComSafeArray<VARIANT> source = result.parray; 24}
###補足
知りたいことの項番2については自身でも検証中ですので、解決したらこちらに追記します。
ただ、項番1については調べた限りでは原因が特定できずにおりますので、お力添え頂けますと幸甚です。
ちなみにこの事象で困っていることですが、
result を CComVariant 型にしている為、result のスコープを抜けると result に VariantClear が実行されます。
その後で params のスコープを抜ける際、こちらも CComVariant 型ですので VariantClear をするわけですが、
既に result.parray[1][1] に対して SysFreeString が実行されている為、例外が投げられてしまいます。
ですので、短絡的には params を VARIANT 型にして、VariantClear を明示しなければ解決とは思いますが、
当事象が必ず発生することが確約できない限り、解放を明示しないのは宜しくないと考えた為質問させて頂いております。
###追記
知りたいことの項番2について、
2回目の IDispatch::Invoke() の実行前に params.Clear() を実行することで事象は発生しませんでした。
回答1件
あなたの回答
tips
プレビュー