why does knitr caching fail for data.table `:=`?

前端 未结 2 1321
我在风中等你
我在风中等你 2021-02-04 02:53

This is related in spirit to this question, but must be different in mechanism.

If you try to cache a knitr chunk that contains a data.table

2条回答
  •  逝去的感伤
    2021-02-04 03:57

    Speculation:

    Here is what appears to be going on.

    knitr quite sensibly caches objects as as soon as they are created. It then updates their cached value whenever it detects that they have been altered.

    data.table, though, bypasses R's normal copy-by-value assignment and replacement mechanisms, and uses a := operator rather than a =, <<-, or <-. As a result knitr isn't picking up the signals that DT has been changed by DT[, c:=5].

    Solution:

    Just add this block to your code wherever you'd like the current value of DT to be re-cached. It won't cost you anything memory or time-wise (since nothing except a reference is copied by DT <- DT) but it does effectively send a (fake) signal to knitr that DT has been updated:

    ```{r, cache=TRUE, echo=FALSE}
    DT <- DT 
    ```
    

    Working version of example doc:

    Check that it works by running this edited version of your doc:

    ```{r}
    library(data.table)
    ```
    Data.Table Markdown
    ========================================================
    Suppose we make a `data.table` in **R Markdown**
    ```{r, cache=TRUE}
    DT = data.table(a = rnorm(10))
    ```
    
    Then add a column using `:=`
    ```{r, cache=TRUE}
    DT[, c:=5] 
    ```
    
    ```{r, cache=TRUE, echo=FALSE}
    DT <- DT 
    ```
    
    Then we display that in a non-cached block
    ```{r, cache=FALSE}
    DT
    ```
    The first time you run this, the above will show a `c` column. 
    The second, third, and nth times, it will as well.
    

提交回复
热议问题