首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >用dtw计算距离矩阵

用dtw计算距离矩阵
EN

Stack Overflow用户
提问于 2018-03-27 05:11:35
回答 2查看 2.4K关注 0票数 1

在从day1到day26的时间序列中,我有两个用于控制和处理的归一化读取计数矩阵。我想通过动态时间包装来计算距离矩阵,然后将其用于聚类,但似乎太复杂了。我这样做了;谁可以帮助更多的澄清?非常感谢

代码语言:javascript
复制
> head(control[,1:4])
               MAST2     WWC2  PHYHIPL   R3HDM2
Control_D1  6.591024 5.695156 3.388652 5.756384
Control_D1 8.043454 5.365221 6.859768 6.936970
Control_D3 7.731590 4.868267 6.919972 6.931073
Control_D4 8.129948 5.105528 6.627016 7.090268
Control_D5 7.690863 4.729501 6.824746 6.904610
Control_D6 8.101723 5.334501 6.868990 7.115883
> 

> head(lead[,1:4])
              MAST2     WWC2  PHYHIPL   R3HDM2
Lead30_D1  6.418423 5.610699 3.734425 5.778046
Lead30_D2 7.918360 4.295191 6.559294 6.780952
Lead30_D3 7.807142 4.294722 6.599187 6.716040
Lead30_D4 7.856720 4.432136 6.572337 6.848483
Lead30_D5 7.827311 4.204738 6.607107 6.784094
Lead30_D6 7.848760 4.458451 6.581216 6.943003
>
> dim(control)
[1]   26 2603
> dim(lead)
[1]   26 2603
library(dtw)

for (i in control) { 
  for (j in lead) { 
    result[i,j] <- dtw( dist(control[,,i],lead[,,j]), distance.only=T )$normalizedDistance 
  }
}

这么说的

代码语言:javascript
复制
Error in lead[, , j] : incorrect number of dimensions
EN

回答 2

Stack Overflow用户

发布于 2018-06-10 01:09:36

已经有类似于你的问题,但答案还不是太详细。以下是您需要了解的详细信息,在R的特定情况下。

计算交叉距离矩阵

proxy软件包是专门为计算交叉距离矩阵而开发的。您应该检查它的小插曲,以了解它已经实现了哪些措施。它的用法示例如下:

代码语言:javascript
复制
set.seed(1L)
sample_data <- matrix(rnorm(50L), nrow = 5L, ncol = 10L)

suppressPackageStartupMessages(library(proxy))
distance_matrix <- proxy::dist(sample_data, method = "euclidean", 
                               upper = TRUE, diag = TRUE)
print(distance_matrix)
#>          1        2        3        4        5
#> 1 0.000000 2.636027 3.834764 5.943374 3.704322
#> 2 2.636027 0.000000 2.587398 4.515470 2.310364
#> 3 3.834764 2.587398 0.000000 4.008678 3.899561
#> 4 5.943374 4.515470 4.008678 0.000000 5.059321
#> 5 3.704322 2.310364 3.899561 5.059321 0.000000

注意:在时间序列的上下文中,proxy将矩阵中的每个视为一个序列,这可以从上面的sample_data5x10矩阵以及产生的交叉距离矩阵是5x5的事实中得到证实。

使用DTW距离

dtw包实现了DTW的许多变体,并且它还利用了proxy。您可以使用以下命令计算DTW距离矩阵:

代码语言:javascript
复制
suppressPackageStartupMessages(library(dtw))
dtw_distmat <- proxy::dist(sample_data, method = "dtw", 
                           upper = TRUE, diag = TRUE)
print(distance_matrix)
#>          1        2        3        4        5
#> 1 0.000000 2.636027 3.834764 5.943374 3.704322
#> 2 2.636027 0.000000 2.587398 4.515470 2.310364
#> 3 3.834764 2.587398 0.000000 4.008678 3.899561
#> 4 5.943374 4.515470 4.008678 0.000000 5.059321
#> 5 3.704322 2.310364 3.899561 5.059321 0.000000

使用自定义距离

proxy的一个优点是它提供了注册自定义函数的选项。您似乎对DTW的规范化版本感兴趣,所以您可以这样做:

代码语言:javascript
复制
ndtw <- function(x, y = NULL, ...) {
    dtw::dtw(x, y, ..., distance.only = TRUE)$normalizedDistance
}

pr_DB$set_entry(
  FUN = ndtw,
  names = "ndtw",
  loop = TRUE,
  distance = TRUE
)

ndtw_distmat <- proxy::dist(sample_data, method = "ndtw",
                            upper = TRUE, diag = TRUE)
print(ndtw_distmat)
#>           1         2         3         4         5
#> 1 0.0000000 0.4046622 0.5075772 0.6789465 0.5290478
#> 2 0.4046622 0.0000000 0.3630849 0.4866252 0.3612722
#> 3 0.5075772 0.3630849 0.0000000 0.5678698 0.3303344
#> 4 0.6789465 0.4866252 0.5678698 0.0000000 0.5078112
#> 5 0.5290478 0.3612722 0.3303344 0.5078112 0.0000000

