从栅格中提取值并附加到现有数据帧

问题描述 投票:0回答:2

我试图从rasterstack中提取值并将它们附加到现有的数据帧。这些值是一组指标(来自r包SDMtools的PatchStat),我可以将其提取为列表格式,但我仍然试图将值绑定到现有的数据帧。

输入数据:

library(sp)
library(sf)
library(raster)
library(dplyr)
library(SDMTools)

mydata <- read.table(header=TRUE, text = "
                         animal        X       Y ord.year
1 pb_20414 157978.9 2323819     2009168
2 pb_20414 156476.3 2325586     2009168
3 pb_06817 188512.0 2299679     2006263
4 pb_06817 207270.9 2287248     2006264")

# add rasters

s <- stack(system.file("external/rlogo.grd", package="raster")) 
names(s) <- c('masie_ice_r00_v01_2009168_4km', 'masie_ice_r00_v01_2006263_4km', 'masie_ice_r00_v01_2006264_4km')

# Create sp object

projection <-CRS('+proj=stere +lat_0=90 +lat_ts=60 +lon_0=-80 +k=1 +x_0=0 +y_0=0 +ellps=WGS84 +units=m + datum=WGS84 +no_defs +towgs84=0,0,0') # matches MASIE raster
coords <- cbind(mydata$X, mydata$Y)
mydata.sp <- SpatialPointsDataFrame(coords = coords, data = mydata, proj4string = projection)

# Create sf object

mydata.sf <- st_as_sf(mydata) 
mydata.buf30 <- st_buffer(mydata.sf, 30000)

我的目标是将每个GPS点(X,Y)与日期正确的GeoTIFF(mydata$ord.year)相匹配,将光栅裁剪为(空间显式的)30 km缓冲区,在程序SDMtools中运行PatchStat for R,并将结果附加到原始数据帧。问题是PatchStat结果是在数据框中提供的,因此我无法将这些结果与现有的数据帧进行匹配。

以下是运行PatchStat时提供的结果示例:

 patchID n.cell n.core.cell n.edges.perimeter n.edges.internal area core.area perimeter
2       3     73          13                86              206   73        13        86
  perim.area.ratio shape.index frac.dim.index core.area.index
2         1.178082    2.388889       1.430175       0.1780822 

以下是我迄今为止所做的事情:

# separate date component of TIF name to correspond to mydata$ord.year 

stack <- list()
date<-vector()
for (i in 1:length(rasterlist)) {
  stack[[i]]<-raster(rasterlist[i])
  tt<-unlist(strsplit(names(stack[[i]]), "[_]"))
  date[i]<-tt[which(nchar(tt)==max(nchar(tt)))]
}

st <- stack(stack) # Create rasterstack object

# crop raster to buffer

mydata.sp <- as(mydata.sf, 'Spatial') # back to sp object

# pull raster data from GeoTIFF that corresponds to ordinal date

pat <- list()
for (i in 1:nrow(mydata.sp)) {
  st2<-st[[which(date==mydata.sp$ord.year[i])]]
  GeoCrop <- raster::crop(st2, mydata.sp[i,])
  GeoCrop_mask <- raster::mask(GeoCrop, mydata.sp[i,])
  pat[[i]] <- PatchStat(GeoCrop_mask)}

另外,我已经消除了两种土地覆盖类型中的一种,因此列表中的每个元素只有一行:pat2 <- lapply(pat, `[`, -1,) # remove first row in each list element so only one row remains (using program plyr for R)

现在,我想将这些行与原始数据帧匹配,以便pat2[[1]]像这样附加到mydata.sp[1,](假设a,b和c是我原始SpatialPointsDataFrame中的元数据列)。我想添加PatchStat的所有数据列,但为了节省时间和空间,我只包括前三个:

a   b   c   PatchID   n.cell   n.core.cell 
1   2   3   3         73       13

注意:如果可能的话,我希望将整个过程包含在for循环中,以最大限度地减少错误空间和处理时间。

非常感谢!

r spatial r-raster sp
2个回答
0
投票

感谢您努力提供示例数据。但它仍然不完整(它指的是我们没有的文件。你可以这样做

library(raster)
library(SDMTools)

s <- stack(system.file("external/rlogo.grd", package="raster")) 
s <- round(s / 50) # to have fewer patches
names(s) <- c('masie_ice_r00_v01_2009168_4km', 'masie_ice_r00_v01_2006263_4km', 'masie_ice_r00_v01_2006264_4km')

df <- data.frame(ord.year=c("2009168", "2009168", "2006263", "2006264"))
pts <- SpatialPoints(cbind(c(20,40,60,80), c(20,40,60,20)))
crs(pts) <- crs(s)
pts <- SpatialPointsDataFrame(pts, df)

制作缓冲区

b <- buffer(pts, 15, dissolve=FALSE)

获取匹配的名称

nms <- names(s)
nms <- gsub('masie_ice_r00_v01_', '', nms)
nms <- gsub('_4km', '', nms)

循环以匹配名称,并将结果放入列表中

p <- list()
for (i in 1:length(b)) {
    j <- which(b$ord.year[i] == nms)
    r <- s[[j]]
    z <- crop(r, b[i,])
    z <- mask(z, b[i,])
    p[[i]] <- PatchStat(z)
}

请注意,p的每个元素都有一个包含多个行和列的data.frame。

 p[[1]]
 #patchID n.cell n.core.cell n.edges.perimeter n.edges.internal area core.area perimeter perim.area.ratio shape.index frac.dim.index core.area.index
 #1       1     53           5                68              144   53         5        68        1.2830189    2.266667       1.427207      0.09433962
 #2       2    123           8               182              310  123         8       182        1.4796748    3.956522       1.586686      0.06504065
 #3       3    149          31               190              406  149        31       190        1.2751678    3.800000       1.543074      0.20805369
 #4       4     54           2               114              102   54         2       114        2.1111111    3.800000       1.679578      0.03703704
 #5       5    337         206               146             1202  337       206       146        0.4332344    1.972973       1.236172      0.61127596

如果你只想要第一行

 pp <- t(sapply(p, function(i) i[1,]))

将其与原始data.frame结合起来并非易事

 dfpp <- cbind(df, pp)

0
投票

好吧,我做了这个非常丑陋的事情并得到了我想要的东西。但我不喜欢它。如果有人有更好的主意,我很乐意听到它!

# Change objects to df  
pat2 <- lapply(pat, `[`, -1,) # remove first row in each list element

library(plyr) # ldply command

pat3 <- ldply (pat2, data.frame)

pat4 <- bind_cols(pb, pat3)
© www.soinside.com 2019 - 2024. All rights reserved.