ツイッター出題回答
オブジェクトの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 ・・・新着記事一覧を見る

イータ縮小ラムダ(eta reduced lambda)|エクセル入門(2023-11-20)
PIVOTBY関数(縦軸と横軸でグループ化して集計)|エクセル入門(2023-11-19)
GROUPBY関数(縦軸でグループ化して集計)|エクセル入門(2023-11-18)
PY関数(Pythonコードをセル内で実行)|エクセル入門(2023-11-17)
画像「セルに配置」のVBAについて(365の新機能)|VBA技術解説(2023-11-13)
スピルのゴーストの範囲を選択するVBA|ツイッター出題回答 (2023-10-31)
エクセル試験3:月間の所定労働時間|エクセル練習問題(2023-10-04)
エクセル試験2:所得税の計算|エクセル練習問題(2023-10-04)
エクセル試験1:曜日別の平均客単価|エクセル練習問題(2023-10-04)
列全体を指定する時のRangeとColumnsの違い|ツイッター出題回答 (2023-09-24)


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

1.最終行の取得(End,Rows.Count)|VBA入門
2.RangeとCellsの使い方|VBA入門
3.繰り返し処理(For Next)|VBA入門
4.セルのコピー&値の貼り付け(PasteSpecial)|VBA入門
5.変数宣言のDimとデータ型|VBA入門
6.条件分岐(IF)|VBA入門
7.ブックを閉じる・保存(Close,Save,SaveAs)|VBA入門
8.並べ替え(Sort)|VBA入門
9.マクロとは?VBAとは?VBAでできること|VBA入門
10.Range以外の指定方法(Cells,Rows,Columns)|VBA入門




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


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



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