Python: Числа с плавающей точкой
В математике есть разные типы чисел. Например:
- Натуральные: целые положительные числа: 1, 2, 3 и т.д.
- Рациональные: дробные числа, которые можно представить в виде деления, например: 0.5, 1.75, 3.14.
С точки зрения математики все просто. Но с точки зрения компьютера между этими типами чисел лежит настоящая пропасть. Попробуйте мысленно решить: Сколько будет 0.2 + 0.1? Очевидно, 0.3. А вот что скажет Python:
print(0.2 + 0.1) # => 0.30000000000000004Вместо привычных 0.3 мы получаем 0.30000000000000004.
Ожидание: 0.1 + 0.2 → 0.3
Реальность: 0.1 + 0.2 → 0.30000000000000004
└── погрешность храненияПочему так происходит?
Такое поведение характерно для JavaScript, C++ и большинства других языков программирования.
Причина в устройстве компьютера. Компьютер работает с ограниченной памятью, тогда как рациональные числа бесконечно точны. Между 0.1 и 0.2 можно поместить бесконечно много других чисел. Но компьютер не может хранить бесконечность. Он приближает число, стараясь уместить его в доступное количество бит.
Такие приближенные значения называются числами с плавающей точкой (floating point numbers). Хранение и вычисления с ними подчиняются строгим правилам, описанным в специальном стандарте IEEE 754, на который ориентируются большинство языков программирования.
Когда появляются такие числа
Числа с плавающей точкой появляются в программах чаще, чем может показаться. Вот основные случаи:
- Когда вы явно пишете дробное число, например 0.1, 2.5, 3.14.
- Когда выполняете деление, даже если делите два целых числа:
print(1 / 2) # => 0.5
print(2 / 3) # => 0.6666666666666666Даже если результат кажется "красивым", внутри он все равно представлен в виде приближенного значения. Некоторые дроби, такие как 1 / 3, вообще не могут быть точно представлены в двоичной системе, поэтому их точность всегда ограничена.
Где это критично и как с этим работают
Обычно небольшая погрешность не мешает. Но в финансовых расчетах, научных и инженерных задачах, а также при точном сравнении результатов она может стать проблемой. Например, ошибка в доли копейки способна дать неправильную итоговую сумму, а длинная цепочка вычислений может постепенно накапливать неточность.
В реальных программах с этим работают по-разному. Деньги часто хранят в минимальных единицах, например в копейках, то есть используют целые числа вместо дробных. В других случаях результат округляют до нужного количества знаков, сравнивают числа с допустимой погрешностью или используют специальные типы данных и библиотеки для точных вычислений.
Что нужно запомнить
Операции с числами с плавающей точкой не всегда точны, и это нормально. Такое поведение характерно для большинства языков программирования и объясняется устройством компьютерной памяти. Точность можно контролировать, например с помощью округления или сравнения чисел с заданной погрешностью. А при работе с деньгами, точными измерениями или научными расчетами лучше использовать специальные типы данных, которые обеспечивают контроль над точностью.
Задание
Вы оплачиваете два товара в интернет-магазине: первый стоит 0.1 рубля, второй — 0.2 рубля. Рассчитайте и выведите итоговую сумму.
Ожидаете увидеть 0.3 — но Python выдаст кое-что интересное.
Полезное
Команда проекта находится в телеграм-сообществе. Там можно задать любой вопрос и повлиять на проект
Если вы зашли в тупик, то самое время поговорить с нашим асситентом Тота во вкладке "ИИ-помощник":
Тесты устроены таким образом, что они проверяют решение разными способами и на разных данных. Часто решение работает с одними входными данными, но не работает с другими. Чтобы разобраться с этим моментом, изучите вкладку «Тесты» и внимательно посмотрите на вывод ошибок, в котором есть подсказки.
Это нормально 🙆, в программировании одну задачу можно выполнить множеством способов. Если ваш код прошел проверку, то он соответствует условиям задачи. В редких случаях бывает, что решение подогнано под тесты, но это видно сразу.
Создавать обучающие материалы, понятные для всех без исключения, довольно сложно. Мы очень стараемся, но всегда есть что улучшать. Если вы встретили материал, который вам непонятен, опишите проблему в обратной связи нашего сообщества
Ваше упражнение проверяется по этим тестам
import runpy
def test(capsys):
expected = "0.30000000000000004"
runpy.run_module('solution')
out, _ = capsys.readouterr()
assert out.strip() == expectedРешение учителя откроется через:
20:00
