std :: lexical_cast - чи є таке?


77

Чи визначає стандартну бібліотеку C ++ цю функцію, чи мені доводиться вдаватися до Boost?

Я шукав в Інтернеті і не міг знайти нічого, крім Boost, але я подумав, що мені краще запитати тут.


6
Ви також можете "вдатися" до струнних потоків :)
Гонки легкості на орбіті

Відповіді:


92

Лише частково.

C ++ 11 <string>має std::to_stringдля вбудованих типів:

[n3290: 21.5/7]:

string to_string(int val);
string to_string(unsigned val);
string to_string(long val);
string to_string(unsigned long val);
string to_string(long long val);
string to_string(unsigned long long val);
string to_string(float val);
string to_string(double val);
string to_string(long double val);

Повертає: Кожна функція повертає stringоб'єкт , що містить представлення символу значення його аргументу , який буде згенеровано шляхом виклику sprintf(buf, fmt, val)з специфікатор форматом з "%d", "%u", "%ld", "%lu", "%lld", "%llu", "%f", "%f", або "%Lf", відповідно, де bufпозначає внутрішній буфер символів достатнього розміру.

Є також такі, що йдуть навпаки:

[n3290: 21.5/1, 21.5/4]:

int stoi(const string& str, size_t *idx = 0, int base = 10);
long stol(const string& str, size_t *idx = 0, int base = 10);
unsigned long stoul(const string& str, size_t *idx = 0, int base = 10);
long long stoll(const string& str, size_t *idx = 0, int base = 10);
unsigned long long stoull(const string& str, size_t *idx = 0, int base = 10);
float stof(const string& str, size_t *idx = 0);
double stod(const string& str, size_t *idx = 0);
long double stold(const string& str, size_t *idx = 0);

Однак у вас немає нічого загального, що ви можете використовувати (принаймні, до TR2 , можливо!), І взагалі нічого в C ++ 03.


2
Хам, насправді є stoiі друзі :)
Matthieu M.

3
@mcheema: stroustrup.com/C++11FAQ.html#delegating-ctor тоді. Або він має на увазі Boost, або він використовує його як довільну абстракцію з метою висловлення думки, або помиляється.
Гонки легкості на орбіті

2
@mcheema: Ми всі можемо бути зацікавлені в поширенні хороших практик стилю кодування, але це не робить їх менш суб’єктивними!
Гонки легкості на орбіті

3
@mcheema Згадуючи залежність Страуструпа від використання жирного курсиву пропорційного шрифту для коду, я не думаю, що колись можна серйозно говорити про стиль його коду. * жарт *
polkovnikov.ph

1
@LightnessRacesinOrbit Я не бачив більше жодного шуму з цього приводу, що перетворюється на Технічну специфікацію. Чи помер він TT
Джонатан Мі

19

Ні, це не так, навіть у C ++ 11, але його пропонують включити до Технічного звіту 2, наступного набору розширень бібліотеки std.


17
Який статус, до речі, у C ++ 17?
einpoklum

Згідно з Вікіпедією, не буде TR2, оскільки були змінені процедури ISO, а лише Технічні специфікації ( en.wikipedia.org/wiki/… ).
Matias Haeussler

11

Немає std :: lexical_cast, але ви завжди можете зробити щось подібне за допомогою stringstreams :

template <typename T>
T lexical_cast(const std::string& str)
{
    T var;
    std::istringstream iss;
    iss.str(str);
    iss >> var;
    // deal with any error bits that may have been set on the stream
    return var;
}

3
спасибі, але це було б навіть повільніше, ніж boost :: lexical_cast
smallB

4
Рішення на основі потоку не буде таким швидким, як щось на зразок snprintf, але у своєму запитанні ви не згадали проблеми з продуктивністю.
luke

12
Ефективність завжди турбує, але це ніколи не є єдиною проблемою. Простота, чіткість, правильність та ремонтопридатність також дуже важливі. Я б не використовував вищезгаданий код в щільному циклі, але це може бути чудово для інших ситуацій. Типовим питанням щодо продуктивності на SO є те, чи ви профайлювали свій код і виявили, що жертвування іншими проблемами варто: :)
luke

8
Якщо ви в кінцевому підсумку скористаєтесь lexical_cast<string>(string)цим, буде повернуто лише перше слово, а не те, що ви передасте. (Ви можете використовувати його в шаблонній функції або щось інше, а не безпосередньо.) Щось, на що слід звернути увагу.
tmandry

2
Цю версію можна значно пришвидшити, зробивши issстатичну
MM


5

Якщо ви не хочете посилення, тоді полегшена бібліотека під назвою fmt реалізує наступне:

// Works with all the C++11 features and AFAIK faster then boost or standard c++11
std::string string_num = fmt::format_int(123456789).str(); // or .c_str()

Більше прикладів з офіційної сторінки .

Доступ до аргументів за позицією:

format("{0}, {1}, {2}", 'a', 'b', 'c');
// Result: "a, b, c"
format("{}, {}, {}", 'a', 'b', 'c');
// Result: "a, b, c"
format("{2}, {1}, {0}", 'a', 'b', 'c');
// Result: "c, b, a"
format("{0}{1}{0}", "abra", "cad");  // arguments' indices can be repeated
// Result: "abracadabra"

Вирівнювання тексту та вказівка ​​ширини:

format("{:<30}", "left aligned");
// Result: "left aligned                  "
format("{:>30}", "right aligned");
// Result: "                 right aligned"
format("{:^30}", "centered");
// Result: "           centered           "
format("{:*^30}", "centered");  // use '*' as a fill char
// Result: "***********centered***********"

Заміна% + f,% -f та% f та вказівка ​​знаку:

format("{:+f}; {:+f}", 3.14, -3.14);  // show it always
// Result: "+3.140000; -3.140000"
format("{: f}; {: f}", 3.14, -3.14);  // show a space for positive numbers
// Result: " 3.140000; -3.140000"
format("{:-f}; {:-f}", 3.14, -3.14);  // show only the minus -- same as '{:f}; {:f}'
// Result: "3.140000; -3.140000"

Заміна% x та% o та перетворення значення на різні бази:

format("int: {0:d};  hex: {0:x};  oct: {0:o}; bin: {0:b}", 42);
// Result: "int: 42;  hex: 2a;  oct: 52; bin: 101010"
// with 0x or 0 or 0b as prefix:
format("int: {0:d};  hex: {0:#x};  oct: {0:#o};  bin: {0:#b}", 42);
// Result: "int: 42;  hex: 0x2a;  oct: 052;  bin: 0b101010"

Мені дуже подобається fmtбібліотека, і мені подобається бачити, як її пропагують. fmtоднак буде виконувати лише перетворення в рядок . (Зараз це робить краще, ніж будь-яка інша бібліотека C ++, яку я бачив.) fmtНе виконує перетворення рядків із функції lexical_cast або std :: stoxyz.
Філ
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.