当ページのリンクには一部広告が含まれています。
うさねこ気まぐれ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

免責事項

本記事のサンプルプログラムは実行を保証するものではなく、利用に伴う結果について一切の責任を負いません。


広告