AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 79464245
Accepted
robertspierre
robertspierre
Asked: 2025-02-25 00:29:37 +0800 CST2025-02-25 00:29:37 +0800 CST 2025-02-25 00:29:37 +0800 CST

Converter um tipo de aviso específico em um erro

  • 772

Considere o seguinte código:

> df <- tibble(gender=c(1,1,0))
> df$male
Warning: Unknown or uninitialised column: `male`.
NULL

Como posso converter esse tipo de aviso específico em um erro?

Gostaria de algo assim options(warn = 2), mas apenas para esse tipo específico de aviso (ou seja, referindo-se a uma coluna em um tibble que não existe)

  • 5 5 respostas
  • 128 Views

5 respostas

  • Voted
  1. Billy34
    2025-02-25T01:07:18+08:002025-02-25T01:07:18+08:00

    É um pouco hackeado, mas você pode corrigir a função $, ou seja, o getter. O aviso é lançado por rlang::warnquando o nome da coluna não é encontrado em df. Ao substituir esta chamada por uma para rlang::aborttransformamos este aviso em um erro.

    patch_tibble <- function() {
      if(is.null(attr(tibble:::`$.tbl_df`, "patched"))) {
        tt <- get("$.tbl_df", envir=asNamespace("tibble"), inherits=FALSE)
        
        body(tt) <- methods::substituteDirect(body(tt), list(warn=quote(abort)))
        attr(tt, "patched") <- TRUE
        
        unlockBinding("$.tbl_df", asNamespace("tibble"))
        assign("$.tbl_df", tt, envir=asNamespace("tibble"), inherits = FALSE)
        lockBinding("$.tbl_df", asNamespace("tibble"))
        packageStartupMessage("Patched tibble!")
      }
    }
    
    library(tibble)
    patch_tibble()
    
    df <- tibble(gender=c(1,1,0))
    df$male
    
    Error in `df$male`:
    ! Unknown or uninitialised column: `male`.
    Run `rlang::last_trace()` to see where the error occurred.
    
    • 4
  2. Best Answer
    random_walk
    2025-02-25T01:37:52+08:002025-02-25T01:37:52+08:00

    A função original para o método S3 $.tbl_dfé definida como:

    function (x, name) 
    {
        out <- .subset2(x, name)
        if (is.null(out)) {
            warn(paste0("Unknown or uninitialised column: ", tick(name), 
                "."))
        }
        out
    }
    

    Você pode simplesmente substituir o aviso por um erro usando stopou rlang:abort():

    `$.tbl_df` <- function (x, name) {
      out <- .subset2(x, name)
      if (is.null(out)) {
        rlang::abort(paste0("Unknown or uninitialised column: ", name, "."))
      }
      out
    }
    

    Então, como esperado:

    > df$male
    Error in `$.tbl_df`(df, male) : Unknown or uninitialised column: male.
    

    Adenda

    Conforme apontado nos comentários, essa solução pode causar problemas se $.tbl_dfestiver sendo chamada internamente. Seguindo o exemplo de @Billy34, podemos substituir a $.tbl_dffunção interna pela nossa função modificada assim:

    patch_tibble <- function(){
      
      # The modified "$.tbl_df" function
      f <- function (x, name) {
        out <- .subset2(x, name)
        if (is.null(out)) {
          abort(paste0("Unknown or uninitialised column: ", tick(name), 
                       "."))
        }
        out
      }
      
      # Set the function's namespace to tibble
      environment(f) <- asNamespace("tibble")
      
      # Replace the original function with our modified version
      utils::assignInNamespace("$.tbl_df", f, "tibble")
      
      print("Patched tibble!")
    }
    

    Isso produz a saída:

    > patch_tibble()
    [1] "Patched tibble!"
    > df$male
    Error in `df$male`:
    ! Unknown or uninitialised column: `male`.
    
    • 3
  3. Roland
    2025-02-25T14:33:55+08:002025-02-25T14:33:55+08:00

    Com base na ideia da resposta de @BenBolker, você pode criar um manipulador global:

    globalCallingHandlers(warning = function(w) {
      if(grepl("Unknown or uninitialised column", w$message, fixed = TRUE)) 
        stop(errorCondition(w$message, class = "unknown_column_extract_error")) else w
      })
    
    
    library(tibble)
    df <- tibble(gender=c(1,1,0))
    df$male
    #Error: Unknown or uninitialised column: `male`.
    
    #doesn't affect other warnings
    as.numeric("a")
    #[1] NA
    #Warning message:
    #  NAs introduced by coercion 
    

    Claro que isso seria mais simples se o aviso fosse classificado.

    • 3
  4. Ben Bolker
    2025-02-25T01:08:23+08:002025-02-25T01:08:23+08:00

    Isso pode ser o mais próximo que você chegará (sem hackear como na resposta anterior): use um tryCatch()que verifique a mensagem de aviso (isso falhará se alguém usar configurações de idioma diferentes para onde o aviso foi traduzido).

    my_try <- function(expr, warnstr = "Unknown or uninitialised column") { 
       tryCatch(expr,
          warning = function(e) { 
             if(grepl(warnstr, e)) simpleError(e) })
    }
    my_try(df$male)
    
    • 2
  5. user2554330
    2025-02-25T05:30:37+08:002025-02-25T05:30:37+08:00

    Outros lhe deram soluções alternativas para esse problema. Nesta resposta, explicarei o que os tibbleautores poderiam ter feito para facilitar o trabalho.

    R tem suporte para erros e avisos classificados, mas eles não são muito usados. O rlangpacote tem o classargumento para cada um de abort(), warn()e inform(). Você não precisa usar rlang, tudo pode ser feito com funções base, mas elas empacotam bem.

    Então o que tibblepoderia ter feito é definir uma classe separada em cada aviso que eles emitem. Então você poderia capturar a classe específica correspondente ao aviso que lhe interessa e convertê-la em um erro. Ou você poderia criar seu próprio $.tbl_dfmétodo que faz isso e então escolher se quer ou não a conversão.

    Por exemplo, em vez de

    warn(paste0("Unknown or uninitialised column: ", tick(name), 
            "."))
    

    eles poderiam ter escrito

    warn(paste0("Unknown or uninitialised column: ", tick(name), 
            "."), class = "warn.$.tbl_df")
    

    e então você poderia correr

    tryCatch(df$male, `warn.$.tbl_df` = function(w) abort(w$message, class = "error.$.tbl_df", call = NULL))
    

    para capturar esse aviso e convertê-lo em um erro. Eu converti para um erro de classe com class "error.$.tbl_df"apenas para que algum outro código pudesse capturar isso, mas não há necessidade disso. Eu também configurei call = NULLpara que você não seja informado sobre uma linha no código $.tbl_dfque não significaria nada para você. Você pode ser capaz de encontrar a df$maleexpressão se brincar com a configuração; não tenho certeza de quão longe ela estaria.

    • 2

relate perguntas

  • Adicionar número de série para atividade de cópia ao blob

  • A fonte dinâmica do empacotador duplica artefatos

  • Selecione linhas por grupo com 1s consecutivos

  • Lista de chamada de API de gráfico subscritoSkus estados Privilégios insuficientes enquanto os privilégios são concedidos

  • Função para criar DFs separados com base no valor da coluna

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Reformatar números, inserindo separadores em posições fixas

    • 6 respostas
  • Marko Smith

    Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não?

    • 2 respostas
  • Marko Smith

    Problema com extensão desinstalada automaticamente do VScode (tema Material)

    • 2 respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Martin Hope
    Fantastic Mr Fox Somente o tipo copiável não é aceito na implementação std::vector do MSVC 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant Encontre o próximo dia da semana usando o cronógrafo 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor O inicializador de membro do construtor pode incluir a inicialização de outro membro? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul O C++20 mudou para permitir a conversão de `type(&)[N]` de matriz de limites conhecidos para `type(&)[]` de matriz de limites desconhecidos? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann Como/por que {2,3,10} e {x,3,10} com x=2 são ordenados de forma diferente? 2025-01-13 23:24:07 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST

Hot tag

python javascript c++ c# java typescript sql reactjs html

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve