Форум программистов, компьютерный форум CyberForum.ru

Обертка над boost::lexical_cast для работы с unsigned типами - C++

Войти
Регистрация
Восстановить пароль
 
 
Croessmah
Модератор
Эксперт С++
 Аватар для Croessmah
12283 / 6970 / 784
Регистрация: 27.09.2012
Сообщений: 17,283
Записей в блоге: 2
Завершенные тесты: 1
31.01.2016, 15:29     Обертка над boost::lexical_cast для работы с unsigned типами #1
Пару дней назад выяснилось, что boost::lexical_cast
не удовлетворяет нескольким требованиям в моей программе.
boost::lexical_cast отказывается обрабатывать и выбрасывает исключение,
если имеются пробелы в начале или в конце входной строки.
Также для unsigned типов, отрицательные значения обрабатываются нормально, т.е., например,
boost::lexical_cast<unsigned int>("-1") даст вполне валидное значение.
Для моих целей эти два факта были неприемлемы.
Собственно, Ticket #5494 как раз об этом,
и там был представлен такой велосипед:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
#include <boost/lexical_cast.hpp>
#include <boost/type_traits/is_unsigned.hpp>
 
template <bool is_unsigned>
struct unsigned_checker
{
    template<typename String_type>
    static inline void do_check(const String_type & str) { }
};
 
template <>
struct unsigned_checker<true>
{
    template<typename String_type>
    static inline void do_check(const String_type & str)
    {
        if( str[0] == '-' ) boost::throw_exception( boost::bad_lexical_cast() );
    }
};
 
template<typename Target, typename Source>
inline Target forced_lexical_cast(const Source &arg)
{
    unsigned_checker< boost::is_unsigned<Target>::value >::do_check(arg);
    return boost::lexical_cast<Target>( arg );
}
Но для моих целей он также не подходит, т.к. кидает исключение и для строки "-0",
а ведь "-0", как и "+0" - вполне валидные значения.
Поэтому был написан велосипед:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
namespace detail
{
    struct string_cast_impl
    {
        template<typename TargetType>
        static
        std::enable_if_t<std::is_unsigned<TargetType>::value, TargetType>
        cast(const std::string& source)
        {
            if (!source.empty() && source[0] == '-'){
                TargetType result = boost::lexical_cast<TargetType>(source) ;
                return (result!=0) ? throw boost::bad_lexical_cast(),TargetType(0) : result  ;
            }
            return boost::lexical_cast<TargetType>(source);
        }
 
 
 
        template<typename TargetType>
        static
        std::enable_if_t<!std::is_unsigned<TargetType>::value, TargetType>
        cast(const std::string& source)
        {
            return boost::lexical_cast<TargetType>(source) ;
        }
    };
}//end of detail
 
 
 
template<typename TargetType, typename StringType>
std::decay_t<TargetType> string_cast(StringType&& source)
{
    using DecayType = std::decay_t<TargetType> ;
    std::string str(std::forward<StringType>(source)) ;
    boost::trim(str) ;
    return detail::string_cast_impl::cast<DecayType>(str) ;
}
Решил выложить, вдруг кому пригодится, или будут дополнения/исправления.

Использование:
C++
1
2
3
4
5
int main(int argc, char ** argv)
{
    std::cout << string_cast<int>("-100") << std::endl ;
    std::cout << string_cast<unsigned int>("-100") << std::endl ;
}

P.S. Спасибо пользователям Kastaneda и hoggy за правки и дополнения.

Добавлено через 31 минуту
Вариант под C++11:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
namespace detail
{
    struct string_cast_impl
    {
        template<typename TargetType>
        static
        typename std::enable_if<std::is_unsigned<TargetType>::value, TargetType>::type
        cast(const std::string& source)
        {
            if (!source.empty() && source[0] == '-'){
                TargetType result = boost::lexical_cast<TargetType>(source) ;
                return (result!=0) ? throw boost::bad_lexical_cast(),TargetType(0) : result  ;
            }
            return boost::lexical_cast<TargetType>(source);
        }
 
 
 
        template<typename TargetType>
        static
        typename std::enable_if<!std::is_unsigned<TargetType>::value, TargetType>::type
        cast(const std::string& source)
        {
            return boost::lexical_cast<TargetType>(source) ;
        }
    };
}//end of detail
 
 
 