有关更多信息,请参阅pr_DB的文档。

其他DTW实现

dtwclust包(我制作的)实现了一个基本但速度更快的DTW版本,它可以使用多线程,还可以利用proxy

代码语言:javascript
复制
suppressPackageStartupMessages(library(dtwclust))
dtw_basic_distmat <- proxy::dist(sample_data, method = "dtw_basic", normalize = TRUE)
print(dtw_basic_distmat)
#>      [,1]      [,2]      [,3]      [,4]      [,5]     
#> [1,] 0.0000000 0.4046622 0.5075772 0.6789465 0.5290478
#> [2,] 0.4046622 0.0000000 0.3630849 0.4866252 0.3612722
#> [3,] 0.5075772 0.3630849 0.0000000 0.5678698 0.3303344
#> [4,] 0.6789465 0.4866252 0.5678698 0.0000000 0.5078112
#> [5,] 0.5290478 0.3612722 0.3303344 0.5078112 0.0000000

dtw_basic实现只支持两个步骤模式和一个窗口类型,但它的速度要快得多:

代码语言:javascript
复制
suppressPackageStartupMessages(library(microbenchmark))
microbenchmark(
  proxy::dist(sample_data, method = "dtw", window.type = "sakoechiba", window.size = 5L),
  proxy::dist(sample_data, method = "dtw_basic", window.size = 5L)
)

Unit: microseconds
                                                                                        expr      min       lq     mean
 proxy::dist(sample_data, method = "dtw", window.type = "sakoechiba",      window.size = 5L) 5279.124 5621.742 6070.069
                            proxy::dist(sample_data, method = "dtw_basic", window.size = 5L)  657.966  710.418  776.474
   median       uq       max neval cld
 5802.354 6348.199 10411.000   100   b
  752.282  814.037  1161.626   100  a

另一个多线程实现包含在parallelDist包中,尽管我没有亲自测试过它。

多变量或多维时间序列

单个多变量序列通常是一个矩阵,其中时间跨越行,多个变量跨越列。DTW也适用于它们:

代码语言:javascript
复制
mv_series1 <- matrix(rnorm(15L), nrow = 5L, ncol = 3L)
mv_series2 <- matrix(rnorm(15L), nrow = 5L, ncol = 3L)
print(dtw_distance <- dtw_basic(mv_series1, mv_series2))
#> [1] 22.80421

proxy的好处是,它还可以计算列表中包含的对象之间的距离,因此您可以在矩阵列表中放置几个多元序列:

代码语言:javascript
复制
mv_series <- lapply(1L:5L, function(dummy) {
  matrix(rnorm(15L), nrow = 5L, ncol = 3L)
})

mv_distmat_dtwclust <- proxy::dist(mv_series, method = "dtw_basic")
print(mv_distmat_dtwclust)
#>      [,1]     [,2]     [,3]     [,4]     [,5]    
#> [1,]  0.00000 27.43599 32.14207 36.42211 31.19279
#> [2,] 27.43599  0.00000 20.88470 23.88436 29.73219
#> [3,] 32.14207 20.88470  0.00000 22.14376 29.99899
#> [4,] 36.42211 23.88436 22.14376  0.00000 28.81111
#> [5,] 31.19279 29.73219 29.99899 28.81111  0.00000

你的案例

不管您选择什么,您都可以使用proxy获得结果,但是由于您没有提供全部数据,所以我不能给您提供更具体的示例。我假设dtwclust::dtw_basic(control[, 1:4], lead[, 1:4], normalize = TRUE)会给出一对序列之间的距离,假设你把每一对序列都看作是一个有4个变量的多变量序列。

票数 5
EN

Stack Overflow用户

发布于 2018-03-27 05:45:47

如果你的问题是“为什么我会得到这个错误?”答案是,你正在尝试对一个矩阵进行子集,这是一个根据第三维的二维数组。

请参见:

代码语言:javascript
复制
dim(lead)
# [1] 26 2603
lead[,,6.418423] # yes, that's the value j has the first time through the loop
# This will reproduce your error
lead[,,1]
# This will also reproduce your error

希望您现在能看到您有几个问题:

你正在尝试根据一个三维矩阵对一个矩阵进行子集,你的ij值分别是controllead中的值。你可以使用它们作为它们的值,或者你可以生成索引,例如,for(i in seq_along(control),如果你计划使用它来做其他事情,而不是得到相同的值。

  • 将它带到下一步,不清楚你想要传递给dist函数的是什么。dist采用单个矩阵并计算其行之间的距离。您似乎正在尝试传递来自两个不同矩阵的两个值,或者两个不同矩阵的两个子集。看起来您可能需要返回并查看xtr

文档中的示例

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/49500668

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档