如何在excel vba中获取数组中元素的子集?

1rhkuytd  于 2023-03-24  发布在  其他
关注(0)|答案(3)|浏览(400)

如何在excel vba中获取数组中的第3到第6个元素?类似于以下内容。

  1. Dim x(9) as variant, y(3) as variant
  2. y(0:3) = x(2:5)
sdnqo3pr

sdnqo3pr1#

在VBA中,与Python不同,我们不能直接“子集”数组。
我们只能通过指定ij在某些边界之间来循环索引,例如i在第1行和第2行之间,j在第2列和第3列之间。当然,我们也可以直接通过位置索引数组,例如arr(1)i只是一个表示行索引的变量,j列索引。
或者,我们可以使用Index来“切片”出特定的行或列;我猜你可能会称之为子集,但从你的语法来看,我认为你是在以Python的方式思考。
Application.WorksheetFunction.Index(array,n, 0)将从阵列中分割行n
Application.WorksheetFunction.Index(array, 0, n)将列n从数组中切片出来

  1. Option Explicit
  2. Public Sub ArrayStuff()
  3. Dim arr(0 To 5) As Long, i As Long
  4. For i = 0 To 5
  5. arr(i) = i
  6. Next
  7. 'Loop only required indices
  8. For i = 2 To 3
  9. Debug.Print arr(i)
  10. Next
  11. 'Slice via Application.WorksheetFunction.Index
  12. Dim arr2(0 To 2, 0 To 2) As Long, j As Long, counter As Long
  13. For i = LBound(arr2, 1) To UBound(arr2, 1) '<== here can specify particular rows
  14. For j = LBound(arr2, 2) To UBound(arr2, 2) '<== here can specify particular columns
  15. counter = counter + 1
  16. arr2(i, j) = counter
  17. Next
  18. Next
  19. MsgBox Join(Application.WorksheetFunction.Transpose(Application.WorksheetFunction.Index(arr2, 0, 1)), ",") 'slice a column out
  20. MsgBox Join(Application.WorksheetFunction.Index(arr2, 1, 0), ",") 'slice a row out
  21. MsgBox Join(Application.WorksheetFunction.Index(arr2, 2, 0), ",") 'slice a row out
  22. End Sub
展开查看全部
k0pti3hp

k0pti3hp2#

对于任何不想使用循环或范围方法来处理这个问题的人,这里有一个经过测试的解决方案来处理数组的子集(不需要是整行或整列):

  1. Application.Sum(Application.Index(Array(1, 2, 3, 4, 5), 0, Evaluate("ROW(2:5)")))

它将数组的第2个到第5个元素(1,2,3,4,5)求和。对于二维数组,只需执行以下操作:

  1. Application.Average(.Index(DataArray, Evaluate("ROW(17:21)"), 5))

将计算二维数组“DataArray”的五个元素子集(行17到21,列5)的平均值。
顺便说一句,我还没有尝试评估这种方法的效率,但它确实是一种变通方法。希望这能有所帮助。

cxfofazt

cxfofazt3#

Python切片克隆

我最近想要的功能与您在示例中给出的功能相同,它基本上模仿了Python的slice方法-所以我决定编写代码来实现这一点。
请随意阅读我的代码注解,以了解它是如何工作的,但它应该与python的[start:stop:step]相同。
需要注意的是,我写这段代码时确实期望将base设置为0,所以如果你的base不同,代码可能需要首先增强。

  1. ' The Slice function takes an input array and returns a new array
  2. ' that contains a subset of the elements from the input array.
  3. '
  4. ' The subset is defined by the `startIndex`, `stopIndex`, and `stepCount`
  5. ' parameters.
  6. '
  7. ' The `startIndex` parameter specifies the index of the first
  8. ' element to include in the result. If `startIndex` is negative,
  9. ' it counts from the end of the input array.
  10. '
  11. ' The `stopIndex` parameter specifies the index of the first element
  12. ' to exclude from the result. If `stopIndex` is negative, it counts
  13. ' from the end of the input array.
  14. '
  15. ' The `stepCount` parameter specifies how many elements to skip
  16. ' between each element in the result. For example, if `stepCount`
  17. ' is 2, every other element will be included in the result. If
  18. ' `stepCount` is negative then it works in reverse order.
  19. '
  20. ' @author Robert Todar <robert@roberttodar.com>
  21. Public Function Slice( _
  22. ByVal arr As Variant, _
  23. Optional ByVal startIndex As Long = 0, _
  24. Optional ByVal stopIndex As Long = 0, _
  25. Optional ByVal stepCount As Long = 1 _
  26. ) As Variant
  27. Dim result As Variant
  28. Dim currentIndex As Long
  29. Dim resultIndex As Long
  30. If stepCount = 0 Then
  31. Err.Raise 5, "Slice", "Step count must be greater than or less than zero."
  32. Exit Function
  33. End If
  34. ' Handle negative indices for startIndex and stopIndex by
  35. ' converting them to positive indices.
  36. If startIndex < 0 Then startIndex = UBound(arr) + 1 + startIndex
  37. If stopIndex < 0 Then stopIndex = UBound(arr) + 1 + stopIndex
  38. If stepCount > 0 Then
  39. ' Calculate the stop index if it is not provided (i.e., if it is zero).
  40. If stopIndex = 0 Then stopIndex = UBound(arr) + 1
  41. ' Calculate the size of the result array based on
  42. ' start/stop indices and step count.
  43. resultIndex = (stopIndex - startIndex) \ stepCount - 1
  44. ' Adjust the stopIndex as it is excluded from the results.
  45. ' This is done after getting the resultIndex as the array
  46. ' is base 0 and will still need to account for that 0 index.
  47. stopIndex = stopIndex - 1
  48. Else
  49. ' If no input is provided then the start and
  50. ' stops need to get flipped from the start to
  51. ' the end.
  52. If startIndex = 0 Then startIndex = UBound(arr)
  53. If stopIndex = 0 Then stopIndex = -1
  54. ' Need to add 1 to the stopindex as it needs to
  55. ' be excluded from the results. In this senerio,
  56. ' this is done before the resultIndex is calculated as
  57. ' we don't have to account for the 0 index.
  58. stopIndex = stopIndex + 1
  59. resultIndex = (startIndex - stopIndex) \ Abs(stepCount)
  60. End If
  61. ' Create a new array with appropriate size to store results.
  62. ' and get the starting result index. This will be incremented
  63. ' each iteration.
  64. ReDim result(0 To resultIndex)
  65. resultIndex = LBound(result)
  66. ' Fill in values for resulting array using start/stop indices
  67. ' and step count as specified by user inputs.
  68. For currentIndex = startIndex To stopIndex Step stepCount
  69. result(resultIndex) = arr(currentIndex)
  70. resultIndex = resultIndex + 1
  71. Next currentIndex
  72. ' Return resulting sliced array back to user.
  73. Slice = result
  74. End Function

