Добро пожаловать на Pro Pawn - Портал о PAWN-скриптинге.
Страница 2 из 2 ПерваяПервая 1 2
Показано с 11 по 16 из 16
  1. #11
    Аватар для Daniel_Cortez
    "Это не хак, это фича"

    Статус
    Оффлайн
    Регистрация
    06.04.2013
    Адрес
    Novokuznetsk, Russia
    Сообщений
    2,192
    Репутация:
    2590 ±
    Цитата Сообщение от m1n1vv Посмотреть сообщение
    Не обращал на это внимание. Учту.
    Справедливости ради, стоит также отметить, что и функция strins игнорирует передаваемый ей размер массива и может записать лишние символы за его пределы. Наверное, именно это vvw и имел в виду изначально.

    Цитата Сообщение от Геннадий Литвинов Посмотреть сообщение
    А точки или запятые можно ли так поставить
    Да, для этого в приведённых в этой теме функциях есть параметр delimiter.



    Немного поэкспериментировал и, кажется, нашёл оптимальный вариант реализации функции:
    1. stock bool:FormatIntWithDelimiters(value, outstr[], delimiter = '.', size = sizeof(outstr))
    2. {
    3. static delimiter_string[2] = { '\0', '\0' };
    4. static len;
    5. delimiter_string[0] = delimiter;
    6. if (value != cellmin)
    7. format(outstr, size, "%d", value);
    8. else
    9. outstr[0] = '\0', strcat(outstr, "-2147483648", size);
    10. len = strlen(outstr);
    11. switch (len - _:(value < 0))
    12. {
    13. case 10:
    14. {
    15. if (len + 3 >= size)
    16. return false;
    17. strins(outstr, delimiter_string, len - 3, 1);
    18. strins(outstr, delimiter_string, len - 6, 1);
    19. strins(outstr, delimiter_string, len - 9, 1);
    20. }
    21. case 7..9:
    22. {
    23. if (len + 2 >= size)
    24. return false;
    25. strins(outstr, delimiter_string, len - 3, 1);
    26. strins(outstr, delimiter_string, len - 6, 1);
    27. }
    28. case 4..6:
    29. {
    30. if (len + 1 >= size)
    31. return false;
    32. strins(outstr, delimiter_string, len - 3, 1);
    33. }
    34. }
    35. return true;
    36. }

    Отличается от других представленных в этой теме вариантов тем, что не подвержена багу с возвратом строки (сформатированная строка возвращается косвенно) и не производит лишних манипуляций с отдельной вставкой знака "-" и возвратом строки через стек.
    Индивидуально в ЛС по скриптингу не помогаю. Задавайте все свои вопросы здесь (click).

  2. Пользователь сказал cпасибо:
    vvw (29.08.2019)
  3. #12
    Аватар для vvw
    Пользователь

    Статус
    Оффлайн
    Регистрация
    09.08.2019
    Сообщений
    45
    Репутация:
    9 ±
    Цитата Сообщение от m1n1vv Посмотреть сообщение
    Приведи пример
    Только сейчас заметил, что space, он же delimiter, имеет неявный размер, что также может привести к ошибке OOB.

    - - - Добавлено - - -

    Цитата Сообщение от Daniel_Cortez Посмотреть сообщение
    Вообще, ИМХО, вариант DeimoS'а выглядит почти идеальным, не считая перестановки всех элементов массива при вставке "-" (наверняка можно реализовать это как-то получше) и ещё пары уже упомянутых нюансов. Также настораживает в том варианте повторяющийся код внутри switch, но это такая особенность Pawn, что нельзя сделать, чтобы из одного case нельзя было "провалиться" в следующий, как в C/C++. Впрочем, это не ещё значит, что такую возможность нельзя добавить в компилятор :)
    В последнее время очень много добавляют в компилятор. Это, с одной стороны, хорошо, а с другой стороны - уже другой язык без документации. Хотя сделать fallthrough (__fallthrough) нужно.

  4. #13
    Аватар для tnc
    Пользователь

    Статус
    Оффлайн
    Регистрация
    01.09.2019
    Сообщений
    121
    Репутация:
    26 ±
    Если подключен Pawn.Regex, то можно паттерном это решить (это просто альтернативное решение)

  5. #14
    Аватар для Kovshevoy
    Пользователь

    Статус
    Оффлайн
    Регистрация
    11.07.2015
    Сообщений
    190
    Репутация:
    25 ±
    Вставлю и свои 5 копеек.

    1. FormatNumber(number, prefix[] = "$")
    2. {
    3. new value[15],
    4. length;
    5.  
    6. format(value, sizeof(value), "%d", (number < 0) ? (-number) : (number));
    7.  
    8. if ((length = strlen(value)) > 3)
    9. {
    10. for (new i = length, l = 0; --i >= 0; l ++)
    11. {
    12. if ((l > 0) && (l % 3 == 0)) strins(value, ",", i + 1);
    13. }
    14. }
    15. if (prefix[0] != 0)
    16. strins(value, prefix, 0);
    17.  
    18. if (number < 0)
    19. strins(value, "-", 0);
    20.  
    21. return value;
    22. }


    Выглядит вот так:
      Открыть/закрыть

    Если не работает скриншот под спойлером - Клик-клик-клик

    Команда для теста (зависимость от sscanf, zcmd/DC_CMD/Pawn.CMD/izcmd и прочих обработчиков синтаксиса zcmd):
    1. CMD:test(playerid, params[])
    2. {
    3. new const fmt_str[] = "{FF9900}[debug]{F6F6F6}: %s";
    4. new arr_sizeof[sizeof fmt_str + (-2 + 15)];
    5.  
    6. extract params -> new value; else
    7. return SendClientMessage(playerid, -1, "/test [value]");
    8.  
    9. format(arr_sizeof, sizeof(arr_sizeof), fmt_str, FormatNumber(value));
    10. SendClientMessage(playerid, -1, arr_sizeof);
    11. return 1;
    12. }
    Последний раз редактировалось Kovshevoy; 15.12.2019 в 01:13.

  6. #15
    Аватар для DeimoS
    Модератор?

    Статус
    Оффлайн
    Регистрация
    27.01.2014
    Адрес
    Восточный Мордор
    Сообщений
    5,588
    Репутация:
    1984 ±
    Цитата Сообщение от Kovshevoy Посмотреть сообщение
    Вставлю и свои 5 копеек.

    1. FormatNumber(number, prefix[] = "$")
    2. {
    3. new value[15],
    4. length;
    5.  
    6. format(value, sizeof(value), "%d", (number < 0) ? (-number) : (number));
    7.  
    8. if ((length = strlen(value)) > 3)
    9. {
    10. for (new i = length, l = 0; --i >= 0; l ++)
    11. {
    12. if ((l > 0) && (l % 3 == 0)) strins(value, ",", i + 1);
    13. }
    14. }
    15. if (prefix[0] != 0)
    16. strins(value, prefix, 0);
    17.  
    18. if (number < 0)
    19. strins(value, "-", 0);
    20.  
    21. return value;
    22. }


    Выглядит вот так:
      Открыть/закрыть

    Если не работает скриншот под спойлером - Клик-клик-клик

    Команда для теста (зависимость от sscanf, zcmd/DC_CMD/Pawn.CMD/izcmd и прочих обработчиков синтаксиса zcmd):
    1. CMD:test(playerid, params[])
    2. {
    3. new const fmt_str[] = "{FF9900}[debug]{F6F6F6}: %s";
    4. new arr_sizeof[sizeof fmt_str + (-2 + 15)];
    5.  
    6. extract params -> new value; else
    7. return SendClientMessage(playerid, -1, "/test [value]");
    8.  
    9. format(arr_sizeof, sizeof(arr_sizeof), fmt_str, FormatNumber(value));
    10. SendClientMessage(playerid, -1, arr_sizeof);
    11. return 1;
    12. }
    Эмм, и смысл перебирать каждый символ, при этом, тратя время на деление, если можно сразу узнать количество символов в числе и расставить разделители? Собственно, как в моём варианте функции сделано.
    Связаться со мной в VK можно через личные сообщения этой группы
    Заказы не принимаю

    Широко известно, что идеи стоят 0.8333 цента каждая (исходя из рыночной цены 10 центов за дюжину).
    Великих идей полно, на них нет спроса.
    Воплощение идеи в законченную игру требует долгой работы,
    таланта, терпения и креативности, не говоря уж о затратах денег, времени и ресурсов.
    Предложить идею просто, воплотить – вот в чём проблема

    Steve Pavlina

  7. #16
    Аватар для $continue$
    Пользователь

    Статус
    Оффлайн
    Регистрация
    02.08.2014
    Адрес
    г. Киров (aka Вятка)
    Сообщений
    1,487
    Репутация:
    276 ±
    Цитата Сообщение от Daniel_Cortez Посмотреть сообщение
    Справедливости ради, стоит также отметить, что и функция strins игнорирует передаваемый ей размер массива и может записать лишние символы за его пределы. Наверное, именно это vvw и имел в виду изначально.


    Да, для этого в приведённых в этой теме функциях есть параметр delimiter.



    Немного поэкспериментировал и, кажется, нашёл оптимальный вариант реализации функции:
    1. stock bool:FormatIntWithDelimiters(value, outstr[], delimiter = '.', size = sizeof(outstr))
    2. {
    3. static delimiter_string[2] = { '\0', '\0' };
    4. static len;
    5. delimiter_string[0] = delimiter;
    6. if (value != cellmin)
    7. format(outstr, size, "%d", value);
    8. else
    9. outstr[0] = '\0', strcat(outstr, "-2147483648", size);
    10. len = strlen(outstr);
    11. switch (len - _:(value < 0))
    12. {
    13. case 10:
    14. {
    15. if (len + 3 >= size)
    16. return false;
    17. strins(outstr, delimiter_string, len - 3, 1);
    18. strins(outstr, delimiter_string, len - 6, 1);
    19. strins(outstr, delimiter_string, len - 9, 1);
    20. }
    21. case 7..9:
    22. {
    23. if (len + 2 >= size)
    24. return false;
    25. strins(outstr, delimiter_string, len - 3, 1);
    26. strins(outstr, delimiter_string, len - 6, 1);
    27. }
    28. case 4..6:
    29. {
    30. if (len + 1 >= size)
    31. return false;
    32. strins(outstr, delimiter_string, len - 3, 1);
    33. }
    34. }
    35. return true;
    36. }

    Отличается от других представленных в этой теме вариантов тем, что не подвержена багу с возвратом строки (сформатированная строка возвращается косвенно) и не производит лишних манипуляций с отдельной вставкой знака "-" и возвратом строки через стек.
    С fixes вызывает выходы за пределы массива. (Не знаю почему, не было времени разобраться. Скорее всего что-то связанное с фиксом размера maxlength, но я не уверен. Нужно дебагать). Банальный фикс выглядит примерно так:

    1. #include <a_samp>
    2. #include <fixes>
    3.  
    4. forward bool:FormatIntWithDelimiters(value, outstr[], delimiter = '.', size = sizeof(outstr));
    5.  
    6. main()
    7. {
    8. static const nums[] =
    9. {
    10. 500,
    11. 1_000,
    12. 10_000,
    13. 100_000,
    14. 1_000_000,
    15. cellmax
    16. };
    17.  
    18. new str_num[64];
    19.  
    20. for (new i = 0; i < sizeof(nums); i++)
    21. {
    22. FormatIntWithDelimiters(nums[i], str_num);
    23. printf("str_num: %s", str_num);
    24. str_num[0] = '\0';
    25. }
    26. }
    27.  
    28. stock bool:FormatIntWithDelimiters(value, outstr[], delimiter = '.', size = sizeof(outstr))
    29. {
    30. static delimiter_string[2] = { '\0', '\0' };
    31. static len;
    32. delimiter_string[0] = delimiter;
    33. if (value != cellmin)
    34. format(outstr, size, "%d", value);
    35. else
    36. outstr[0] = '\0', strcat(outstr, "-2147483648", size);
    37. len = strlen(outstr);
    38. switch (len - _:(value < 0))
    39. {
    40. case 10:
    41. {
    42. if (len + 3 >= size)
    43. return false;
    44.  
    45. format(outstr, size, "%.*s%s%s", len - 3, outstr, delimiter_string, outstr[len - 3]);
    46. format(outstr, size, "%.*s%s%s", len - 6, outstr, delimiter_string, outstr[len - 6]);
    47. format(outstr, size, "%.*s%s%s", len - 9, outstr, delimiter_string, outstr[len - 9]);
    48. }
    49. case 7..9:
    50. {
    51. if (len + 2 >= size)
    52. return false;
    53.  
    54. format(outstr, size, "%.*s%s%s", len - 3, outstr, delimiter_string, outstr[len - 3]);
    55. format(outstr, size, "%.*s%s%s", len - 6, outstr, delimiter_string, outstr[len - 6]);
    56. }
    57. case 4..6:
    58. {
    59. if (len + 1 >= size)
    60. return false;
    61.  
    62. format(outstr, size, "%.*s%s%s", len - 3, outstr, delimiter_string, outstr[len - 3]);
    63. }
    64. }
    65. return true;
    66. }


    P.S: Не знаю насколько оправдан тут format, но мне нужен был быстрый фикс
    Value your freedom or you will lose it, teaches history. "Don't bother us with politics," respond those who don't want to learn. (c) Richard Stallman

 

 
Страница 2 из 2 ПерваяПервая 1 2

Информация о теме

Пользователи, просматривающие эту тему

Эту тему просматривают: 1 (пользователей: 0 , гостей: 1)

Ваши права

  • Вы не можете создавать новые темы
  • Вы не можете отвечать в темах
  • Вы не можете прикреплять вложения
  • Вы не можете редактировать свои сообщения
  •