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 / 79016821
Accepted
VainMan
VainMan
Asked: 2024-09-24 10:53:21 +0800 CST2024-09-24 10:53:21 +0800 CST 2024-09-24 10:53:21 +0800 CST

a asserção estática dispara do tipo de retorno quando a restrição inicial não é satisfeita

  • 772

O GCC aceita esse código, o Clang e o MSVC o rejeitam devido à falha de asserção estática em assert. O que o padrão diz?

https://godbolt.org/z/PKMKzYGsc

template<typename T>
constexpr int assert() {
    static_assert(sizeof(T) == 1);
    return 1;
}

template<auto I>
using Int = int;

template<typename T> requires (sizeof(T) == 1)
constexpr auto foo(T) -> Int<assert<T>()> {
    return 1;
}

template<typename T> requires (sizeof(T) > 1)
constexpr auto foo(T a) -> Int<1> {
    return 2;
}

static_assert(foo('1') == 1);
static_assert(foo(2) == 2);

Saída do Clang:

<source>:3:19: error: static assertion failed due to requirement 'sizeof(int) == 1'
    3 |     static_assert(sizeof(T) == 1);
      |                   ^~~~~~~~~~~~~~
<source>:11:30: note: in instantiation of function template specialization 'assert<int>' requested here
   11 | constexpr auto foo(T) -> Int<assert<T>()> {
      |                              ^
<source>:21:15: note: while substituting deduced template arguments into function template 'foo' [with T = int]
   21 | static_assert(foo(2) == 2);
      |               ^
<source>:3:29: note: expression evaluates to '4 == 1'
    3 |     static_assert(sizeof(T) == 1);
c++
  • 2 2 respostas
  • 133 Views

2 respostas

  • Voted
  1. Best Answer
    duck
    2024-09-24T19:47:09+08:002024-09-24T19:47:09+08:00

    Durante a dedução do argumento do template, a verificação de satisfação das restrições associadas do template de função ocorre antes da substituição dos argumentos do template no tipo de função. Isso é descrito em [temp.deduct.general]/5 :

    Se o modelo de função tiver restrições associadas, essas restrições serão verificadas quanto à satisfação. Se as restrições não forem satisfeitas, a dedução de tipo falhará. [...] Se a dedução de tipo ainda não tiver falhado, todos os usos de parâmetros de modelo no tipo de função serão substituídos pelos valores de argumento deduzidos ou padrão correspondentes.

    Isso significa que a dedução de tipo deve falhar antes de encontrar o problema assertneste exemplo, tornando-o bem formado.

    Este resultado é o resultado do CWG2369 , antes do qual os dois passos acima aconteceram na ordem oposta. Evidentemente, Clang e MSVC não implementam essa mudança ainda, levando à divergência de implementação observada.

    • 6
  2. user12002570
    2024-09-24T11:37:13+08:002024-09-24T11:37:13+08:00

    O programa está bem formado e o gcc está correto ao aceitar o código conforme explicado abaixo.

    De over.over :

    Todas as funções com restrições associadas que não são satisfeitas ([temp.constr.decl]) são eliminadas do conjunto de funções selecionadas. Se mais de uma função permanecer no conjunto, todas as especializações de modelo de função no conjunto são eliminadas se o conjunto também contiver uma função que não seja uma especialização de modelo de função. Qualquer função não modelo F0 é eliminada se o conjunto contiver uma segunda função não modelo que seja mais restrita do que F0 de acordo com as regras de ordenação parcial de [temp.constr.order]. Qualquer especialização de modelo de função F1 é eliminada se o conjunto contiver uma segunda especialização de modelo de função cujo modelo de função seja mais especializado do que o modelo de função de F1 de acordo com as regras de ordenação parcial de [temp.func.order]. Após tais eliminações, se houver, deve permanecer exatamente uma função selecionada.

    (ênfase minha)

    Isso significa que para a chamada foo(2), a versão #1com sizeof(T)==1será eliminada do conjunto de funções selecionadas. E a outra (versão #2) que satisfaz a restrição será a única candidata viável e usada.

    • -2

relate perguntas

  • Por que os compiladores perdem a vetorização aqui?

  • Erro de compilação usando CMake com biblioteca [fechada]

  • Erro lançado toda vez que tento executar o premake

  • Como criar um tipo de octeto semelhante a std::byte em C++?

  • Somente operações bit a bit para std::byte em C++ 17?

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