template<typename TargetType, typename StringType>
typename std::decay<TargetType>::type string_cast(StringType&& source)
{
    using DecayType = typename std::decay<TargetType>::type ;
    std::string str(std::forward<StringType>(source)) ;
    boost::trim(str) ;
    return detail::string_cast_impl::cast<DecayType>(str) ;
}
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
31.01.2016, 15:29     Обертка над boost::lexical_cast для работы с unsigned типами
Посмотрите здесь:

Boost C++ Обертка над boost::asio
C++ Операции над множествами - переписать код для работы с буквами
C++ Предусмотреть использование шаблонного класса для работы с различными типами данных
C++ обертка для dll C C++
Unary minus operator applied to unsigned type, result still unsigned C++
C++ Builder Обертка для класса TRegistry не работает как положено
Lexical_cast и десятичный разделитель Boost C++
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Avazart
Нарушитель
6954 / 5192 / 256
Регистрация: 10.12.2010
Сообщений: 22,770
Записей в блоге: 17
02.02.2016, 11:13     Обертка над boost::lexical_cast для работы с unsigned типами #21
Цитата Сообщение от Kastaneda Посмотреть сообщение
??? От языка пользователя зависит формат даты, времени, плавающая точка и т.д.
Абсолютно не зависит, пользователь может их поменять
Миниатюры
Обертка над boost::lexical_cast для работы с unsigned типами  
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Kastaneda
Модератор
Эксперт С++
 Аватар для Kastaneda
4248 / 2780 / 219
Регистрация: 12.12.2009
Сообщений: 7,109
Записей в блоге: 1
Завершенные тесты: 1
02.02.2016, 11:15     Обертка над boost::lexical_cast для работы с unsigned типами #22
Цитата Сообщение от Avazart Посмотреть сообщение
Абсолютно не зависит, пользователь может их поменять
Мы с разных ракурсов на задачу смотрим. У меня пользователь не то, что поменять не может, он даже свой язык не может указать (при первом посещении).
Avazart
Нарушитель
6954 / 5192 / 256
Регистрация: 10.12.2010
Сообщений: 22,770
Записей в блоге: 17
02.02.2016, 11:31     Обертка над boost::lexical_cast для работы с unsigned типами #23
Цитата Сообщение от Kastaneda Посмотреть сообщение
У меня веб-приложение, юзеру нужно показать дату в его формате, бразуер в HTTP запросе посылает язык пользователя, по языку определяется в каком виде ему показывать дату. Если всем показывать в одном формате, то русский и американец поймет эту запись 3/2/2016 совершенно по-разному.
Ну у меня была такая проблема, когда принимаешь данные с сайта в JSON формате, там "." а у меня стоит ","

Добавлено через 15 минут
Цитата Сообщение от Kastaneda Посмотреть сообщение
У меня пользователь не то, что поменять не может, он даже свой язык не может указать (при первом посещении).
Мм а если переносить программу на другой комп, распространять? Кроме того есть главный пользователь, необходимость переустановки системы итп.

Кстати в Qt тоже нет возможности устанавливать, но там по крайней мере можно создавать объект локали.
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
02.02.2016, 11:37     Обертка над boost::lexical_cast для работы с unsigned типами
Еще ссылки по теме:

Namespace boost не содержит члена lexical_cast C++
Как настроить проект Visual C++ 2010 для работы с Boost C++? Visual C++
Обертка над ждущими таймерами: ошибка приведения типа C++ WinAPI
Обертка для list C++
Шаблонный класс для работы с разными типами данных и объявление методов в main C++

Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
Kastaneda
Модератор
Эксперт С++
 Аватар для Kastaneda
4248 / 2780 / 219
Регистрация: 12.12.2009
Сообщений: 7,109
Записей в блоге: 1
Завершенные тесты: 1
02.02.2016, 11:37     Обертка над boost::lexical_cast для работы с unsigned типами #24
Цитата Сообщение от Avazart Посмотреть сообщение
Мм а если переносить программу на другой комп, распространять?
веб-сервис, все крутится на линуксе, юзер через браузер с программой общается.
Yandex
Объявления
02.02.2016, 11:37     Обертка над boost::lexical_cast для работы с unsigned типами
Ответ Создать тему

Метки
boost, c++, string, преобразование, шаблоны
Опции темы

Текущее время: 18:18. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru