【问题标题】:Insert vertical slices into array将垂直切片插入数组
【发布时间】:2020-07-03 21:04:47
【问题描述】:

在最近的 post 中,我演示了如何在 Application.Index() 函数中使用 数组参数(而不是单个索引) 以任意方向重新排列当前列的顺序(切换列、省略/删除列)。

    Application.Index(data, {vertical rows array}, {horizontal columns array})

这种方法不需要循环,并且允许仅通过在 OP 中列出新列位置来定义任何新列顺序,例如通过

     Array(1, 4, 2)

换句话说

  • 第一列,
  • (第三个省略=删除),
  • 剩余的第 4 列和第 2 列按顺序交换*:
Sub DeleteAndSwitch()
'[1]get data
    Dim data: data = Sheet1.Range("A1:D4")
'[2]reorder columns via Array(1, 4, 2), i.e. get 1st column, 4th and 2nd column omitting the 3rd one
'   (evaluation gets all existing rows as vertical 2-dim array)
    data = Application.Index(data, Evaluate("row(1:" & UBound(data) & ")"), Array(1, 4, 2))
'[3]write to any target
    Sheet2.Range("A1").Resize(UBound(data), UBound(data, 2)) = data
End Sub

询问相关评论:

“我可以切片一个二维数组,我可以消除列,重新排序列,但我不能►在这样的数组中插入列切片... 其实我可以用迭代来做,但是你有没有找到类似的方法来插入这样一个垂直切片?”

有条不紊的提示

至少众所周知,给定的列(例如第 4 列)可以通过

从数组(例如 data)中切分出来
    Column4Data = Application.Index(data, 0, 4)

产生一个基于 1 的二维“垂直”数组。

不可能,但是将垂直切片分配给另一个切片;以下代码会引发 1004 错误(应用程序定义的或对象定义的错误):

Application.Index(data, 0, 4) = Application.Index(data, 0, 1)

问题

是否有任何可能在数组中插入列切片(无需迭代)?

确实存在将此类列数据排列在一个临时数组数组中的可能性(“锯齿状数组”),并以此为基础构建一个二维数组。

为了不过度收费这篇文章,我将展示这种相当未知的方法作为►单独的答案,期待任何其他或更好的方法。

相关链接 Some pecularities of the Application.Index() function

【问题讨论】:

  • 为什么不直接将数据粘贴到工作表中,根据需要在工作表上移动,然后将范围复制回数组?
  • 感谢您在评论中的提示:这篇文章主要是为了告知Application.Index() 的进一步可能性,作为对先前答案的补充。 - 如果目标是只插入一列而没有进一步的列顺序重新排序,那么毫无疑问,我更喜欢使用 Excel VBA 时的方法(另请参见添加部分的前言)。 @johnywhy

标签: arrays excel vba slice


【解决方案1】:

使用Application.Index()的锯齿状数组方法

为了完整起见,我展示这种方法是为了证明Application.Index() 函数的进一步且广泛未知的可能性。

通过首先将(转置)切片添加到临时“数组数组”,可以在第二步中通过 双零 参数创建一个二维数组使用以下语法(参见 [2]b 部分):

    data = Application.Transpose(Application.Index(data, 0, 0))
Sub InsertSlices()
'Auth: https://stackoverflow.com/users/6460297/t-m
'[0]define extra array (or slice AND transpose from other data source)
    Dim Extra: Extra = Array(100, 200, 300, 400)   ' example data
'[1]get data
    Dim data: data = Tabelle7.Range("A1:D4")
'[2]a) rewrite data as 1-dim array of sliced column arrays
    data = Array(Extra, Slice(data, 1), Slice(data, 4), Slice(data, 2))
'~~~~~~~~~~~~~~~~~~~~~~~~~~~~
'[2]b) rebuild as 2-dim array
'~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    data = Application.Transpose(Application.Index(data, 0, 0))
'[3]write to any target
    Tabelle7.Range("F1").Resize(UBound(data), UBound(data, 2)) = data
End Sub
Function Slice(DataArr, ByVal colNo As Long) As Variant()
'Purpose: return entire column data as 2-dim array and
'         transpose them here to get a "flat" 1-dim array of column data
With Application
    Slice = .Transpose(.Index(DataArr, 0, colNo))
End With
End Function

警告: 对于较大的数据集,分两步重复转换数据可能会很耗时。


解决方法

因此,我更喜欢通过 Application.Index() 函数中的 ►array arguments 引用帖子中的基本方法,但是通过将(例如临时)列插入到物理数据范围 首先并最终通过在任何新位置(例如,在顶部)重新排列包含新添加的额外数据(最后位置)的列。

Sub DelSwitchAndInsert()
'Auth: https://stackoverflow.com/users/6460297/t-m
'[0]add other array data as last column to existing range
    Sheet1.Range("E1:E4") = Application.Transpose(Array(1, 2, 3, 4))
'[1]get data
    Dim data: data = Tabelle7.Range("A1:E4")
'[2]reorder via Array(1, 4, 2), i.e. get 1st column, 4th and 2nd column omitting the 3rd one
    data = Application.Index(data, Evaluate("row(1:" & UBound(data) & ")"), Array(UBound(data, 2), 1, 4, 2))
'[3]write to any target
    Sheet2.Range("A1").Resize(UBound(data), UBound(data, 2)) = data
End Sub

针对最近的 cmets 的解决方法附录 //Edit/2020-07-07

在任何给定的“列”编号处插入垂直额外单列数据的解决方法逻辑的灵活示例如下;我不会假装这既不是最好的方法也不是最好的编码方式:

    InsCol data, extra, 3        ' insertion e.g. as new 3rd "column"
Sub InsertExtraData()
'Purpose:  insert a single-column array (2-dim, 1-based)
    '[0]define extra array (or slice AND transpose from other data source)
        Dim extra: extra = Application.Transpose(Array(100, 200, 300, 400))   ' example data
    '[1]get data (or existing 2-dim array)
        Dim data: data = Sheet1.Range("A1:D4")
    '~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    '[2]insert extra as >>3rd<< column
    '~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
        InsCol data, extra, 3
    '[3]write to any target
        Sheet2.Range("A1").Resize(UBound(data), UBound(data, 2)) = data
End Sub
Sub InsCol(data, extra, Optional ByVal colNo As Long = 1)
With Sheets.Add
    '[0]add data to temporary range
    .Range("B1").Resize(UBound(data), UBound(data, 2)) = data
    .Range("B1").Offset(0, UBound(data, 2)).Resize(UBound(extra) - LBound(extra) + 1, 1) = extra
    '[1]get data
        data = .Range("B1").Resize(UBound(data), UBound(data, 2) + 1)
    '[2]reorder via Array(5, 1, 2, 3, 4)
        data = Application.Index(data, Evaluate("row(1:" & UBound(data) & ")"), getColNums(data, colNo))
    '[3]delete temporary sheet
        Application.DisplayAlerts = False: .Delete
        Application.DisplayAlerts = True
End With
End Sub
Function getColNums(main, Optional ByVal colNo As Long = 1) As Variant()
    'c.f. : https://stackoverflow.com/questions/53727578/joining-two-arrays-in-vba/60082345#60082345
    'Purp.: return ordinal element counters of combined 0-based 1-dim arrays
    Dim i&, n&: n = UBound(main) + 1    ' +1 as one column, +1 from 0-based~>one-based
    ReDim tmp(0 To n - 1)               ' redim to augmented size (zero-based)
    If colNo > n Then colNo = n
    If colNo < 1 Then colNo = 1
    For i = 0 To colNo - 1: tmp(i) = i + 1: Next i
    tmp(colNo - 1) = n
    For i = colNo To UBound(tmp): tmp(i) = i: Next i
    getColNums = tmp        ' return combined elem counters,  e.g. Array(1,2, >>5<< ,3,4)
End Function

【讨论】:

  • 非常好!投票赞成。 Insert slices 就像一个魅力,解决了我的“困境”。我正在尝试以 only 插入另一列(前一个 Sub 的 Extra 列)的方式转换 DelSwitchAndInsert,而不进行任何重新排序,但我不明白......
  • 这有帮助吗? - 解决方法的基本思想是将[0]添加/追加额外数据作为新列添加到现有范围,或者用任何现有数组加上最右边的数组填充临时或隐藏目标范围额外的数据,[1] 将新的“最后”列包含到扩展数组赋值中,[2] 将最右边的列号 Array(UBound(data, 2) 插入到当前列数组参数 Array(1,2,3,4) 中的任何所需位置@ ~> 例如Array(UBound(data,2), 1, 2, 3, 4) 最终重新排列为新的第一列。
  • 当然有帮助!但我(已声明)的兴趣与在现有列之间插入 新建列 有关。正如Insert slices 所做的那样。如果我没有遗漏任何内容,第二个只会切换或插入现有列。我无法调整它以接受基于先前Sub 机制的新创建(列)数组。我的意思是,插入Extra 数组...但这并不意味着呈现的所有内容都不是非常有趣... :) 我喜欢使用数组,直到现在我无法以紧凑的方式插入一列。跨度>
  • 如果我也能成功改编第二个从外部插入一列,我也会非常满意! :)
  • 感谢您的关注。 - 我不太确定,但是要正确理解你:结果是一样的,它在任何想要的列位置插​​入额外的数据,但是通过临时范围重新分配组合数据 A:D 加上一个附加的临时 E右侧的列(仅在之后通过 `Application.Index 移动)。由于这个额外的步骤,我认为这种工作(顺便说一句更快)方法是原始问题的解决方法。 我也想在这里找到一种缩写方式 :-) @FaneDuru
猜你喜欢
  • 1970-01-01
  • 2018-12-22
  • 1970-01-01
  • 2017-05-19
  • 1970-01-01
  • 2018-05-30
  • 2012-11-10
  • 1970-01-01
  • 2021-09-04
相关资源
最近更新 更多