ツイッター出題回答
オブジェクトのByRef、ByVal、Variant

ExcelマクロVBAとエクセル関数についての私的雑感
公開日:2022-06-22 最終更新日:2022-06-22

オブジェクトのByRef、ByVal、Variant


ツイッターで出題した問題です。


オブジェクトを引数にした場合の、ByRef、ByValの違いと、仮引数にVariantを指定した場合の動作違いを問う問題です。


出題ツイート

【VBA問題】
Sub main()
Range("A1") = 1
Dim a As Range: Set a = Range("A1")
Call func(a)
Debug.Print a
End Sub
Function func(□ b As △)
b = 2
End Function

mainを実行した結果、「2」と出力されました。
□と△の組み合わせで間違っているものはどれか?

□:ByVal , △:Range
□:ByVal , △:Variant
□:ByRef , △:Range
□:ByRef, △:Variant

Excel VBA問題
https://twitter.com/yamaoka_ss/status/1539118680809799683


解説ツイート

案外割れましたね。
正解は
「□:ByVal , △:Variant」
まず、
ByRefは参照渡しです。
ByValは値渡しです。
https://excel-ubara.com/excelvba4/EXCEL218.html
ByValが値渡し、ByRefが参照渡しです。ここまでは、どこにでも書いてあります。しかし、なんとなく理解できるけど、なんとなく理解できない… 結局のところ実際の活用がなかなか出来ない事が多いようです。


ByRef参照渡しの場合は、変数そのものを呼ぶ側(実引数)と呼ばれる側(仮引数)で共有しているので、funcで変更したものがmainでも変更された状態で戻ります。
ByRef値渡しの場合は、呼ぶ側の変数の中の値を渡しているので、funcで変更したものはmainに戻りません。


しかし、引数がオブジェクトの場合はちょっと事情が異なります。
オブジェクト変数の中の値は、オブジェクトのアドレスが入っています。
同じ値のアドレスは同じオブジェクトを参照します。
つまりByVal値渡しでも同じ値のアドレス先のオブジェクトは同じものになります。


従ってByVal値渡しでも、オブジェクトの場合は、呼び出し先の変更が呼び出し元にも影響します。
ここまでは、一般的なByRef,ByRefの説明でした。
今回の問題の厄介なところは、
ByValでオブジェクトを渡しているが、受け取り側(仮引数)がVariantの場合にどうなるかです。


funcではRangeオブジェクトを受け取っています。これはRange型を指定した場合と同じです。
問題は代入式です。
型Rangeの変数なら、プロパティを省略した場合はValueになります。
しかし、型Variantへの代入では、代入時点で変数が作り直されます。
単純な例.
Set v = Range("A1")
v = 2
エクセルの神髄


この例で解ると思いますが、vは2(Valiant/Integer)になります。
「□:ByVal , △:Variant」の場合は、これと同じことが発生しています。
これを避けたい場合は、受け取り側で何らかの対処が必要になります。
引数がオブジェクトかどうかで処理を分岐させたりが必要です。
エクセルの神髄


If TypeName(b) = "Range" Then
  b.Value = 2
Else
  b = 2
End If
Rangeしか入ってこないのであれば、分岐させずに.Valueを付けてしまえば良いのですが、それなら最初から型をRangeで指定すれば良い事になります。
基本的には、ByRefの場合は型を一致させた方が良いと思います。
以上です。


エクセル VBA問題
https://twitter.com/yamaoka_ss/status/1539493559736205312




同じテーマ「ツイッター出題回答 」の記事

数値変数の値を別の変数を使わずに入れ替える
Rangeオブジェクトを受け取り"行数,列数"で埋める
数式の関数の使用回数、関数名を配列で返す
日付型と通貨型のValueとValue2について
小文字"abc"を大文字"ABC"に変換する方法
オブジェクトのByRef、ByVal、Variant
「マクロの登録」で登録できないプロシージャーは?
ジャグ配列から順列を作成する
シート内の全テーブルを1つに統合
VBA穴埋め問題「On Error GoToの挙動」
数珠順列(配置に条件付き)を全て出力する


新着記事NEW ・・・新着記事一覧を見る

付録:AI×VBA実践リソース集|生成AI活用研究(2025-05-25)
おわりに:AI×VBAはあなたの未来を変える強力な武器|生成AI活用研究(2025-05-25)
第7章:さらなる高みへ!AI×VBA応用テクニックと未来への備え|生成AI活用研究(2025-05-24)
第6章:AIとVBAを学び続ける!あなたの「超効率化」を止めないヒント|生成AI活用研究(2025-05-22)
第5章:AI×VBAでつまづかない!トラブルシューティングとAIとの付き合い方 |生成AI活用研究(2025-05-21)
第4章:【事例で学ぶ】AIとVBAでExcel作業を劇的に効率化する! |生成AI活用研究(2025-05-20)
第3章:AIを「自分だけのVBA先生」にする!質問・相談の超実践テクニック|生成AI活用研究(2025-05-19)
第2章 VBAって怖くない!Excelを「言葉で動かす」(超入門)|生成AI活用研究(2025-05-18)
第1章:AIって一体何?あなたのExcel作業をどう変える?(AI超基本)|生成AI活用研究(2025-05-18)
AI時代のExcel革命:AI×VBAで“書かない自動化”超入門|生成AI活用研究(2025-05-17)


アクセスランキング ・・・ ランキング一覧を見る

1.最終行の取得(End,Rows.Count)|VBA入門
2.繰り返し処理(For Next)|VBA入門
3.変数宣言のDimとデータ型|VBA入門
4.セルのコピー&値の貼り付け(PasteSpecial)|VBA入門
5.ひらがな⇔カタカナの変換|エクセル基本操作
6.RangeとCellsの使い方|VBA入門
7.メッセージボックス(MsgBox関数)|VBA入門
8.セルのクリア(Clear,ClearContents)|VBA入門
9.FILTER関数(範囲をフィルター処理)|エクセル入門
10.条件分岐(Select Case)|VBA入門




このサイトがお役に立ちましたら「シェア」「Bookmark」をお願いいたします。


記述には細心の注意をしたつもりですが、
間違いやご指摘がありましたら、「お問い合わせ」からお知らせいただけると幸いです。
掲載のVBAコードは動作を保証するものではなく、あくまでVBA学習のサンプルとして掲載しています。
掲載のVBAコードは自己責任でご使用ください。万一データ破損等の損害が発生しても責任は負いません。



このサイトがお役に立ちましたら「シェア」「Bookmark」をお願いいたします。
本文下部へ