※当サイトの一部記事には広告を含みます。
うさねこ気まぐれPG開発室

Excel VBA 複数「列」を高速削除「Range・Union」使用で高速化する

複数「列」を削除高速化

うさこちゃん
うさこちゃん

条件に合う列を高速に削除したい場合にデータ量が大量にあると非常に削除が(かなり)遅い場合があります。
その場合は「Range」に削除データを貯めておいて一括して削除すると高速化できます。

①元データ

うさこちゃん
うさこちゃん

2行目に0のデータがある列を削除する。

ヘッダ1234567891011
条件21015005010
データデータデータデータデータデータデータデータデータデータデータデータ

②処理後データ

うさこちゃん
うさこちゃん

2行目の0のデータを列削除した結果。

ヘッダ1245810
条件211551
データデータデータデータデータデータデータ

処理速度の違い

うさこちゃん
うさこちゃん

今回は以下のデータ件数で約10倍の速度差になりました。
列データ500件
行データ100件
削除対象100件

サンプル①0.5273
サンプル②0.0625

サンプル①

Dim ObjThisSh  As Object
Dim LngColMax As Long
Set ObjThisSh = ThisWorkbook.Sheets("Sheet1")

With ObjThisSh

    LngColMax = .Cells(2, .Columns.Count).End(xlToLeft).Column

    For LngIndex = 2 To LngColMax
        If .Cells(2, LngIndex).Value = 0 Then
            '列を削除
            .Columns(LngIndex).Delete
        End If
    Next LngIndex

End With

サンプル② 高速化

Dim ObjThisSh  As Object
Dim LngColMax As Long
Dim LngIndex As Long
Dim RngDelete As Range
Set ObjThisSh = ThisWorkbook.Sheets("Sheet1")

With ObjThisSh

    LngColMax = .Cells(2, .Columns.Count).End(xlToLeft).Column

    For LngIndex = 2 To LngColMax
        If .Cells(2, LngIndex).Value = 0 Then
            If RngDelete Is Nothing Then
                Set RngDelete = .Cells(2, LngIndex)
            Else
                Set RngDelete = Union(RngDelete, .Cells(2, LngIndex))
            End If
        End If
    Next LngIndex

    If Not RngDelete Is Nothing Then
        'RngDelete.EntireColumn.Select 'テスト用 削除予定列を表示させる
        '列を一括削除
        RngDelete.EntireColumn.Delete
    End If

End With

サンプル③ さらに高速化したコード(配列判定方式)

改善ポイント

  • VarValues = Range(...).Value で行2を一気に配列化。
    .Cells() を何度も参照しないので 劇的に高速化
  • 判定は配列内で完結するため、Excelとのやり取り(COM呼び出し)が最小限。
  • 最後にまとめて Delete 実行。
Dim ObjThisSh  As Object
Dim LngColMax  As Long
Dim LngIndex   As Long
Dim RngDelete  As Range
Dim VarValues  As Variant

Set ObjThisSh = ThisWorkbook.Sheets("Sheet1")

With ObjThisSh
    ' 行2を配列に読み込み
    LngColMax = .Cells(2, .Columns.Count).End(xlToLeft).Column
    VarValues = .Range(.Cells(2, 2), .Cells(2, LngColMax)).Value

    ' 配列で判定(For文はセルにアクセスしないので速い)
    For LngIndex = 1 To UBound(VarValues, 2)
        If VarValues(1, LngIndex) = 0 Then
            If RngDelete Is Nothing Then
                Set RngDelete = .Cells(2, LngIndex + 1)
            Else
                Set RngDelete = Union(RngDelete, .Cells(2, LngIndex + 1))
            End If
        End If
    Next LngIndex

    ' 一括削除
    If Not RngDelete Is Nothing Then
        RngDelete.EntireColumn.Delete
    End If
End With

免責事項

本記事のサンプルプログラムは動作・結果を保証するものではありません。 利用により発生したいかなるトラブル・損害についても、当方は責任を負いません。


広告
PR
エレコム モニターアーム シングルアーム 17~32インチ対応 耐荷重:9kg ガス式 VESA規格対応 ホワイト DPA-SS08WH
エレコム モニターアーム シングルアーム 17~32インチ対応 耐荷重:9kg ガス式 VESA規格対応 ホワイト DPA-SS08WH
17~32インチサイズのモニターに対応するガススプリング式のシングルモニターアームです。
モニター下のスペースを有効活用してデスクの上を広く使うことができます。
軽い力で動かせるガススプリング式です。
モニターの位置を自在に動かせて、お好みのポジションにセットできます。
モニターを1台設置できるシングルアームタイプです。
可動域の広い5軸可動のアームタイプです。
VESA規格(75mm×75mm、100mm×100mm)に準拠したモニターに対応しています。
デスクの天板に挟んで固定するクランプ式と、天板に穴を開けてボルトを通して固定するグロメット式の2通りの取り付け方法が選べます。
モニター背面にネジをつけ、アームに取り付けたVESAマウントに下からネジ部分を引っ掛けて位置固定してご使用ください。
モニターの使用角度が決まったら、接続部のネジをしっかりと締めて固定してください。
Amazonで見る