在C ++中,我有一个只需要对数组进行只读访问但是被错误地声明为接收非const指针的函数:
size_t countZeroes( int* array, size_t count )
{
size_t result = 0;
for( size_t i = 0; i < count; i++ ) {
if( array[i] == 0 ) {
++result;
}
}
return result;
}
我需要为const数组调用它:
static const int Array[] = { 10, 20, 0, 2};
countZeroes( const_cast<int*>( Array ), sizeof( Array ) / sizeof( Array[0] ) );
这会是未定义的行为吗?如果是这样 - 程序什么时候运行到UB-进行const_cast并调用functon或访问数组时?
答案 0 :(得分:13)
是的,允许(如果危险!)。它是对const
对象的实际写入,它引发了未定义的行为,而不是转换本身(7.1.5.1/4 [dcl.type.cv])。
正如5.2.11 / 7 [expr.const.cast]中的标准注释,根据对象的类型,尝试通过抛弃const
的结果写入指针可能会产生未定义行为。
答案 1 :(得分:1)
由于您的代码没有修改数组,并且您告诉编译器您使用const_cast
知道自己在做什么,所以实际上您可以。但是,我相信你在技术上调用了未定义的行为。最好修复函数声明,或者编写,声明和使用它的const安全版本。
答案 2 :(得分:1)
是的,你可以这样做。不,只要该函数没有尝试写入数组,它就不是未定义的行为。
答案 3 :(得分:1)
const_cast
的问题总是一样的 - 它允许你“违反规则”,就像在void*
之间进行投射一样 - 确定你可以这样做,但问题是你为什么要?
在这种情况下,它当然可以,但您应该问问自己为什么不首先声明size_t countZeroes( const int* array, size_t count )
?
作为关于const_cast
的一般规则:
答案 4 :(得分:-2)
在最初定义为const_cast
的对象上使用const
是UB,因此未定义的行为会在您调用const_cast
时立即出现。