如何将包含逗号分隔值的多列拆分为不同行,同时保持不同列中的值顺序?

huangapple go评论107阅读模式
英文:

How to split multiple columns with comma separated values into different rows while maintaining value sequence in the different columns?

问题

使用VBA,如何拆分一个包含共同数据的Excel表格,以及包含逗号分隔值的其余列。表格格式类似于下图所示:

如何将包含逗号分隔值的多列拆分为不同行,同时保持不同列中的值顺序?

预期结果类似于下图所示:

如何将包含逗号分隔值的多列拆分为不同行,同时保持不同列中的值顺序?

尝试使用Power Query,但静态数据在对每个逗号分隔列数据执行操作时会重复。

英文:

Using VBA, how to split an excel table containing few columns with common data and remaining columns containing comma separated values. The table format is something like

如何将包含逗号分隔值的多列拆分为不同行,同时保持不同列中的值顺序?

And the expected result is something like

如何将包含逗号分隔值的多列拆分为不同行,同时保持不同列中的值顺序?

Tried using Power Query but static data is repeated with actions done on each comma separated column data.

答案1

得分: 2

以下是您要翻译的内容:

这是Power Query 中的一种方式

  1. "Added Custom" = Table.AddColumn(Source, "Custom", each Table.FromColumns({Text.Split([Column3],","),Text.Split([Column4],","),Text.Split([Column5],",")})),
  2. "Removed Columns" = Table.RemoveColumns("Added Custom",{Column3, Column4, Column5}),
  3. "Expanded Custom" = Table.ExpandTableColumn("Removed Columns", "Custom", {"Column1", "Column2", "Column3"}, {"Column3", "Column4", "Column5"})
  4. in "Expanded Custom"```
  5. [![enter image description here][1]][1]
  6. <details>
  7. <summary>英文:</summary>
  8. This is one way in powerquery
  9. let Source = Excel.CurrentWorkbook(){[Name=&quot;Table1&quot;]}[Content],
  10. #&quot;Added Custom&quot; = Table.AddColumn(Source, &quot;Custom&quot;, each Table.FromColumns({Text.Split([Column3],&quot;,&quot;),Text.Split([Column4],&quot;,&quot;),Text.Split([Column5],&quot;,&quot;)})),
  11. #&quot;Removed Columns&quot; = Table.RemoveColumns(#&quot;Added Custom&quot;,{&quot;Column3&quot;, &quot;Column4&quot;, &quot;Column5&quot;}),
  12. #&quot;Expanded Custom&quot; = Table.ExpandTableColumn(#&quot;Removed Columns&quot;, &quot;Custom&quot;, {&quot;Column1&quot;, &quot;Column2&quot;, &quot;Column3&quot;}, {&quot;Column3&quot;, &quot;Column4&quot;, &quot;Column5&quot;})
  13. in #&quot;Expanded Custom&quot;
  14. [![enter image description here][1]][1]
  15. [1]: https://i.stack.imgur.com/nFsVL.png
  16. </details>
  17. # 答案2
  18. **得分**: 1
  19. 以下是您要翻译的内容:
  20. 该算法以以下方式作为输入:
  21. 1)ListObject对象,
  22. 2)要放置新表格的单元格,
  23. 3)分隔符字符串以及
  24. 4)我们要将其内容拆分为多行的列的索引。
  25. 表格可以有任意多列,并且还有一件事:我们要分割的单元格不必具有相同数量的元素,例如column2 a,b,c column3 a,b,c,d
  26. 接下来是另一个函数,它执行完全相反的操作。它将新扩展的表格转换为原始表格的形式,还有一个选项,可以选择在由连接生成的列中不包含(内部)重复值。尽管它不回答这个问题,但我在这里发布它,因为它与答案相关,它是答案结果的“撤消”。
  27. 这是一段VBA代码,用于扩展和缩小表格。如果您需要任何其他信息,请随时提问。
  28. <details>
  29. <summary>英文:</summary>
  30. The algorithm takes as input 1) a ListObject object, 2) a cell where to place the new table 3) the separator string and 4) the index of the columns whose content we want to break into multiple lines. The table can have as many columns as need and one more thing: The cells we want to partition do not have to have the same number of elements, for example column2 a,b,c column3 a,b,c,d
  31. Then follows another Function which does the exact opposite. It takes the new expanded table and converts it to the form of the original one, also having an option not to have (internal) duplicate values in the columns made by concatenation. Although it doesn&#39;t answer the question, I&#39;m posting it here because it&#39;s interrelated with the answer, it&#39;s an &quot;undo&quot; of the result of the answer.
  32. Option Explicit
  33. &#39;this EVENT in a sheet module, call the sub with the parameters
  34. Private Sub BT_EXPAND_Click()
  35. Dim expTbl As ListObject
  36. Const separator = &quot;,&quot;
  37. Set expTbl = expand_table(Me.ListObjects(&quot;TABLEA&quot;), Me.Range(&quot;A8&quot;), separator, 3, 4, 5)
  38. If Not expTbl Is Nothing Then
  39. Call shrink_table(expTbl, Me.Range(&quot;A&quot; &amp; (expTbl.ListRows.Count + 11)), False, separator, 3, 4, 5)
  40. End If
  41. End Sub
  42. &#39;****************************************************************************
  43. &#39;* COPY THIS PUBLIC FUNCTION IN A MODULE
  44. &#39;* CAN USE IT FOR ANY TABLE WITH ANY ROWS/COLUMNS
  45. &#39;* ----------------PARAMETERS--------------------
  46. &#39;* ListObject&gt; the table object to expand its data
  47. &#39;* topLeftOfNewTable&gt; the cell to fit the top-left corner of the new table
  48. &#39;* separator&gt; separator string of concatenated values at columns we tranform
  49. &#39;* colsWithComma&gt; ParamArray with indexes of columns to transform eg 3,5,6,7
  50. &#39;****************************************************************************
  51. Public Function expand_table(ByRef t As ListObject, topLeftOfNewTable As Range, separator As String, _
  52. ParamArray colsWithComma() As Variant) As ListObject
  53. Dim rws As Long, clmns As Long, r As Long, c As Long, tex As ListObject, clarr() As Variant, strFlag As String
  54. Dim lbWc As Integer, ubWc As Integer, z As Integer, zidx As Integer, tmpUb As Integer, maxExp As Integer, lrw As ListRow
  55. Dim ccex As Integer
  56. lbWc = LBound(colsWithComma)
  57. ubWc = UBound(colsWithComma)
  58. If ubWc &lt; 0 Then Exit Function &#39;nothing todo
  59. On Error GoTo Lerr
  60. rws = t.DataBodyRange.Rows.CountLarge
  61. clmns = t.DataBodyRange.Columns.CountLarge
  62. &#39;CREATE A STRING WITH AS MANY SPACES AS THE COLUMNS OF TABLE
  63. &#39;AT POSITIONS OF colsWithComma() SET &quot;*&quot; (FLAG THE POSITION COLUMNS TO EXPAND)
  64. strFlag = Space$(clmns)
  65. For z = lbWc To ubWc
  66. If colsWithComma(z) &lt;= clmns Then
  67. Mid$(strFlag, colsWithComma(z), 1) = &quot;*&quot;
  68. Else
  69. MsgBox (&quot;expand_table&gt; Column &quot; &amp; colsWithComma(z) &amp; &quot; out of table bounds - max=&quot; &amp; clmns)
  70. End If
  71. Next
  72. Application.ScreenUpdating = False
  73. &#39;IF ALREADY EXISTS A TABLE IN THE POSTOTION OF THE NEW, DELETE IT
  74. If Not topLeftOfNewTable.ListObject Is Nothing Then
  75. topLeftOfNewTable.ListObject.Delete
  76. End If
  77. &#39;CREATE NEW TABLE TO EXPORT THE ORIGINAL
  78. Set tex = topLeftOfNewTable.Worksheet.ListObjects.Add(xlSrcRange, topLeftOfNewTable.Resize(1, t.DataBodyRange.Columns.Count), , xlYes)
  79. &#39;COPY THE HEADER LINE
  80. t.HeaderRowRange.Copy topLeftOfNewTable
  81. &#39;redim the array to hold the arrays of columns values, when split them
  82. ReDim clarr(lbWc To ubWc)
  83. For r = 1 To rws
  84. &#39; in every row find the column with the maximum elements after split
  85. maxExp = 0
  86. For z = lbWc To ubWc
  87. clarr(z) = Split(t.ListRows(r).Range.Cells(, colsWithComma(z)), separator)
  88. tmpUb = UBound(clarr(z))
  89. If tmpUb &gt; maxExp Then maxExp = tmpUb
  90. Next
  91. &#39;WE LOOP FROM 0 TO maximum found above
  92. For ccex = 0 To maxExp
  93. zidx = 0
  94. Set lrw = tex.ListRows.Add
  95. For c = 1 To clmns
  96. With lrw.Range.Cells(, c)
  97. &#39;IF THE COLUMN HAVE TO SPLIT
  98. If Mid$(strFlag, c, 1) &lt;&gt; &quot; &quot; Then
  99. tmpUb = UBound(clarr(zidx))
  100. &#39;IF THE ARRAY OF SPLITED VALUES EXIST AND INDEX (ccex) &lt;= UPPER BOUND
  101. If tmpUb &gt;= 0 And ccex &lt;= tmpUb Then
  102. &#39;COPY THE VALUE FROM ARRAY (splitted data)
  103. .Value2 = clarr(zidx)(ccex)
  104. End If
  105. &#39;NEXT SPLITING COLUMN
  106. zidx = zidx + 1
  107. Else
  108. &#39;COPY THE ORIGINAL VALUE AS IS
  109. .Value2 = t.ListRows(r).Range.Cells(, c)
  110. End If
  111. End With
  112. Next
  113. Next
  114. Next
  115. Set expand_table = tex
  116. Exit Function
  117. Lerr:
  118. If Err.Number &gt; 0 Then
  119. MsgBox (&quot;expand_table&gt;&quot; &amp; vbCrLf &amp; Err.Description &amp; vbCrLf &amp; &quot;error number&gt; &quot; &amp; Err.Number)
  120. End If
  121. On Error GoTo 0
  122. End Function
  123. &#39;*****************************************************************************
  124. &#39;* COPY THIS PUBLIC FUNCTION IN A MODULE
  125. &#39;* This function reverse the result of the previus, creates the original
  126. &#39;* table from the expanded one, having the option not to have duplicate values
  127. &#39;* in the columns made by concatenation
  128. &#39;* ----------------PARAMETERS--------------------
  129. &#39;* ListObject&gt; the table object to expand its data
  130. &#39;* topLeftOfNewTable&gt; the cell to fit the top-left corner of the new table
  131. &#39;* separator&gt; separator string to concatenate values at columns we tranform
  132. &#39;* colsToConcat&gt; ParamArray with indexes of columns to be tranformed eg 3,5,6,7
  133. &#39;*****************************************************************************
  134. Public Function shrink_table(ByRef t As ListObject, topLeftOfNewTable As Range, _
  135. allowDuplicatesInConcat As Boolean, separator As String, ParamArray colsToConcat() As Variant) As ListObject
  136. Dim rws As Long, clmns As Long, r As Long, c As Long, tex As ListObject, strFlag As String
  137. Dim lbWc As Integer, ubWc As Integer, lrw As ListRow, stmp As Variant
  138. lbWc = LBound(colsToConcat)
  139. ubWc = UBound(colsToConcat)
  140. If ubWc &lt; 0 Then Exit Function &#39;nothing todo
  141. On Error GoTo Lerr
  142. rws = t.DataBodyRange.Rows.CountLarge
  143. clmns = t.DataBodyRange.Columns.CountLarge
  144. &#39;CREATE A STRING WITH AS SPACES AS THE COLUMNS OF TABLE
  145. &#39;AT POSITIONS OF colsToConcat() SET &quot;*&quot; (FLAG THE POSITION COLUMNS TO CONCAT)
  146. strFlag = Space$(clmns)
  147. For c = lbWc To ubWc
  148. If colsToConcat(c) &lt;= clmns Then
  149. Mid$(strFlag, colsToConcat(c), 1) = &quot;*&quot;
  150. Else
  151. MsgBox (&quot;shrink_table&gt; Column &quot; &amp; colsToConcat(c) &amp; &quot; out of table bounds - max=&quot; &amp; clmns)
  152. Exit Function
  153. End If
  154. Next
  155. Application.ScreenUpdating = False
  156. &#39;IF ALREADY EXISTS A TABLE IN THE POSTOTION OF THE NEW, DELETE IT
  157. If Not topLeftOfNewTable.ListObject Is Nothing Then
  158. topLeftOfNewTable.ListObject.Delete
  159. End If
  160. &#39;COPY THE TABLE
  161. &#39;t.Range.Copy topLeftOfNewTable
  162. Set tex = topLeftOfNewTable.Worksheet.ListObjects.Add(xlSrcRange, topLeftOfNewTable.Resize(1, t.DataBodyRange.Columns.Count), , xlYes)
  163. &#39;tex.Name = &quot;TABLEA_CONCAT&quot;
  164. t.HeaderRowRange.Copy topLeftOfNewTable
  165. Set lrw = tex.ListRows.Add
  166. t.ListRows(1).Range.Copy lrw.Range
  167. For r = 2 To rws
  168. &#39;Set lrw = tex.ListRows.Add
  169. For c = 1 To clmns
  170. &#39;With lrw.Range.Cells(, c)
  171. If Mid$(strFlag, c, 1) = &quot; &quot; Then
  172. If t.ListRows(r).Range.Cells(, c) &lt;&gt; lrw.Range.Cells(, c) Then
  173. Set lrw = tex.ListRows.Add
  174. t.ListRows(r).Range.Copy lrw.Range
  175. GoTo LnextRow
  176. End If
  177. End If
  178. &#39;End With
  179. Next
  180. For c = 1 To clmns
  181. If Mid$(strFlag, c, 1) = &quot; &quot; Then
  182. &#39;copy value
  183. lrw.Range.Cells(, c) = t.ListRows(r).Range.Cells(1, c)
  184. Else
  185. &#39;add value
  186. With lrw.Range.Cells(, c)
  187. stmp = t.ListRows(r).Range.Cells(1, c)
  188. If stmp &lt;&gt; vbNullString Then
  189. If allowDuplicatesInConcat Then
  190. .value = .value &amp; separator &amp; stmp
  191. Else
  192. If InStr(1, separator &amp; .value &amp; separator, separator &amp; stmp &amp; separator) &lt;= 0 Then
  193. .value = .value &amp; separator &amp; stmp
  194. End If
  195. End If
  196. End If
  197. End With
  198. End If
  199. Next
  200. LnextRow:
  201. Next
  202. Set shrink_table = tex
  203. Exit Function
  204. Lerr:
  205. If Err.Number &gt; 0 Then
  206. MsgBox (&quot;shrink_table&gt;&quot; &amp; vbCrLf &amp; Err.Description &amp; vbCrLf &amp; &quot;error number&gt; &quot; &amp; Err.Number)
  207. End If
  208. On Error GoTo 0
  209. End Function
  210. [![enter image description here][1]][1]
  211. [1]: https://i.stack.imgur.com/WeA0s.png
  212. </details>

huangapple
  • 本文由 发表于 2023年5月22日 22:32:43
  • 转载请务必保留本文链接:https://go.coder-hub.com/76307261.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定