演示

下面是一个演示,展示了Python方法与我的版本的比较。

  1. Sub DemoSliceFunction()
  2. Dim a As Variant
  3. a = Array(0, 1, 2, 3, 4, 5)
  4. Debug.Print "Starting Data", Join(a)
  5. Debug.Print "Python Example: [::]", Join(Slice(a))
  6. Debug.Print "Python Example: [1:]", Join(Slice(a, 1))
  7. Debug.Print "Python Example: [-1:]", Join(Slice(a, -1))
  8. Debug.Print "Python Example: [-2:]", Join(Slice(a, -2))
  9. Debug.Print "Python Example: [:4]", Join(Slice(a, , 4))
  10. Debug.Print "Python Example: [1:4]", Join(Slice(a, 1, 4))
  11. Debug.Print "Python Example: [:-3]", Join(Slice(a, , -3))
  12. Debug.Print "Python Example: [1:-2]", Join(Slice(a, 1, -2))
  13. Debug.Print "Python Example: [-3:-2]", Join(Slice(a, -3, -2))
  14. Debug.Print "Python Example: [::2]", Join(Slice(a, , , 2))
  15. Debug.Print "Python Example: [::-1]", Join(Slice(a, , , -1))
  16. Debug.Print "Python Example: [-3::-1]", Join(Slice(a, -3, , -1))
  17. Debug.Print "Python Example: [-4::-1]", Join(Slice(a, -4, , -1))
  18. Debug.Print "Python Example: [:-3:-1]", Join(Slice(a, , -3, -1))
  19. Debug.Print "Python Example: [:-4:-1]", Join(Slice(a, , -4, -1))
  20. Debug.Print "Python Example: [-2:-5:-1]", Join(Slice(a, -2, -5, -1))
  21. End Sub

测试

此外,我还提供了这个方法的测试脚本。

  1. Sub TestSliceFunction()
  2. Dim a As Variant
  3. a = Array(0, 1, 2, 3, 4, 5)
  4. Debug.Assert Join(a) = "0 1 2 3 4 5"
  5. Debug.Assert Join(Slice(a)) = "0 1 2 3 4 5"
  6. Debug.Assert Join(Slice(a, 1)) = "1 2 3 4 5"
  7. Debug.Assert Join(Slice(a, -1)) = "5"
  8. Debug.Assert Join(Slice(a, -2)) = "4 5"
  9. Debug.Assert Join(Slice(a, , 4)) = "0 1 2 3"
  10. Debug.Assert Join(Slice(a, 1, 4)) = "1 2 3"
  11. Debug.Assert Join(Slice(a, , -3)) = "0 1 2"
  12. Debug.Assert Join(Slice(a, 1, -2)) = "1 2 3"
  13. Debug.Assert Join(Slice(a, -3, -2)) = "3"
  14. Debug.Assert Join(Slice(a, , , 2)) = "0 2 4"
  15. Debug.Assert Join(Slice(a, , , -1)) = "5 4 3 2 1 0"
  16. Debug.Assert Join(Slice(a, -3, , -1)) = "3 2 1 0"
  17. Debug.Assert Join(Slice(a, -4, , -1)) = "2 1 0"
  18. Debug.Assert Join(Slice(a, , -3, -1)) = "5 4"
  19. Debug.Assert Join(Slice(a, , -4, -1)) = "5 4 3"
  20. Debug.Assert Join(Slice(a, -2, -5, -1)) = "4 3 2"
  21. Debug.Print "TestSliceFunction: PASS"
  22. End Sub
展开查看全部

相关问题