在 Julia 中定义任意维度的多维数组

Posted

技术标签:

【中文标题】在 Julia 中定义任意维度的多维数组【英文标题】:Defining a multiple-dimensional array of arbitrary dimension in Julia 【发布时间】:2022-01-16 20:51:13 【问题描述】:

上下文

这个问题与this one有关。

在 Julia 中,我想创建一个 5 x 5 的二维数组,其中 (i, j) 元素具有 [i,j],如下所示:

5×5 MatrixVectorInt64:
 [1, 1]  [1, 2]  [1, 3]  [1, 4]  [1, 5]
 [2, 1]  [2, 2]  [2, 3]  [2, 4]  [2, 5]
 [3, 1]  [3, 2]  [3, 3]  [3, 4]  [3, 5]
 [4, 1]  [4, 2]  [4, 3]  [4, 4]  [4, 5]
 [5, 1]  [5, 2]  [5, 3]  [5, 4]  [5, 5]

我尝试使用array comprehension:

N = 5
L_2 = [[x1,x2] for x1 = 1:N, x2 = 1:N]

我想做什么

我想将这个定义推广到任意维度D

L_1 = [[x1] for x1 = 1:N] # 1-dimensional
L_2 = [[x1,x2] for x1 = 1:N, x2 = 1:N] # 2-dimensional
L_3 = [[x1,x2,x3] for x1 = 1:N, x2 = 1:N,x3 = 1:N] # 3-dimensional
...

#L_D = ??? # D-dimensional

我该如何定义?

不使用数组解析也没关系。

任何信息将不胜感激。

【问题讨论】:

那么你想用这个做什么?如果它是为了索引到以前存在的 N 维数组 X,那么你可以只做 CartesianIndices(X)。这适用于任何维度和大小。 @DNF 换句话说,CartesianIndices(X)CartesianIndices(size(X)) 做同样的事情。只有当X 不存在并且您需要一个 NxNxNx... 索引数组时,才需要所有这些数字运算。 【参考方案1】:

您似乎不需要 CartesianIndices,但为了记录,CartesianIndices 可以采用任何 Int 元组(更准确地说是 Dims aka NTupleN,Int where N)来表示数组的大小。 CartesianIndices((5,5)) 用于 5x5,CartesianIndices((2,8,3)) 用于 2x8x3 等。您可以快速创建 NxNxNx... 表示具有 D 维度的大小的元组与 NtotheD(N,D) = ntuple(i -> N, D)

【讨论】:

谢谢。我想在我的项目中使用这个 CartesianIndices。但是,我需要在我的项目中为每个格点使用 Float64 的索引。 Int64转Float64容易吗? Int64转Float64是很容易的,但是你不是说反过来问吗?我必须提醒您 CartesianIndexN 是仅 NTupleN,Int 的包装器,因此 NTupleN,Float64 不起作用。如果您像A[3.0] 那样进行索引,那么您的getindex 方法需要将3.0 转换为3。将Float64 转换为Int64 有点困难,因为如果您使用A[3.1],您必须决定如何舍入到最接近的整数。 我未来想要的是制作一个“单元”格子,它有 5 x 5 个点并按规则间隔划分。 → 单位格: 0.2, 0.4, 0.6, 0.8, 1.0 。这是 CartesianIndices 的工作吗? 可能。 CartesianIndexN 必须只包含 Int,但您可以根据需要从 Int 计算 Float64。在我的脑海中,您可以创建一个子类型 AbstractArray 的结构(请参阅 Julia 文档的接口页面)并简单地包装 CartesianIndices。该结构的 getindex 将索引 CartesianIndices 以获取 CartesianIndex,将 Int 转换为 Float64 值,然后返回它们。就像 CartesianIndices 一样,lattice 值是按需生成的,不需要预先在堆上分配内存(尽管您可以collect 它一次性生成所有值)。【参考方案2】:

您可以概括我在另一个答案中发布的vcat 方法,如下所示:

julia> lattice(N, D) = vcat.((reshape(1:N, ntuple(j -> j == i ? N : 1, D)) for i in 1:D)...)
lattice (generic function with 1 method)

julia> lattice(2, 1)
2-element VectorVectorInt64:
 [1]
 [2]

julia> lattice(2, 2)
2×2 MatrixVectorInt64:
 [1, 1]  [1, 2]
 [2, 1]  [2, 2]

julia> lattice(2, 3)
2×2×2 ArrayVectorInt64, 3:
[:, :, 1] =
 [1, 1, 1]  [1, 2, 1]
 [2, 1, 1]  [2, 2, 1]

[:, :, 2] =
 [1, 1, 2]  [1, 2, 2]
 [2, 1, 2]  [2, 2, 2]

julia> lattice(2, 4)
2×2×2×2 ArrayVectorInt64, 4:
[:, :, 1, 1] =
 [1, 1, 1, 1]  [1, 2, 1, 1]
 [2, 1, 1, 1]  [2, 2, 1, 1]

[:, :, 2, 1] =
 [1, 1, 2, 1]  [1, 2, 2, 1]
 [2, 1, 2, 1]  [2, 2, 2, 1]

[:, :, 1, 2] =
 [1, 1, 1, 2]  [1, 2, 1, 2]
 [2, 1, 1, 2]  [2, 2, 1, 2]

[:, :, 2, 2] =
 [1, 1, 2, 2]  [1, 2, 2, 2]
 [2, 1, 2, 2]  [2, 2, 2, 2]

julia> lattice(2, 5)
2×2×2×2×2 ArrayVectorInt64, 5:
[:, :, 1, 1, 1] =
 [1, 1, 1, 1, 1]  [1, 2, 1, 1, 1]
 [2, 1, 1, 1, 1]  [2, 2, 1, 1, 1]

[:, :, 2, 1, 1] =
 [1, 1, 2, 1, 1]  [1, 2, 2, 1, 1]
 [2, 1, 2, 1, 1]  [2, 2, 2, 1, 1]

[:, :, 1, 2, 1] =
 [1, 1, 1, 2, 1]  [1, 2, 1, 2, 1]
 [2, 1, 1, 2, 1]  [2, 2, 1, 2, 1]

[:, :, 2, 2, 1] =
 [1, 1, 2, 2, 1]  [1, 2, 2, 2, 1]
 [2, 1, 2, 2, 1]  [2, 2, 2, 2, 1]

[:, :, 1, 1, 2] =
 [1, 1, 1, 1, 2]  [1, 2, 1, 1, 2]
 [2, 1, 1, 1, 2]  [2, 2, 1, 1, 2]

[:, :, 2, 1, 2] =
 [1, 1, 2, 1, 2]  [1, 2, 2, 1, 2]
 [2, 1, 2, 1, 2]  [2, 2, 2, 1, 2]

[:, :, 1, 2, 2] =
 [1, 1, 1, 2, 2]  [1, 2, 1, 2, 2]
 [2, 1, 1, 2, 2]  [2, 2, 1, 2, 2]

[:, :, 2, 2, 2] =
 [1, 1, 2, 2, 2]  [1, 2, 2, 2, 2]
 [2, 1, 2, 2, 2]  [2, 2, 2, 2, 2]

【讨论】:

感谢您向我展示了许多使我易于理解的示例。

以上是关于在 Julia 中定义任意维度的多维数组的主要内容,如果未能解决你的问题,请参考以下文章

Matlab的多维数组操作

在julia脚本/ print()中使用shell Array输出格式化的多维数组

多维数组

如何从多维数组中获取维度(切片)

python如何减小维度

matlab如何创建多维数组