Семь раз отмерь – один отрежь (с)
Ну или вернее о той ерунде которую я сегодня написал.
Задача была такая: подменяем оконную процедуру, при получении определенного сообщения берем wParam, преобразуем к указателю на нужный класс и вызываем функцию член. Сразу представили ужасы, ждущие пользователя если нам придет это сообщение (откуда-нибудь), но без указателя на наш класс. А значит надо, как-то проверять. Как?
Первое что приходит в голову – dynamic_cast<>(), читаем что бы освежить запас мудрости:
Безопасное приведение по иерархии наследования, в том числе и для виртуального наследования.
dynamic_cast(base_class_ptr_expr)
Используется RTTI (Runtime Type Information), чтобы привести один указатель на объект класса к другому указателю на объект класса. Классы должны быть полиморфными, то есть в базовом классе должна быть хотя бы одна виртуальная функция. Если эти условие не соблюдено, ошибка возникнет на этапе компиляции. Если приведение невозможно, то об этом станет ясно только на этапе выполнения программы и будет возвращен NULL.
dynamic_cast(base_class_ref_expr)
Работа со ссылками происходит почти как с указателями, но в случае ошибки во время исполнения будет выброшено исключение bad_cast.
Спасибо Алёне, там же читаем и про остальные возможности приведения типов в c++.
dynamic_cast<>() приведение конечно безопасное, но wParam в XXX* преобразовывать оно разумеется не хочет, не компилится. Выкрутился следующим образом, благо у меня был базовый класс:
XXX * p = NULL;
try
{
p = dynamic_cast<XXX>(t);
}
catch(...) {}
if (p)
{
// all ok...
}
Абсолютно не представляю насколько это идеологически правильно, переносимо или хорошо, скорее всего неправильно, криво и непереносимо:). Но у меня работает, почему dymamic_cast кидает исключение вместо того что бы вернуть NULL, я правда не понимаю, но практика (Visual Studio 2008) показывает что так и есть.
Тестировалось передачей 0 в качестве wParam и заменой первой строчки на
XXXBase * t = reinterpret_cast<XXXBase>(tt);
Все ок – исключение ловится, замалчивается, указатель остается нулевой.