为什么环境的`object.size`小于环境中对象的`object.size`?

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

在编写函数工厂时遇到了令人困惑的情况。我有一个识别信息列和一个包含数据的列表列。

tib <-
    tibble(
        id_1 = letters[1:3],
        id_2 = LETTERS[1:3],
        data = list(mtcars, iris, volcano)
    )

我创建了一个函数工厂,以便轻松获取存储在此tibble中的数据集。

data_getter <- function(data) {
    force(data)
    function(id_1, id_2) {
        where <- data$id_1 == id_1 & data$id_2 == id_2
        data[where, ]$data %>% 
            purrr::flatten_df()
    }
}

get_from_tib <- data_getter(tib)

实际上,我正在使用的数据框非常大。我的理解是,get_from_tib捕获了一个包含tib的环境。当我这样做时,我担心通过复制这个大型数据集,我会开始耗尽内存。然而,当我的函数的object.size小于据称被它捕获的数据集时,我感到非常惊讶!

object.size(tib)
21752 bytes

object.size(get_from_tib)
8416 bytes

即使我从工作环境中删除tib,这仍然有效

rm(tib)

object.size(get_from_tib)
8416 bytes

为什么环境要小得多? object.size是否只是用于获取环境中包含的所有对象的大小的错误函数?

以下代表

library(magrittr)

tib <-
    dplyr::tibble(
        id_1 = letters[1:3],
        id_2 = LETTERS[1:3],
        data = list(mtcars, iris, USArrests)
    )


data_getter <- function(data) {
    force(data)
    function(id_1, id_2) {
        where <- data$id_1 == id_1 & data$id_2 == id_2
        data[where, ]$data %>% 
            purrr::flatten_df()
    }
}

get_from_tib <- data_getter(tib)

get_from_tib('c', 'C')
#> # A tibble: 50 x 4
#>    Murder Assault UrbanPop  Rape
#>     <dbl>   <int>    <int> <dbl>
#>  1   13.2     236       58  21.2
#>  2   10       263       48  44.5
#>  3    8.1     294       80  31  
#>  4    8.8     190       50  19.5
#>  5    9       276       91  40.6
#>  6    7.9     204       78  38.7
#>  7    3.3     110       77  11.1
#>  8    5.9     238       72  15.8
#>  9   15.4     335       80  31.9
#> 10   17.4     211       60  25.8
#> # … with 40 more rows


object.size(tib)
#> 21752 bytes
object.size(get_from_tib)
#> 8416 bytes


rm(tib)
object.size(get_from_tib)
#> 8416 bytes

reprex package创建于2019-04-30(v0.2.1)

r rlang
1个回答
3
投票

这与?object.size帮助页面中的部分有关

相关空间(例如,函数的环境和EXTPTRSXP中指针指向的内容)不包括在计算中。

data变量在闭包中被捕获;它并不是真正“在”data_getter返回的函数代码中。

pyry::object_size()函数在考虑封闭环境方面做得更好。

pryr::object_size(tib)
# 21 kB
pryr::object_size(get_from_tib)
# 26.9 kB
© www.soinside.com 2019 - 2024. All rights reserved.