- ベストアンサー
矩形範囲の複数列を縦1列に並べ替えVBA(続)
- ExcelのVBAを使用して、矩形範囲内の複数の列を縦に一列に並べ替えるコードについて教えてください。
- 現在のコードでは、M1からRの範囲のデータをM1-M**/N1-N**/O1-O**のように縦に並べ替えています。
- 縦に並べ替える順序を任意にしたい場合、どの部分を修正すればよいのでしょうか?
- みんなの回答 (7)
- 専門家の回答
質問者が選んだベストアンサー
遅くなりました。 修正、最適化を行った最終のコードを最後に記載しております。 現状において、下記の不明な点がありますので、併せてご確認ください。 ■データパターンの出力順番について (1)元データエリアA,B → パターン3 (2)並べ替えエリアE1,J150 → パターン1 (3)タブ貼り付けエリアN1:T150 → パターン2 とのことですので 「パターン1、2、3」の順番で区切り記号「******」により出力するということでしょうか? 過去質問を読み返したところ >ただ、NTデータがBCの前になってしまっていたのでコメントをたよりに入れかえました。 と記述されており、ご提示されているコードからも「パターン1、3、2」の順になっているようですが。 現在は「パターン1、3、2」の順でコードを記述しておりますので、ご確認の上修正願います。 ■パターン3について いままでB,Cで処理を行ってきましたが、A,Bに変更いたしました。 変更される場合はコード内の「'(1)A,B列をタブ区切りでテキストデータへ出力」において 「"A"」、「"B"」の箇所を変更してください。 今までB,Cで行っていたときですが、1行目から「B列の最終行」までの範囲で、 「B列が空白以外」であれば「B列とC列をタブで結合して出力」しておりました。 今回、A,B列の最終行のうち、最終行が多い方をデータ範囲の最終行として選定し、 A,B列を結合したものが空白以外であれば、テキストデータとして出力しています。 ■データについて すでにご覧いただいたかと思いますが、ヒントとしてプロフィールを再度ご確認願います。 ■VBAコード Sub action() '型宣言 Dim st As String, ed As String Dim stcol As Long, edcol As Long Dim strow As Long, edrow As Long Dim retu As Long, gyou As Long Dim fname As String, tpath As String, dpath As String '◆ Dim fcnt As Long, r_max As Long Dim i As Long, j As Long '★ Dim srt As String, word As String '★ Dim retu_s As Variant, retu2_s As Variant '★ Dim myRng As Range '★ Dim ngs As Variant, ng As Variant '★ Dim objFileSys As Object, objTS As Object '★ 'ファイルの出力先 'dpath = "H:\■DATA\DATAB" dpath = ThisWorkbook.Path 'データの範囲、取得列の指定 st = "E1" ed = "O150" Set myRng = Range("E1:K150") '★ srt = "E,M,F,G,L,N,O" '★ 'セルアドレスより各行列番号を取得 stcol = Range(st).Column edcol = Range(ed).Column strow = Range(st).Row edrow = Range(ed).Row 'セル範囲が選択中の場合 If Selection.Count > 1 Then stcol = Selection(1).Column edcol = Selection(Selection.Count).Column strow = Selection(1).Row edrow = Selection(Selection.Count).Row End If '出力先のファイル名を処理 If Cells(strow, stcol).Text = "" Then fname = "不明(" & Cells(strow, stcol).Address & ")" Else fname = Cells(strow, stcol).Text End If ngs = Split("■,\,/,:,*,?,"",<,>,|", ",") For Each ng In ngs fname = Replace(fname, ng, "#") Next If Dir(dpath, vbDirectory) = "" Then MsgBox "パスが不正です" & vbCrLf & vbCrLf & dpath '◆ Exit Sub End If tpath = dpath & "\" & fname & ".txt" Do Until Dir(tpath) = "" fcnt = fcnt + 1 tpath = dpath & "\" & fname & "_" & fcnt & ".txt" Loop '///// テキスト書き出し処理 ///// 'テキストファイルを新規作成 Set objFileSys = CreateObject("Scripting.FileSystemObject") Set objTS = objFileSys.CreateTextFile(tpath) '(2)1列に指定した列順で出力(パターン1) '列方向にループ retu_s = Split(srt, ",") '★ For Each retu2_s In retu_s '◆ retu = Range(retu2_s & "1").Column '★ '行方向にループ For gyou = strow To edrow If Cells(gyou, retu).Text <> "" Then '◆ If Left(Cells(gyou, retu).Text, 1) <> "■" Then '★ objTS.WriteLine Cells(gyou, retu).Text End If '★ End If Next gyou Next '◆ objTS.WriteLine "****************************************************" '(1)A,B列をタブ区切りでテキストデータへ出力(パターン3) r_max = WorksheetFunction.Max( _ Range("A" & Rows.Count).End(xlUp).Row, _ Range("B" & Rows.Count).End(xlUp).Row) '★ For i = 1 To r_max '◆ If Range("A" & i).Text & Range("B" & i).Text <> "" Then '◆ objTS.WriteLine Range("A" & i).Text & vbTab & Range("B" & i).Text '◆ End If Next i objTS.WriteLine "***************************************************" '(3)矩形範囲をタブ区切りでテキストデータ出力(パターン2) Dim flag As Integer For j = 1 To myRng.Rows.Count word = "" flag = 0 For i = 1 To myRng.Columns.Count If myRng.Cells(j, i).Value <> "" Then flag = flag + 1 End If word = word & myRng.Cells(j, i).Value If i < myRng.Columns.Count Then word = word & vbTab Next i If flag > 0 And myRng.Cells(j, 1).Value <> "" Then objTS.WriteLine word Next j 'テキストファイルを閉じる objTS.Close MsgBox tpath & vbCrLf & "に出力しました" End Sub
その他の回答 (6)
- eden3616
- ベストアンサー率65% (267/405)
対象のデータが公開されないため、具体的なデータの状況をこちらで把握する必要があります。 そのため順を追って各列の状態を把握しておりますので 回りくどい質問のやり取りが続いておりますがご了承ください。 開示可能なデータであれば外部ストレージサービス等を用いてデータを拝見させていただければ作業が進むかと思います。 (情報の公開が不可能であれば、データの表現される可能性を全て網羅している簡略化されたモデルケースのデータでも構いません) >使用VBAは質問VBA+No2お礼の修正のものです。(No3は無関係です) 質問のVBAコードは古いものですよね? 前回回答分の3パターンの出力が複合されたVBAにNo2の修正を適応されたものと判断致します。 ※もし質問されたVBAを利用されたのであれば、 ご提示の『Set myRng = Range("E1:K150")』はコード内では使用しておりません。 >=IF(B15="","",IF(C15="","■"&$A$1,C15)) >(G列の例・表示は■*****または****) >=IF(B15="","",IF(C15="","■",C15)) >(I列の例・表示は""または****) >=IF(A37="",IF(ISERR(FIND("■",B37,1)),"","■"&B37),A37&"-"&B37&"("&C37&")") >(E列の例・表示は表示は■*****または****) 数式の記入されているセルですが、行数が記載されていませんので 頭二つの数式はそれぞれセルG15、I15で、 最後の数式はE37行目に記載されているということでよろしいでしょうか? >&$A$1がない列もある 無いというのはセルA1が空白""であるという事ですか。 それとも、数式として無いという事でしょうか。 何れにせよ、B列が空白以外でC列が空白ですと「■」とはなりますが。 >すなわちここではE、F、G列は並べ替えとタブ貼り付けデータを共用しているため、扱いを変える必要がある。 このE、F、G列とは 上記数式が入っている箇所ということであればF列でなくIということでしょうか。 それとも数式はI列でなく、F列に入っているのでしょうか。 >両方ともそうであるかとに理解していました。 >いままで「パターン1」も空白行は行ツメされていた No5で以下のように回答した箇所は、「任意の順番」と記述した通り、「空白を無視する処理」ではなく、「並び替えを行う処理(パターン(1)にのみ適応)」という意味になります。 ――――――――――――――――― 任意の順番は上記パターンの(1)であり、左が空白で出力するのはパターン(2)であると認識しています。 元々並び替えを行ったパターン(1)の出力にそのような処理をいれていませんが。 ――――――――――――――――― >(最左列が空白の場合)現状は書き出し後に空白が発生しています。 パターン(2)には並び替えを行わずそのままの塊で出力するものと認識しております。 現在の処理では指定範囲『Set myRng = Range("E1:K150")』での最左の列 つまりE列が空白であればその行は無視されます。 空白が発生する原因が何かをつかむ必要がありますね。 手持ちのコードでは正常に処理されており、確認が出来ません。 関連するパターン(2)の出力処理部を記述いたします。 (※パターン(2)では(1)のような列の並び替え処理はなく、データのまま出力されます) ―――――――――――――― For j = 1 To myRng.Rows.Count word = "" flag = 0 For i = 1 To myRng.Columns.Count If myRng.Cells(j, i).Value <> "" Then flag = flag + 1 End If word = word & myRng.Cells(j, i).Value If i < myRng.Columns.Count Then word = word & vbTab Next i If flag > 0 And myRng.Cells(j, 1).Value <> "" Then objTS.WriteLine word '★ Next j ―――――――――――――― 「 '★」の箇所で範囲(E1:K150)の最左列(E列)が空白であれば出力されません。 >これらのセルのうち■が含まれるセルはパターン1による並び替えに際しセルを省略したい。 パターン(2)での問題点は別として、パターン(1)の処理としての最終形としましては 指定したE1~O150列の範囲において、「文字列の先頭に「■」がある」または「対象が空白""である」場合は除外して1列でパターン("E,M,F,G,L,N,O")順で抜き出したいという事でよろしかったでしょうか。 If Cells(gyou, stcol).Text <> "" Then objTS.WriteLine Cells(gyou, retu).Text End If の箇所を以下のようにしてください。 If Cells(gyou, stcol).Text <> "" Then If Left(Cells(gyou, retu).Text, 1) <> "■" Then objTS.WriteLine Cells(gyou, retu).Text End If End If と、この修正をしていて気付いたのですが、 現状のコードで「If Cells(gyou, stcol).Text <> "" Then」の部分。 「st = "E3"、ed = "O150"」で指定した範囲の最左側の列(E列)が空白だと読み飛ばしていますね。 つまり、パターン(1)の場合でも一番左のE列が空白であれば「E,M,F,G,L,N,O」すべて読み飛ばされます。 今更ながら、このような処理では不都合がありませんでしょうか? 対象のセルが空白、または文字列の先頭が■であれば読み飛ばすという処理にするのであれば If Cells(gyou, retu).Text <> "" Then If Left(Cells(gyou, retu).Text, 1) <> "■" Then objTS.WriteLine Cells(gyou, retu).Text End If End If のようにしてください。
お礼
遅くなりました。 たしかにデータシートを使いながらの方が対応してもらいやすいと思っていますが、このサイトが一般公開されているため、個別データの所在等が提供できない状態で申し訳ないです。(個別に提供できる方法があるなら対応は可能なのですが) ここで再度整理させていただきます。 使用VBAは質問欄のものではなく質問欄にNo1の「当方お礼」にある修正を加えたものです。(タブ貼り付け等追加) 従来のもの (1)元データエリアA,B (2)並べ替えエリアE1,J150 (3)タブ貼り付けエリアN1:T150 にデータがあります。 (1)の内容は、1,2行目に■を含む文字列(タイトル等)、そのあと途中(例えば11行目、23行目等)にもサブタイトルのように■を含む文字列が入っています。 (1)はパターン3として、出力の末尾にそのままタブ状態で追記します。(データエリアの保存) (2)のエリアは列単位で1列に並び替えます。(パターン1) この場合、A列で■のあるサブタイトル部分は各列とも空白表示(="")をしています。 (当方提示の例示関数式は色々あるため無視してください) 並べ替えの際、エリアの最左列が空白セルの場合、その行は 削除して詰めるという処理をしていただいています。 (3)は別フォームに貼り付けるデータのためタブ形式にして矩形そのままテキストに追記貼り付けています。 (パターン2) このエリアは元データに■がある行もそのまま関数式により■入りで表示しており削除する必要がありません。 エリア(2)(3)の列の末尾は、A列にデータがなくなる行以下は150行までは(="")による空白セル表示関数入り、151行以下は関数なしの空白セルになりこれらはいずれも貼り付け並び替えの対象にしません。 以上は現状の概要です。 そこで今回の対処 エリアを整理するため、エリア(3)のうしろの列にエリア(2) を移動させました。 エリア(3)は矩形のまま列移動になるため、パターン2のタブ貼り付けは、VBAにエリアの書き換えだけでそのまま使えると思っています。(N-Tが例えばE-K) またパターン3(AB)もそのまま使えます。 問題は、パターン1の並び替えなのですが、エリア(2)と(3)から任意の列を選択します。そのため、サブタイトル部分の表示がエリア(2)のデータでは空白セルであり、エリア(3)のデータでは■入り表示になってます。 従来は並び替えエリアはサブタイトル部分を空白セルとして表示していたため単純に空白行削除で対応できたのですが 混在になってしまうためこれが不可能となりました。 そこで、サブタイトルの行はすべて■入りにしました。 そして並び替えのパータン1に限っては■入りセルは削除して行詰めにしてしまいたいのです。 なおこのため、1~2行のタイトル行も削除の対象になってしまうので、1~2行については■表示でなく●表示等別の記号に変更しようと思っています。 以上をふまえて再度ご回答いただければ幸いです。
補足
大変すみませんが、複数のBookを同じVBAで動かすためにデータを合体させている最中にあちこち関数式が正しく表示できなくなってしまい、急遽フォームを手直しするはめになってしまいました。そのため少々時間をくださるようよろしくお願いします。
- eden3616
- ベストアンサー率65% (267/405)
ご質問の内容につきまして確認をさせて頂きます。 >データを合併させ、並べ替えとタブ貼り付けを共用させるという >魂胆だったのですが、ここで問題が生じてしまいました。 どのコードに対して作業が行われているのか明確にお願いします。 ・No3の不要部分削除したタブ区切りのデータ出力のコードのことですか? ・No2で修正した大元のコードのことですか? ・上記ではなく、新たに改変されたデータでしょうか? 取り扱っているデータのパターンは以下の分類と認識しています。 (1)並び替え:列を並び替えての1行出力データ → st = "E1"~ed = "N150"で指定した範囲(行) → srt = "M,E,F,N,H,G,I,L"で指定した順番 (2)タブ貼付:短径で左列が空白の場合にその行は飛ばす → Set myRng = Range("E1:K150")で指定した範囲 (3)BC列:行数不定でタブ区切りで出力 → B、C列をタブ区切りで出力 ※上記範囲はNo2の回答で提示した範囲を元に判断致します。 >データの1~2行目と途中10行置きくらいに■を含む文字列を >仕切り線のような形で入れてタブ貼り付けエリアでは表示 タブ貼付エリアとは上記パターンの(2)に該当するものでしょうか。 >1~2行目はタイトル部分なのでいいとして、 >途中の■を含む文字列セルはタブ貼り付けの方では必要なのですが、 タイトル部分である2行目を飛ばして範囲選択されると良いのではないでしょうか。 E3:K150など。 >並べ替えの方は不要なので、並び替えエリヤでは関数で空白("")処理をしていたため、 >空白行の削除ということでつめていただいていました。 If Cells(gyou, stcol).Text <> "" Then objTS.WriteLine Cells(gyou, retu).Text End If 上記コードで空白行『""』以外『<>』の場合、セルの値『Cells(gyou, retu).Text』を出力『objTS.WriteLine』しています。 >ほか、任意の順番ということで、 >今まで最左セルが空白の場合その行は削除というルールが適用されなくなり >空白セルも詰めることなくそのまま出力されてしまいます。 任意の順番は上記パターンの(1)であり、左が空白で出力するのはパターン(2)であると認識しています。 元々並び替えを行ったパターン(1)の出力にそのような処理をいれていませんが。 (2)の処理内では以下の処理により該当の機能を持たせております。 If flag > 0 And myRng.Cells(j, 1).Value <> "" Then objTS.WriteLine word 上記コードで範囲E1:K150の対象の行で文字列が1つのセル以上見つかり、 かつ最左のセルが空白以外である場合は出力します。 >ところが共用にしたため並べ替えの方に一部■が入ってしまうことになり、 >不要なセルも表示されてしまう >そこで、並べ替えの方だけ、■を含む文字列(1~2行目を除く)のあるセルと >空白のセルは削除して詰めてしまうようにしたいのですが可能でしょうか。 空白以外にセルの値に一部「■」が含まれていれば除外するように条件を追加すれば 対象から外すことはできますが、それに伴う問題はありませんでしょうか? また、■が具体的にどのような形で表現されているのかが提示されていませんので 適格なコードが記述できません。 パターン(対象の特徴など)、規則性(全てに適応される共通条件)、対象(セルの場所) などのような具体的な例をもってお願いします。 たとえば、 E1:K150の範囲で、E列の値に「■■■■■■■■」と入っており、F~Kが空欄である 一部■とは「■○○○○」というように項目名が記述されているなど。 もし以下のような場合、データ(4)とデータ(5)の取扱いに困りますし、 存在しないのであれば丸々読み飛ばす処理を行います。 データ(1) データ(2) データ(3) ■■■■ データ(4) データ(5) データ(6) データ(7) データ(8)
お礼
使用VBAは質問VBA+No2お礼の修正のものです。(No3は無関係です) st = "E1" ed = "O150" Set myRng = Range("E1:K150") srt = "E,M,F,G,L,N,O" ※並べ替え(ハターン1) E,M,F,G,L,N,O 途中に =IF(B15="","",IF(C15="","■"&$A$1,C15)) のように入ります。(G列の例・表示は■*****または****) &$A$1がない列もある =IF(A40="","",A15)) のように入ります。(I列の例・表示は""または****) =IF(A37="",IF(ISERR(FIND("■",B37,1)),"","■"&B37),A37&"-"&B37&"("&C37&")") (E列の例・表示は表示は■*****または****) すなわちここではE、F、G列は並べ替えとタブ貼り付けデータを共用しているため、扱いを変える必要がある。 これらのセルのうち■が含まれるセルはパターン1による並び替えに際しセルを省略したい。 「■文字列」または空白セルの表示は横1行すべてに発生しします。 (ハターン3) ※B,Cの書き出しも残す 上記セルは入ったままにしておく(現状どおり) >タブ貼付エリアとは上記パターンの(2)に該当するものでしょうか。 (ハターン2) ※タブ入りの矩形貼り付け書き出しは次になります。 E1..K150 上記セルは入ったままにしておく(現状どおり) >タイトル部分である2行目を飛ばして範囲選択されると良いのではないでしょうか。 そうですね。あとで気がつきました。 >任意の順番は上記パターンの(1)であり、左が空白で出力するのはパターン(2)であると認識しています。元々並び替えを行ったパターン(1)の出力にそのような処理をいれていませんが。 両方ともそうであるかとに理解していました。 いままで「パターン1」も空白行は行ツメされていたので、その処理がされているかと思っていました。(最左列が空白の場合)現状は書き出し後に空白が発生しています。 >E1:K150の範囲で、E列の値に「■■■■■■■■」と入っており、F~Kが空欄である ここでは対象のE,M,F,G,L,N,Oを想定していますが、行単位で■が入る行はすべての列に入ります。 また空白セルの行は元データがない行。(横1行すべて空白) 基本的なパターンはB列(またはA列)に■が入っているデータの場合、その行はすべて■が入る(「またはA列」というのは別Bookの場合ですが今回の措置で統一しようとしています) A列が空白の場合はその行はすべて空白表示になります。(データの1行あけとか元データがない行)
補足
>タイトル部分である2行目を飛ばして範囲選択されると良いのではないでしょうか。 そうですね。あとで気がつきました。 といったもののタイトルも出力範囲に入れる必要があるのではずすわけにはいきませんでした。
- eden3616
- ベストアンサー率65% (267/405)
>「出力先のファイル名を処理」の記述がけっこう必要なんですね。 >ファイル名のとり方がわからないため、自分の修正では、 >前のファイル名がK1からとるようになっていたため、 >苦肉の作で無理やりK1にファイル名を表示する関数を入れていました。 (1)基本となるファイル名を構築する処理 (開始セルの内容を取得してファイルパスとして組み立てています) (2)ファイル名の処理ではWindowsの仕様上利用できない文字列を指定しても置換処理により削除 (3)重複ファイルが存在するか連番を加えていきながらチェックする処理 などがあるため長くなりました。 デバッグ用の表記や、エラー時のダイアログなど2、3行なくても動作する行もありますが。 コードは作成者の個性がでますので、複雑な処理ほど他人様のコードを読解するのは困難になります。 内容にもよりますが、対応出来るかどうかは拝見するまでわかりません。
お礼
たしかにおっしゃるとおり、人によって作り方が色々あるようなので、単純には対応できないかもしれません。ながめていただいて可能であればということでよろしくお願いします。 「VBAの整理」という質問でださせていただきます。 長文のVBAになりますので、質問欄にはいるかどうかわかりません。質疑の中での追加になるかもしれません。
補足
データを合併させ、並べ替えとタブ貼り付けを共用させるという魂胆だったのですが、ここで問題が生じてしまいました。 データの1~2行目と途中10行置きくらいに■を含む文字列を仕切り線のような形で入れてタブ貼り付けエリアでは表示させているのですが、1~2行目はタイトル部分なのでいいとして、途中の■を含む文字列セルはタブ貼り付けの方では必要なのですが、並べ替えの方は不要なので、並び替えエリヤでは関数で空白("")処理をしていたため、空白行の削除ということでつめていただいていました。 ところが共用にしたため並べ替えの方に一部■が入ってしまうことになり、不要なセルも表示されてしまうほか、任意の順番ということで、今まで最左セルが空白の場合その行は削除というルールが適用されなくなり空白セルも詰めることなくそのまま出力されてしまいます。 そこで、並べ替えの方だけ、■を含む文字列(1~2行目を除く)のあるセルと空白のセルは削除して詰めてしまうようにしたいのですが可能でしょうか。
- eden3616
- ベストアンサー率65% (267/405)
>色々修正していただいた後のVBAで影響が出るのを心配しましたが不要でした。 以前のデータも当初コードから継ぎ足しで作成しましたので、 根本から置き換えない限りは同様の修正が適応される可能性は高くなります。 (必ずしもというわけではありませんが) >不要部分を整理できればと思っています。 コードを以下の内容で最適化しました。 (1)以下を削除 ・'データの範囲(左上のセルと右下のセル)アドレスを指定 ・'セルアドレスより各行列番号を取得 ・'セル範囲が選択中の場合 (2)末尾に「'◆」がついている箇所を変更 (3)末尾に「'★」がついている箇所を追加 (4)「'型宣言」から使わない変数を排除 「'◆」の変更箇所にて出力時のファイル名の取得場所を変更しています。 以前は『st = "M1"』、『ed = "R200"』で設定した範囲の左上のセルをファイル名に使用していましたが 関連個所を削除しましたので、『Set myRng = Range("Z1:AF400")』で指定した 左上のセル内容をファイル名として取得するよう変更しました。 ※余談ですが VBAの場合、「Option Explicit」の記述が無ければ型宣言の「Set ~」以外はごっそり消しても動作するんですけどね・・・。 ■VBAコード Sub action2() '型宣言 Dim fname As String, tpath As String, dpath As String '◆ Dim fcnt As Long Dim objFileSys As Object, objTS As Object Dim word As String Dim flag As Integer Dim i As Long, j As Long '★ Dim myRng As Range '★ Dim ngs As Variant, ng As Variant '★ Set objFileSys = CreateObject("Scripting.FileSystemObject") 'データの範囲(左上のセルと右下のセル)アドレスを指定 Set myRng = Range("Z1:AF400") '出力先のファイル名を処理 If myRng.Cells(1, 1).Text = "" Then '◆ fname = "不明(" & myRng.Cells(1, 1).Address & ")" '◆ Else fname = myRng.Cells(1, 1).Text '◆ End If ngs = Split("■,\,/,:,*,?,"",<,>,|", ",") For Each ng In ngs fname = Replace(fname, ng, "#") Next dpath = ThisWorkbook.Path If Dir(dpath, vbDirectory) = "" Then Debug.Print "dpath = " & dpath MsgBox "パスが不正です" Exit Sub End If tpath = dpath & "\" & fname & ".txt" Do Until Dir(tpath) = "" fcnt = fcnt + 1 tpath = dpath & "\" & fname & "_" & fcnt & ".txt" Loop '///// テキスト書き出し処理 ///// 'テキストファイルを新規作成 Set objTS = objFileSys.CreateTextFile(tpath) '区切りをテキストデータへ出力 objTS.WriteLine "***************************************************" 'タブ区切りでテキストデータへ出力 For j = 1 To myRng.Rows.Count word = "" flag = 0 For i = 1 To myRng.Columns.Count If myRng.Cells(j, i).Value <> "" Then flag = flag + 1 End If word = word & myRng.Cells(j, i).Value If i < myRng.Columns.Count Then word = word & vbTab Next i If flag > 0 And myRng.Cells(j, 1).Value <> "" Then objTS.WriteLine word Next j 'テキストファイルを閉じる objTS.Close MsgBox tpath & vbCrLf & "に出力しました" End Sub
お礼
重ね重ねありがとうござしました。 「出力先のファイル名を処理」の記述がけっこう必要なんですね。 ファイル名のとり方がわからないため、自分の修正では、前のファイル名がK1からとるようになっていたため、苦肉の作で無理やりK1にファイル名を表示する関数を入れていました。 おかげさまでうまくできました。 ありがとうございました。 それから大変あつかましいご相談なのですが、別のVBA(WEBからの取込みと表作成)を人から教えてもらったあと、それをベースに追加事項をその都度別の人からきいたり自分なりに色々加除していったら、だんだんまわりくどくなったせいか、30~40秒かかるようになってしまいました。 そこでこれを見ていただいて効率いいVBAにできるのか知りたいのですが、別質問で出したいと思いますので、もしごめいわくでなかったら対応していただけますでしょうか。 なお、質問内容のVBA記述にURLが入りますが伏せ字になっていますので当方の自己紹介欄にヒントがありますのでご了解ください。 もし不都合でしたら無視していただいてけっこうです。その場合は数日後の質問消去します。(質問タイトルは次回にて)
- eden3616
- ベストアンサー率65% (267/405)
>今回の趣旨は1列に並べ替える部分(M1~I150)とそれをタブ用に貼り付けるフォーム(N1-T150)のデータがかなりかぶっているため >ひとつの表にしてしまいたいのですが、貼り付けフォームはそのままの順序にしておくとして、 >1列並べ替えはその中から、必要な列を抜き出せればいいと思っています。 >すなわちEからN列までデータ表を作り、最後のタブ貼り付け部分はE1~K150(従来のN1~T150に相当する部分)を対象とし、 >1列並び替えはM/E/F/N/H/G/I/Lの列を対象にこの順にしようというものです。 >もちろん現在でもデータのダブリさえ気にしなければ満足できる仕様となっています。 コードの修正箇所が分からないという事でしょうか? M1~I150が不明ですが・・・ 前回ご質問の最終のデータにおいても、回答No.1の修正内容と同じ修正でご利用できると思います。 「M/E/F/N/H/G/I/L」とのことですので、つまり今回は ・表全体はB,C列(行不定)及び、E1~N150のデータ ・1列に並べる範囲はE1~N150 (のうちM/E/F/N/H/G/I/L列のみ、順番も。) ・タブ区切り範囲はE1~K150 という設定のもと提示致します。 「下記のように修正→」以降のコードに回答No.1の修正箇所を記載します。 ・行の最後に「 '☆」を付けている箇所が削除行 ・行の最後に「 '★」を付けている箇所が追加行 ※ただし、 「 '☆」の行の先頭にはコメントアウトにする記号「'」を付けております。 ■前回質問で作成した控えのデータより '型宣言 Dim i As Long '★(既に有る場合は不要です) Dim srt As String '★ Dim retu_s As Variant '★ 'データの範囲(左上のセルと右下のセル)アドレスを指定 st = "E1" ed = "N150" Set myRng = Range("E1:K150") ■VBAコード「下記のように修正→」以降より '///// テキスト書き出し処理 ///// 'テキストファイルを新規作成 Set objTS = objFileSys.CreateTextFile(tpath) srt = "M,E,F,N,H,G,I,L" '★ '(1)E~J列を1列にテキストデータへ出力 '列方向にループ retu_s = Split(srt, ",") '★ For Each retu2_s In retu_s '★ retu = Range(retu2_s & "1").Column '★ 'For retu = stcol To edcol '☆ '行方向にループ For gyou = strow To edrow If Cells(gyou, stcol).Text <> "" Then objTS.WriteLine Cells(gyou, retu).Text End If Next gyou Next '★ 'Next retu '☆ ~~~~~~~~~~~ 以下省略(変更不要) ~~~~~~~~~~~
お礼
お手数をおかけしました。 前回補正で書きました通り、修正後の最終VBAでも今回のご指示の追加修正で問題なくできるようになりました。 (ご回答をテストする前に「今回の趣旨は…」を書いてしまいました。質問が変わってしまったのであわててご連絡したわけですが、心配は無用でした。どうもすみませんでした。 なお補正の部分で関連ご質問をさせていただきました。 よろしくお願いします。
- eden3616
- ベストアンサー率65% (267/405)
以前回答させて頂いたものです。 以下の変更箇所を修正してください。 ■変更箇所 (1)「'型宣言」内の最後(Dim fcnt As Longの後)に以下を追加 Dim i As Long Dim srt As String Dim retu_s As Variant (2)「'データの範囲~」の最後(ed = "R200"の後)に以下を追加 srt = "M,P,N" (3)「'列方向にループ」の最初(For retu = stcol To edcol)を削除 (4)(3)で削除した箇所(「'列方向にループ」と「'行方向にループ」の間)に以下を追加 retu_s = Split(srt, ",") For Each retu2_s In retu_s retu = Range(retu2_s & "1").Column (5)「'行方向にループ」の最後(Next retu)を以下に変更 (VBA処理的には「列」処理のループ最後) Next ■補足 (2)で指定した列順に取得して表示します。 M~R列を漏れなく半角カンマ「,」区切りで記述してください。 (上記コードでは提示して頂いたM、P、Nのみ取得されます) 手軽であるため、上記のような仕様にしましたが、 ExcelのソートのようにMPNの順で優先ソートして範囲内全てを書き出す場合は処理を別途考えます。
お礼
ご回答いただけて光栄です。 質問欄のVBAは当初のもので途中修正していただいており、最終的に下記部分訂正して使用しております。全文記述すると長くなりますので修正箇所のみ書きます。 '型宣言に追加 Dim objFileSys As Object, objTS As Object Dim word As String Set objFileSys = CreateObject("Scripting.FileSystemObject") 'データの範囲(左上のセルと右下のセル)アドレスを指定 st = "E1" ed = "J150" Set myRng = Range("N1:T150") 以下の部分以降を書き換え Open tpath For Output As #1 '列方向にループ For retu = stcol To edcol ~ MsgBox tpath & vbCrLf & "に出力しました" End Sub 下記のように修正→ '///// テキスト書き出し処理 ///// 'テキストファイルを新規作成 Set objTS = objFileSys.CreateTextFile(tpath) '(1)E~J列を1列にテキストデータへ出力 '列方向にループ For retu = stcol To edcol '行方向にループ For gyou = strow To edrow If Cells(gyou, stcol).Text <> "" Then objTS.WriteLine Cells(gyou, retu).Text End If Next gyou Next retu '区切りをテキストデータへ出力 objTS.WriteLine "****************************************************" '(3)B,C列をタブ区切りでテキストデータへ出力 For i = 1 To Range("B" & Rows.Count).End(xlUp).Row If Cells(i, 2).Value <> "" Then objTS.WriteLine Cells(i, 2).Value & vbTab & Cells(i, 3).Value End If Next i '区切りをテキストデータへ出力 objTS.WriteLine "***************************************************" '(2)N~T列をタブ区切りでテキストデータへ出力 Dim flag As Integer For j = 1 To myRng.Rows.Count word = "" flag = 0 For i = 1 To myRng.Columns.Count If myRng.Cells(j, i).Value <> "" Then flag = flag + 1 End If word = word & myRng.Cells(j, i).Value If i < myRng.Columns.Count Then word = word & vbTab Next i If flag > 0 And myRng.Cells(j, 1).Value <> "" Then objTS.WriteLine word Next j 'テキストファイルを閉じる objTS.Close MsgBox tpath & vbCrLf & "に出力しました" End Sub 以上です。うっかり古いままのVBAで質問してしまいました。 今回の趣旨は1列に並べ替える部分(M1~I150)とそれをタブ用に貼り付けるフォーム(N1-T150)のデータがかなりかぶっているためひとつの表にしてしまいたいのですが、貼り付けフォームはそのままの順序にしておくとして、1列並べ替えはその中から、必要な列を抜き出せればいいと思っています。 もちろん現在でもデータのダブリさえ気にしなければ満足できる仕様となっています。 すなわちEからN列までデータ表を作り、最後のタブ貼り付け部分はE1~K150(従来のN1~T150に相当する部分)を対象とし、1列並び替えはM/E/F/N/H/G/I/Lの列を対象にこの順にしようというものです。 こんな感じでご理解いただけますでしょうか。
補足
修正後のVBAでも、上記ご指示の変更で、必要列の抽出が問題なくできました。 色々修正していただいた後のVBAで影響が出るのを心配しましたが不要でした。 ありがとうございました。 なお、関連してですが、後半のタブ書き出しを単独でVBA処理しているのですが、(N~T列のみの抽出で列並び替え等不要)よくわからないためいただいたVBAから、 '///// テキスト書き出し処理 /////と '区切り(B列)をテキストデータへ出力 をはずしただけでそのまま運用して出力しています。 不要そうなのを削除したら動かなくなってしまっため、それ以外の記述はそのまま手をつけていませんので 関連のない記述もそのまま残っています。 不要部分を整理できればと思っています。 ※単にZ1~AF150を矩形でタプ抽出、テキスト生成のみです。 Sub action2() '型宣言 Dim st As String, ed As String Dim stcol As Long, edcol As Long Dim strow As Long, edrow As Long Dim retu As Long, gyou As Long Dim fname As String, tpath As String Dim fcnt As Long Dim objFileSys As Object, objTS As Object Dim word As String Set objFileSys = CreateObject("Scripting.FileSystemObject") 'データの範囲(左上のセルと右下のセル)アドレスを指定 st = "K1" ed = "P400" Set myRng = Range("Z1:AF400") 'セルアドレスより各行列番号を取得 stcol = Range(st).Column edcol = Range(ed).Column strow = Range(st).Row edrow = Range(ed).Row 'セル範囲が選択中の場合 If Selection.Count > 1 Then stcol = Selection(1).Column edcol = Selection(Selection.Count).Column strow = Selection(1).Row edrow = Selection(Selection.Count).Row End If '出力先のファイル名を処理 If Cells(strow, stcol).Text = "" Then fname = "不明(" & Cells(strow, stcol).Address & ")" Else fname = Cells(strow, stcol).Text End If ngs = Split("■,\,/,:,*,?,"",<,>,|", ",") For Each ng In ngs fname = Replace(fname, ng, "#") Next dpath = ThisWorkbook.Path If Dir(dpath, vbDirectory) = "" Then Debug.Print "dpath = " & dpath MsgBox "パスが不正です" Exit Sub End If tpath = dpath & "\" & fname & ".txt" Do Until Dir(tpath) = "" fcnt = fcnt + 1 tpath = dpath & "\" & fname & "_" & fcnt & ".txt" Loop '///// テキスト書き出し処理 ///// 'テキストファイルを新規作成 Set objTS = objFileSys.CreateTextFile(tpath) '区切りをテキストデータへ出力 objTS.WriteLine "***************************************************" 'Set myRng = Range("N1:T150") '(2)N~T列をタブ区切りでテキストデータへ出力 Dim flag As Integer For j = 1 To myRng.Rows.Count word = "" flag = 0 For i = 1 To myRng.Columns.Count If myRng.Cells(j, i).Value <> "" Then flag = flag + 1 End If word = word & myRng.Cells(j, i).Value If i < myRng.Columns.Count Then word = word & vbTab Next i If flag > 0 And myRng.Cells(j, 1).Value <> "" Then objTS.WriteLine word Next j 'テキストファイルを閉じる objTS.Close MsgBox tpath & vbCrLf & "に出力しました" End Sub
お礼
いつもながら貴重な時間をさいていただいてありがとうございます。 ◆データパターンの出力順番について パターンについては私も混乱しちゃっているのですが、 最初に並び替え、次に元データエリアの貼り付け、最後に矩形エリアの貼り付けということですので、ご指示のとおりも「パターン1、3、2」の順ですね。 ■パターン3について 実は出力フォームは結局同じなのですが、元データのフォームが色々あって、それにあわせて列の関数をその都度はめこんでおりそのフォームごとに集計方法を色々教えていただいていました。 今回一部統一しようとしている最中なのですが、 そのため当初の質問では、データ列がBとC(AはNoが入れてある)のフォームでの方法ですすめていたと思います。 現在もそのフォームも残っているためBCの場合もあり、修正しながら動作させたいと思っています。 ■データについて ちょっと意味が…? ■VBAコード これで問題なく出力できました。ありがとうございました。 なお、End Subの前に Application.Dialogs(xlDialogOpen).Show "H:\SDTXT\*.xlsm" ThisWorkbook.Close False を入れています。 これをこのBookの所在Passにしたい場合 ThisWorkbook.Pathにすればいいでしょうか。 なおあわせてお聞きしておきたいのですが、前述のとおり元データによる集計方法がいくつかパターンがあるため、それを今回統一フォームにしているのですが、集計方法にあわせて actionや action3、あるいはそれ以外のVBAをボタンをいくつか作ってフォーム毎に選択しています。 集計パターンにルールがあるのでフォームが統一できればたぶんactionの中でもその書き方をすれば可能のような気がしますが、今回はとりあえず既存のものを使い分けたいと思っています。 これをcall等を使ってひとつのボタンにすることは可能でしょうか。 具体的には A1の文字列の末尾5文字で判定して 「SD411,SD418,SD425」のときはVBA-1を呼び出す 「SD409」のときはVBA-2を呼び出す 「SD426」のときはVBA-3を呼び出す その他のときはVBA-4を呼び出す といった具合です。
補足
■データについて わかりました。