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 / 78901654
Accepted
Javier
Javier
Asked: 2024-08-22 20:46:50 +0800 CST2024-08-22 20:46:50 +0800 CST 2024-08-22 20:46:50 +0800 CST

O relançamento em multi-catch está formalmente definido? [duplicado]

  • 772
Esta pergunta já tem respostas aqui :
Por que é legal lançar novamente um Throwable em certos casos, sem declará-lo? (3 respostas)
Fechado há 14 horas .

Esta postagem foi editada e enviada para revisão há 13 horas .

Por que é throw epermitido, mas não throw a, no seguinte caso?

  void test() {
    try {
      System.out.println();
    } catch (Error | RuntimeException e) {
      var a = e;
      //throw a; unreported exception Throwable; must be caught or declared to be thrown
      throw e;
    }
  }

Tudo isso intuitivamente faz sentido, mas a partir do JLS 14.20

Uma cláusula multi-catch pode ser considerada uma sequência de cláusulas uni-catch. Ou seja, uma cláusula catch onde o tipo do parâmetro de exceção é denotado como uma união D1|D2|...|Dné equivalente a uma sequência de n cláusulas catch onde os tipos dos parâmetros de exceção são tipos de classe D1, D2, ..., Dnrespectivamente. No bloco de cada uma das n cláusulas catch, o tipo declarado do parâmetro de exceção é lub(D1, D2, ..., Dn).

Como lub(Error,RuntimeException)is Throwable, o código acima deve ser equivalente a:

    try {
      System.out.println();
    } catch (Error e) {
      Throwable lub = e;
      throw lub;
    } catch (RuntimeException e) {
      Throwable lub = e;
      throw lub;
    }

(que obviamente não compila)

Além disso, o tipo de aé o tipo de e"quando tratado como se não aparecesse em um contexto de atribuição" ( JLS 14.4.1 ), mas como mostrado acima não é o mesmo que o tipo de e.

Há algo que eu esqueci?


Editar : esta não é uma duplicata de Por que é legal lançar novamente um Throwable em certos casos, sem declará-lo? porque esta questão é específica para multi-catch (que não é discutida lá) e surge devido a um mal-entendido de um fragmento específico do JLS que aborda multi-catch. As respostas fornecidas nesta pergunta me ajudaram a ligar os pontos :)

java
  • 3 3 respostas
  • 89 Views

