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 / 79486669
Accepted
0___________
0___________
Asked: 2025-03-05 21:41:21 +0800 CST2025-03-05 21:41:21 +0800 CST 2025-03-05 21:41:21 +0800 CST

Esta declaração é UB?

  • 772

Considerar:

typedef union 
{
    int intval;
    const int const_intval;
}myUnion_t;

int main(void)
{
    myUnion_t x = {0};

    x.intval = 5;

    printf("intval = %d\n", x.intval);
    printf("const_intval = %d\n", x.const_intval);
}
  1. O próprio UB é myUnion_tdefinido como é aqui?
  2. Existe alguma atribuição da intvalUB?

Nenhum dos compiladores testados emite aviso (compilados também com compiladores C++)

EDITAR

A duplicata está relacionada apenas à pergunta 2, não à principal.

c
  • 1 1 respostas
  • 1036 Views

1 respostas

  • Voted
  1. Best Answer
    John Bollinger
    2025-03-06T00:09:12+08:002025-03-06T00:09:12+08:00
    1. O próprio UB é myUnion_tdefinido como é aqui?

    Não.

    • O especificador de união fornecido na questão está em conformidade com a gramática C, conforme descrito em C23 6.7.3.2/1

    • Não há restrição contra membros serem constqualificados ou haver uma mistura de conste não- constmembros (C23 6.7.3.2/2-6). As restrições são onde encontraríamos limitações colocadas sobre a gramática, como no uso de constqualificação em particular onde tipos qualificados são permitidos em geral.

    • C permite explicitamente que

      Um membro de uma estrutura ou união pode ter qualquer tipo de objeto completo, exceto um tipo modificado de forma variável.

      (C23 6.7.3.2/11)

      constsendo um qualificador de tipo, ele faz parte dos tipos de quaisquer membros assim qualificados. Ou seja, const inté um tipo diferente de int, e está entre os "quaisquer tipos completos" que são permitidos para membros de união.

    A única coisa a ser questionada é o nome do tipo que termina em _t. C não tem nada em particular a dizer sobre isso, mas se POSIX é importante para você, então você deve estar ciente de que ele reserva nomes dessa forma.


    1. Existe alguma atribuição da intvalUB?

    Uma resposta a uma pergunta semelhante diz o mesmo, com base na versão C11 desta disposição C23 no parágrafo 6.4.7.1/7:

    Se for feita uma tentativa de modificar um objeto definido com um consttipo qualificado por meio do uso de um lvalue com consttipo não qualificado, o comportamento será indefinido.

    Eu tendo a concordar , mas não acho que a aplicação dessa provisão seja totalmente clara. As uniões consistem em membros logicamente distintos com armazenamento sobreposto (C23 6.2.5/25) e, teoricamente, uma união contém um valor para apenas um membro por vez (nota 38 do C23). Há uma questão, então, se atribuir a x.intvaldeve ser interpretado como modificar x.const_intval. Em que nível(is) a provisão acima deve ser aplicada? De fato, a resposta pode ser diferente para C++, que faz uma distinção mais forte entre membros de união ao rejeitar o comportamento definido para trocadilhos de tipo baseados em união -- mas lá, mesmo se a atribuição tivesse comportamento definido, a segunda printf()chamada, em que x.const_intvalé acessada, não teria.

    Um caminho diferente passaria pelas especificações para operações de atribuição — especificamente que operadores de atribuição devem ter lvalues ​​modificáveis ​​como seus operandos esquerdos (C23 6.5.17.1/2). Então, por C23 6.3.3.1/1,

    Um lvalue modificável é um lvalue que [...] se for uma estrutura ou união, não tem nenhum membro [...] com um tipo qualificado const.

    Também temos que quando um membro é acessado por meio de uma estrutura de host ou união pelo uso do operador .or ->, o tipo do lvalue resultante herda os qualificadores do lvalue que designa a estrutura de host ou união. Isso significa que se xfossem declarados, constentão atribuir a x.intvalteria um comportamento indefinido. Mas isso também não nos leva à conclusão esperada, porque x.intvalherda qualificadores do tipo de x, não modificabilidade diretamente. xem si não é um lvalue modificável, mas x.intvalparece satisfazer a definição da especificação de "lvalue modificável".

    Acho que é uma falha na especificação.

    A interpretação mais segura é que a atribuição para x.intvalproduz UB. Duplamente se x.const_intvalfor acessado subsequentemente, e triplamente se x.const_intvaltiver sido inicializado. Como uma questão prática, mesmo se você preferir interpretar a atribuição para x.intvalcomo tendo comportamento definido, há um risco significativo de que sua implementação C não tenha, de modo que a atribuição provoque comportamento inesperado e indesejado de seu programa na prática.

    • 15

relate perguntas

  • Multiplicação mais rápida que *

  • Usando uma macro para comprimento de string no especificador de formato scanf () em C

  • Como você pode definir o tipo de dados de #define para long double?

  • Ponteiros const incompatíveis

  • Mudança de cor não gradual no OpenGL

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