|
|
  |
Выдать из функции указатель на ее статическую переменную |
|
|
|
Dec 5 2014, 12:33
|

Универсальный солдатик
     
Группа: Модераторы
Сообщений: 8 634
Регистрация: 1-11-05
Из: Минск
Пользователь №: 10 362

|
Цитата(SM @ Dec 5 2014, 15:18)  Так тогда strtol вам и нужна в чистом ее виде... У этой функции 3 параметра. Указатель, ползущий по строке, у меня в глобальной структуре имеется, его передавать не нужно. Если попалась строка "-300голов", то мне логично промотать его до space-символов, а не остановиться на "г". Не буду нарушать созданную логику, сделаю, как хотел. Правильно, указатель на локальную статическую переменную выдавать и использовать снаружи функции можно?
|
|
|
|
|
Dec 5 2014, 12:34
|
Гуру
     
Группа: Свой
Сообщений: 7 946
Регистрация: 25-02-05
Из: Moscow, Russia
Пользователь №: 2 881

|
Цитата(AlexandrY @ Dec 5 2014, 15:30)  strtol опасна в плане RTOS, функции этого ряда могут внутри себя лезть в глобальный heap и в глобальные переменные. Это с какого перепуга то? Только errno, но это классика. CODE _CODE_ACCESS long strtol(const char *st, char **endptr, int base) { register unsigned long result = 0; register unsigned int uns_base; register char cp; register const char *fst = st; int digits = 0; unsigned long sign = 0; unsigned long max_prev_result; unsigned long max_curr_addval; unsigned long range_limit;
while (_isspace(*fst)) ++fst; /* SKIP WHITE SPACE */
if ((cp = *fst) == '-') { sign = 1; cp = *++fst; } else if (cp == '+') cp = *++fst;
if (base > 36) base = 0;
/*-------------------------------------------------------------------*/ /* DETERMINE BASE IF ZERO BASE ASKED FOR. */ /*-------------------------------------------------------------------*/ switch (base) { case 0 : if (cp != '0') base = 10; else if ((cp = *++fst) == 'x' || cp == 'X') if (_isxdigit(fst[1])) { base = 16; cp = *++fst; } else { base = 10; cp = *--fst; } else { base = 8; if (cp < '0' || cp > '7') cp = *--fst; } break;
case 16 : if (cp == '0' && ((fst[1] == 'x') || (fst[1] == 'X')) && _isxdigit(fst[2])) cp = *(fst += 2); }
/*-------------------------------------------------------------------*/ /* DETERMINE VALUES NEEDED TO DETECT RANGE ERROR. THE MAX UNSIGNED */ /* VALUE THAT IS IN RANGE IS BASED ON THE SIGN OF THE INPUT STRING: */ /* sign == 1 - RESULT WILL BE NEGATIVE, SO MUST BE <= LONG_MAX+1 */ /* sign == 0 - RESULT WILL BE POSITIVE, SO MUST BE <= LONG_MAX */ /* SO range_limit is LONG_MAX + sign. */ /* */ /* ALSO NEED max_prev_result AND max_curr_addval WHERE: */ /* max_prev_result * uns_base + max_curr_addval == range_limit */ /* */ /* THIS ALLOWS US TO TEST FOR RANGE ERROR BEFORE WE COMPUTE THE NEXT */ /* RESULT. WE HAVE TWO CASES: */ /* - PREVIOUS result IS GREATER THAN max_prev_result, SO NEXT */ /* RESULT MUST BE OUT OF RANGE. */ /* - PREVIOUS result IS EQUAL TO max_prev_result, SO NEXT RESULT */ /* IS OUT OF RANGE IF addval > max_curr_addval. */ /*-------------------------------------------------------------------*/ uns_base = (unsigned int) base; range_limit = (unsigned long)LONG_MAX + sign; max_prev_result = range_limit / uns_base; max_curr_addval = range_limit % uns_base;
/*-------------------------------------------------------------------*/ /* CONVERT THE NUMBER USING THE SPECIFIED BASE. */ /*-------------------------------------------------------------------*/ for (;; cp = *++fst) { register unsigned long addval;
if (!((_isdigit(cp) && (addval = cp - '0') < uns_base) || (_isupper(cp) && (addval = cp - 'A' + 10) < uns_base) || (_islower(cp) && (addval = cp - 'a' + 10) < uns_base))) break;
/*--------------------------------------------------------------*/ /* CHECK OVERFLOW STATUS BEFORE COMPUTATION. */ /* result CONTAINS THE RESULT OF THE PREVIOUS COMPUTATION */ /*--------------------------------------------------------------*/ if (result > max_prev_result || (result == max_prev_result && addval > max_curr_addval)) { /* SET ERRNO */ errno = ERANGE; if (endptr) *endptr = (char *)st; return sign ? LONG_MIN : LONG_MAX; }
/*--------------------------------------------------------------*/ /* COMPUTE NEXT RESULT. */ /*--------------------------------------------------------------*/ result = result * uns_base + addval;
digits++; }
/*-------------------------------------------------------------------*/ /* MARK THE END OF THE CONVERTED INPUT. NEGATE THE RESULT IF A */ /* MINUS SIGN WAS SEEN. */ /*-------------------------------------------------------------------*/ if (endptr) *endptr = (char *)(digits ? fst : st); if (sign) return (long)-result;
return (long)result; }
|
|
|
|
|
Dec 5 2014, 15:23
|

Adept
     
Группа: Свой
Сообщений: 3 469
Регистрация: 6-12-04
Из: Novosibirsk
Пользователь №: 1 343

|
Возвращать результат из функции, храня его во внутреннем статическом объекте можно, но есть серьёзные недостатки:
1. Функция получается нереентерабельной. 2. Результат может быть не когерентен вызову даже в момент возврата из функции (опять же из-за возможности асинхронного вызова функции). 3. В случае возвращаемого целого обычный возврат по значению будет, скорее всего, эффективней. 4. Есть рабочий конкурент - strtol, про него уже сказали, ни разу не сложная функция, всяко проще, чем самому огороды городить. Не нравится она в таком виде, написать обёртку.
--------------------
«Отыщи всему начало, и ты многое поймёшь» К. Прутков
|
|
|
|
|
Dec 5 2014, 19:16
|
Местный
  
Группа: Участник
Сообщений: 202
Регистрация: 10-04-05
Из: Санкт-Петербург
Пользователь №: 4 011

|
Указатель на переменную успеха/неудачи передавать в качестве параметра. Объявление: Код double toDouble(const char * str, bool * ok); Применение: Код const char * str = "0.252"; bool ok; double var = toDouble(str, &ok);
|
|
|
|
|
  |
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0
|
|
|