Throw and ternary operator in C++

后端 未结 4 1369
盖世英雄少女心
盖世英雄少女心 2020-12-03 22:19

The following code compiles with G++ 4.6.1, but not with Visual Studio 2008

return (m_something == 0) ? 
    throw std::logic_error(\"Something wrong happene         


        
相关标签:
4条回答
  • 2020-12-03 22:46

    It is standard C++. Either (or both) of the then/else expressions in a conditional expression is allowed to be a throw-expression instead (C++98 5.16/2).

    If Visual Studio crashes when compiling it... that would seem to be unfortunate!

    0 讨论(0)
  • 2020-12-03 22:48

    Comeau compiles it without errors (here's my minimal compilable test case):

    int main(void)
    {
        int x = 17;
        return x ? throw "Something wrong happened" : 5;
    }
    

    which is pretty good evidence that it's allowed by the standard. So is the fact that MSVC crashes, rather than failing cleanly with an error.

    Also, it appears to be fixed in VC++ 2010

    R:\>cl ternarythrowtest.cpp
    Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 16.00.40219.01 for 80x86
    Copyright (C) Microsoft Corporation.  All rights reserved.
    
    ternarythrowtest.cpp
    Microsoft (R) Incremental Linker Version 10.00.40219.01
    Copyright (C) Microsoft Corporation.  All rights reserved.
    
    /out:ternarythrowtest.exe
    ternarythrowtest.obj
    

    and x64 version:

    R:\>cl ternarythrowtest.cpp
    Microsoft (R) C/C++ Optimizing Compiler Version 16.00.40219.01 for x64
    Copyright (C) Microsoft Corporation.  All rights reserved.
    
    ternarythrowtest.cpp
    Microsoft (R) Incremental Linker Version 10.00.40219.01
    Copyright (C) Microsoft Corporation.  All rights reserved.
    
    /out:ternarythrowtest.exe
    ternarythrowtest.obj
    

    Upgrade your compiler if possible, this is far from the only bug fixed in 2010.

    0 讨论(0)
  • 2020-12-03 23:03

    From the C++11 February Draft

    § 5.16/2 If either the second or the third operand has type (possibly cv-qualified) void, then the lvalue-to-rvalue (4.1), array-to-pointer (4.2), and function-to-pointer (4.3) standard conversions are performed on the second and third operands, and one of the following shall hold:
    — The second or the third operand (but not both) is a throw-expression (15.1); the result is of the type of the other and is a prvalue.
    — Both the second and the third operands have type void; the result is of type void and is a prvalue. [ Note: This includes the case where both operands are throw-expressions. —end note ]

    It appears that throw counts as evaluating to a void, and that this is allowed.

    0 讨论(0)
  • 2020-12-03 23:09

    The internal crash can be considered a bug of Visual Studio. A compiler should never crash because of the code being compiled.

    This is a very strange usage of the ternary operator, a simple if before the return would be a much preferable idiom:

    if(m_something == 0)
        throw std::logic_error("Something wrong happened");
    
    return m_something;
    
    0 讨论(0)
提交回复
热议问题