<Array to CSV> 「配列」を「CSV」へ。Excelで保存。指数表示対策済み。
以下のマクロを使用していて、問題が発生しました。
以下のようなシートを、配列に入れ、CSVファイルに入れるマクロです。ここで、B4セルに注目してください。中身は、「100000000000」ですが、セルには「1E+11」と指数表示となっています。12桁以上になると、指数表示なるようです。
配列に入れるまでは、以下の通り問題ありません。
このまま、CSV形式で保存すると、以下の通り、指数表示のまま、保管されてしまいます。
これを回避するための、マクロが、以下です。
Dim arr() As Variant
Private Sub test、シートを配列に入れ、csvファイルに出力()
' (ws, ByRef arr, Optional RejectRw = 0)
Call sheet_to_array("TEST", arr)
' (ByRef arr, FileName, Optional 種類)
Call array_to_csv_Excel(arr, ThisWorkbook.Path & "\sample5.csv")
End Sub
'*******
'* 二次元配列をCSVファイルに出力、高速化。20230912
'* https://excel-vba.work/2021/01/03/%E3%80%90vba%E3%80%91%E6%8C%87%E5%AE%9A%E3%81%97%E3%81%9F%E7%AF%84%E5%9B%B2%E3%82%92csv%E3%83%95%E3%82%A1%E3%82%A4%E3%83%AB%E3%81%B8%E5%87%BA%E5%8A%9B%E3%81%99%E3%82%8B/
'* arr:二次元配列。これが、csvファイルに出力される
'* FileName:csvファイル名、フルパス
Private Sub array_to_csv_Excel(ByRef arr As Variant, ByVal FileName As String)
Dim csvFile As String
' Dim targetRange As Range
Dim wb As Workbook
Dim fso As Object
'出力するCSVファイルを指定
csvFile = FileName
'CSVファイルへ出力する範囲を指定 ※例としてシート「sample」のセル「B2」から続く一連の範囲を指定
'Set targetRange = Worksheets("sample").Range("B2").CurrentRegion
'Set targetRange = arr
Set fso = CreateObject("Scripting.FileSystemObject")
'出力するCSVファイルが既に存在する場合は削除
If fso.FileExists(csvFile) Then
If IsBookOpened(csvFile) Then
Dim rc As Integer
rc = MsgBox("ファイルが開いています。" & vbCrLf & _
"はい:閉じて、削除して、処理を続けます。" & vbCrLf & _
"いいえ:処理を中断します", vbYesNo + vbQuestion, "確認")
If rc = vbYes Then
Dim FileNameOnly As String
FileNameOnly = fso.GetFileName(csvFile)
Windows(FileNameOnly).Close
Else
End
End If
End If
' ファイル削除
fso.DeleteFile csvFile
End If
'新規ブックを作成
Set wb = Workbooks.Add
' 配列をブックにコピー
' wb.Worksheets(1).Range("A1").Resize(UBound(arr), UBound(arr, 2)) = arr この行は削除
' =======================================================以下が今回追加
' 配列をブックにコピー
Dim ws As Worksheet
Set ws = wb.Worksheets(1)
ws.Range("A1").Resize(UBound(arr), UBound(arr, 2)) = arr
' 1列目全体の表示形式を文字列にするオブジェクト式
ws.Columns(1).NumberFormatLocal = "@"
' 指数表示から文字列に変換
Dim rw As Long
For rw = 2 To UBound(arr)
Cells(rw, 1).Value = Format(Cells(rw, 1))
Next rw
' =======================================================ここまでが今回追加
'新規ブックをCSVファイルとして出力
wb.SaveAs FileName:=csvFile, FileFormat:=xlCSV, Local:=True
' 「xlCSV」・・・・・文字コードは「Shift_JIS」となります。
' 「Local:=」に「True」を指定することで、「日付の形式」を「Excelの言語設定で指定されている言語(日本語と思います)」にします
'新規ブックを保存せずに閉じる
wb.Close SaveChanges:=False
'後片付け
Set fso = Nothing
End Sub
'*******
'* 「シート」を「配列」に入れるプロシージャ。20230707
'* SheetName:シート名前
'* myArray:出力される配列は、参照渡し。配列は、モジュールレベル変数としている。
'* RejectRw=1とすると頭の1行目を除いた表が、配列に入る。1行目の見出しが不要な場合に使用。
Sub sheet_to_array(ByVal SheetName As String, ByRef myArray As Variant, Optional RejectRw As Long = 0)
' 表を一気に変数にいれて処理
Dim Target As Range
Set Target = Worksheets(SheetName).UsedRange
With Target
If .Rows.Count > RejectRw And RejectRw > 0 Then
' 最初の行数(RejectRw)を除外する
Set Target = .Resize(.Rows.Count - RejectRw).Offset(RejectRw)
End If
End With
'Variant型の変数に値を一気に代入する
myArray = Target.Value
End Sub
Public Function IsBookOpened(ByVal FilePath As String) As Boolean
On Error Resume Next
Open FilePath For Append As #1
Close #1
If Err.Number > 0 Then
IsBookOpened = True
Else
IsBookOpened = False
End If
End Function
CSV形式で保管する直前のExcelが以下です。
以下のマクロで、A列の表示型式をを文字列に設定して、A列の値をセルごとに指数表示から文字列に変換しています。それで、無事、指数表示から文字列に変換されました。
' 1列目全体の表示形式を文字列にするオブジェクト式
ws.Columns(1).NumberFormatLocal = "@"
' 指数表示から文字列に変換
Dim rw As Long
For rw = 2 To UBound(arr)
Cells(rw, 1).Value = Format(Cells(rw, 1))
Next rw
CSVファイルも、以下の通り、問題なく作成されました。
12桁以上の数字を使用するときは、注意が必要です。
改善マクロ
上のマクロは、1列目だけの対応ですが、以下は、すべてのセルを対象に、文字列を表す「’」を、配列データに追加して、その後、セルに書き込んでいます。このようにすると、どのセルも、文字列になるので、指数表示の問題は起きません。
'*******
'* 二次元配列をCSVファイルに出力、高速化。20230912
'* すべての列に対して指数表示対応。20241114
'* https://excel-vba.work/2021/01/03/%E3%80%90vba%E3%80%91%E6%8C%87%E5%AE%9A%E3%81%97%E3%81%9F%E7%AF%84%E5%9B%B2%E3%82%92csv%E3%83%95%E3%82%A1%E3%82%A4%E3%83%AB%E3%81%B8%E5%87%BA%E5%8A%9B%E3%81%99%E3%82%8B/
'* arr:二次元配列。これが、csvファイルに出力される
'* FileName:csvファイル名、フルパス
Private Sub array_to_csv_Excel_r1(ByRef arr As Variant, ByVal fileName As String)
Dim csvFile As String
Dim wb As Workbook
Dim fso As Object
'出力するCSVファイルを指定
csvFile = fileName
Set fso = CreateObject("Scripting.FileSystemObject")
'出力するCSVファイルが既に存在する場合は削除
If fso.FileExists(csvFile) Then
If IsBookOpened(csvFile) Then
Dim rc As Integer
rc = MsgBox("ファイルが開いています。" & vbCrLf & _
"はい:閉じて、削除して、処理を続けます。" & vbCrLf & _
"いいえ:処理を中断します", vbYesNo + vbQuestion, "確認")
If rc = vbYes Then
Dim FileNameOnly As String
FileNameOnly = fso.GetFileName(csvFile)
Windows(FileNameOnly).Close
Else
End
End If
End If
' ファイル削除
fso.DeleteFile csvFile
End If
'新規ブックを作成
Set wb = Workbooks.Add
' =======================================================以下が今回追加
' 新しい配列を作成して、全ての要素を文字列(頭に「'」追加)に変換
Dim i As Long, j As Long
Dim newArr As Variant
ReDim newArr(1 To UBound(arr), 1 To UBound(arr, 2))
For i = 1 To UBound(arr)
For j = 1 To UBound(arr, 2)
newArr(i, j) = "'" & CStr(arr(i, j))
Next j
Next i
' 新しい配列をブックにコピー
wb.Worksheets(1).Range("A1").Resize(UBound(arr), UBound(arr, 2)).Value = newArr
' =======================================================ここまでが今回追加
'新規ブックをCSVファイルとして出力
wb.SaveAs fileName:=csvFile, FileFormat:=xlCSV, Local:=True
' 「xlCSV」・・・・・文字コードは「Shift_JIS」となります。
' 「Local:=」に「True」を指定することで、「日付の形式」を「Excelの言語設定で指定されている言語(日本語と思います)」にします
'新規ブックを保存せずに閉じる
wb.Close SaveChanges:=False
'後片付け
Set fso = Nothing
End Sub
“<Array to CSV> 「配列」を「CSV」へ。Excelで保存。指数表示対策済み。” に対して1件のコメントがあります。