3 respostas

  • Voted
  1. Best Answer
    Sweeper
    2024-08-22T21:18:42+08:002024-08-22T21:18:42+08:00

    Isso é descrito em 11.2.2 . Isso basicamente se resume a "parâmetros de exceção catch (...)são um caso especial".

    Nosso objetivo é mostrar throw e;compilações. Começamos com esta linha em 11.2.3:

    É um erro em tempo de compilação se um método ou corpo de construtor puder lançar alguma classe de exceção E quando E for uma classe de exceção verificada e E não for uma subclasse de alguma classe declarada na cláusula throws do método ou construtor.

    Então, queremos mostrar que a tryinstrução in testnão pode ser lançada Throwable. Observe que “pode lançar” neste contexto é um termo rigorosamente definido. De acordo com 11.2.2,

    Uma instrução try pode lançar uma exceção de classe E se:

    • O bloco try pode lançar E, [...]

    • Algum bloco catch da instrução try pode lançar E e nenhum bloco final está presente ou o bloco final pode ser concluído normalmente; ou

    • Um bloco final está presente e pode lançar E.

    Apenas o segundo ponto é relevante aqui. Vamos mostrar que o catchbloco não pode ser lançado Throwable. Ou seja, throw e;não pode lançar Throwable.

    Uma instrução throw cuja expressão lançada é um parâmetro de exceção final ou efetivamente final de uma cláusula catch C pode lançar uma classe de exceção E se:

    • E é uma classe de exceção que o bloco try da instrução try que declara C pode lançar; e

    • E é atribuição compatível com qualquer uma das classes de exceção capturáveis ​​de C; e

    • E não é compatível com nenhuma das classes de exceção capturáveis ​​das cláusulas catch declaradas à esquerda de C na mesma instrução try.

    throw e;aqui falha imediatamente o primeiro ponto, então podemos concluir que throw e;não é possível lançar Throwable.

    Se fosse throw a;, então o acima não se aplica, mas outra cláusula se aplica:

    Uma instrução throw cuja expressão lançada tem tipo estático E e não é um parâmetro de exceção final ou efetivamente final pode lançar E ou qualquer classe de exceção que a expressão lançada possa lançar.

    • 6
  2. Mark Rotteveel
    2024-08-22T21:01:50+08:002024-08-22T21:01:50+08:00

    O compilador Java é inteligente o suficiente para reconhecer que capturar ee lançar esignificará lançar RuntimeExceptionou Errore, portanto, não uma exceção verificada que não foi declarada.

    Por outro lado, o tipo de var aé inferido Throwable(e não Error | RuntimeExceptioncomo você pode pensar, já que tal declaração não é realmente válida nesse contexto!) e, portanto, throw asignifica que você lança um Throwablee, portanto, o compilador não sabe com certeza você não está lançando uma exceção verificada e, como tal, gera o erro.

    O compilador poderia ser alterado para reconhecer isso? Provavelmente sim. Faria sentido fazer isso? Provavelmente não (ou pelo menos o valor de fazer provavelmente não vale o investimento).

    Isto segue as regras especificadas em 14.18 A throwDeclaração combinada com 11.2.2 Análise de Exceções de Declarações .

    Segue especificamente desta regra em 11.2.2 para throw e:

    Uma throwinstrução cuja expressão lançada é um parâmetro de exceção final ou efetivamente final de uma catchcláusula C pode lançar uma classe de exceção E se:

    • E é uma classe de exceção que o bloco try da instrução try que declara C pode lançar; e
    • E é atribuição compatível com qualquer uma das classes de exceção capturáveis ​​de C; e
    • E não é compatível com nenhuma das classes de exceção capturáveis ​​das cláusulas catch declaradas à esquerda de C na mesma instrução try.

    Por outro lado, para throw a, aplica-se o seguinte:

    Uma instrução throw ( §14.18 ) cuja expressão lançada tem tipo estático E e não é um parâmetro de exceção final ou efetivamente final pode lançar E ou qualquer classe de exceção que a expressão lançada possa lançar.

    • 4
  3. Stephen C
    2024-08-22T21:17:31+08:002024-08-22T21:17:31+08:00

    A parte saliente do JLS é 11.2.2 que diz:

    Uma throwinstrução cuja expressão lançada é um finalparâmetro de exceção efetivamente final de uma cláusula catch C pode lançar uma classe de exceção E se:

    • E é uma classe de exceção que o bloco try da tryinstrução que declara C pode lançar; e

    • E é atribuição compatível com qualquer uma das classes de exceção capturáveis ​​de C; e

    • ...

    Isso explica por que o seguinte irá lançar um Errorou RuntimeException:

      void test1() {
        try {
          ...
        } catch (Error | RuntimeException e) {
          throw e;
        }
      }
    

    Mas no seguinte:

      void test2() {
        try {
          ...
        } catch (Error | RuntimeException e) {
          var a = e;
          throw a;
        }
      }
    

    o tipo inferido aé Throwable(o limite superior). E isso resultará em um erro de compilação... se o método envolvente não for declarado como throw Throwable.

    Para a vardeclaração, JLS 14.4.1 diz o seguinte:

    Se LocalVariableType for var, então seja T o tipo da expressão inicializadora quando tratada como se não aparecesse em um contexto de atribuição e, portanto, fosse uma expressão autônoma (§15.2). O tipo da variável local é a projeção ascendente de T em relação a todas as variáveis ​​de tipo sintético mencionadas por T (§4.10.5).

    Isso é difícil de analisar, mas meu entendimento é que "a projeção ascendente de T" é o mesmo tipo conhecido como "limite superior mínimo". No test2exemplo seria Throwable.

    Observe que o texto JLS que você citou em sua pergunta é um texto ilustrativo e não normativo. E não descreve o que acontece quando você (re) lança a exceção capturada.

    • 2

relate perguntas

  • Lock Condition.notify está lançando java.lang.IllegalMonitorStateException

  • Resposta de microsserviço Muitos para Um não aparece no carteiro

  • Validação personalizada do SpringBoot Bean

  • Os soquetes Java são FIFO?

  • Por que não é possível / desencorajado definir um lado do servidor de tempo limite de solicitação?

Sidebar

Stats

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

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

    • 1 respostas
  • Marko Smith

    Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle?

    • 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

    Quando devo usar um std::inplace_vector em vez de um std::vector?

    • 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
  • Marko Smith

    Estou tentando fazer o jogo pacman usando apenas o módulo Turtle Random e Math

    • 1 respostas
  • Martin Hope
    Aleksandr Dubinsky Por que a correspondência de padrões com o switch no InetAddress falha com 'não cobre todos os valores de entrada possíveis'? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer Quando devo usar um std::inplace_vector em vez de um std::vector? 2024-10-29 23:01:00 +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
  • Martin Hope
    MarkB Por que o GCC gera código que executa condicionalmente uma implementação SIMD? 2024-02-17 06:17:14 +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