Відобразити десятковий знак у наукових позначеннях


160

Як я можу відобразити це:

Десяткові ("40800000000.00000000000000") як "4.08E + 10"?

Я спробував це:

>>> '%E' % Decimal('40800000000.00000000000000')
'4.080000E+10'

Але це додаткові 0.


3
якесь подвійне розміщення, ви могли використовувати цю тему, яку ви тільки почали: stackoverflow.com/questions/6913166/…
Samuele Mattiuzzo

13
ні, не зовсім. Я хотів розділити це на просте запитання (як це зробити в Python) і важке, незрозуміле питання, на яке я сумніваюся, хтось відповість (як це зробити в Django). Зверніть увагу, як на це вже є відповідь. Я зараз на півдорозі до остаточної відповіді замість 0%, якби я розмістив їх разом. Крім того, розділення питань полегшує пошук людей відповідей. Е., г, якщо Боб шукає запитання щодо форматування десяткової, він може пропустити квест SO з Django в заголовку.
Грег

так, це було просто для мого інтересу: P простіше прослідкувати одну нитку. в основному це схоже на мою відповідь (просто "трохи" конкретніше). Я також сподіваюсь на відповідь джанго, btw.
Самуеле Маттіуццо

Відповіді:


157
from decimal import Decimal

'%.2E' % Decimal('40800000000.00000000000000')

# returns '4.08E+10'

У вашому '40800000000.00000000000000' є багато більш значущих нулів, які мають те саме значення, що і будь-яка інша цифра. Ось чому ви повинні чітко сказати, де ви хочете зупинитися.

Якщо ви хочете видалити всі нулі автоматично, ви можете спробувати:

def format_e(n):
    a = '%E' % n
    return a.split('E')[0].rstrip('0').rstrip('.') + 'E' + a.split('E')[1]

format_e(Decimal('40800000000.00000000000000'))
# '4.08E+10'

format_e(Decimal('40000000000.00000000000000'))
# '4E+10'

format_e(Decimal('40812300000.00000000000000'))
# '4.08123E+10'

22
На відміну, незважаючи на те, що format % valuesсинтаксис все ще використовується навіть у стандартній бібліотеці Python 3, я вважаю, що технічно застаріла в Python 3, або, принаймні, не рекомендований метод форматування, і поточний рекомендований синтаксис, починаючи з Python 2.6, буде '{0:.2E}'.format(Decimal('40800000000.00000000000000'))( або '{:.2E}'в Python 2.7+). Незважаючи на те, що в цій ситуації не є дуже корисним, завдяки додатковим символам без додаткової функціональності, str.formatце дозволяє більш складне змішування / перестановка / повторне використання аргументів формату.
JAB

що з python 3?
Чарлі Паркер

4
@CharlieParker Використання format. Це більше джазі .
Mateen Ulhaq

120

Ось приклад використання format()функції:

>>> "{:.2E}".format(Decimal('40800000000.00000000000000'))
'4.08E+10'

Замість формату ви також можете використовувати f-рядки :

>>> f"{Decimal('40800000000.00000000000000'):.2E}"
'4.08E+10'

3
Цей синтаксис також стосується f-рядків у 3.6+f"{Decimal('40800000000.00000000000000'):.2E}"
Tritium21

37

З огляду на ваш номер

x = Decimal('40800000000.00000000000000')

Починаючи з Python 3,

'{:.2e}'.format(x)

- це рекомендований спосіб зробити це.

eозначає, що ви хочете наукову нотацію, а .2значить, вам потрібно дві цифри після крапки. Так ви отримаєтеx.xxE±n


1
Сенс використання десяткових значень - отримати точну та довільну точність арифметичної десяткової точності. Це не рівнозначно використанню поплавця.
asmeurer

@asmeurer Дякую за пояснення Змінив мою відповідь.
patapouf_ai

Чи є спосіб повернутися з цього плавати?
olenscki

@olenscki тільки що float(x)перетворить x у float.
patapouf_ai

33

Ніхто не згадав про коротку форму .formatметоду:

Потрібно принаймні Python 3.6

f"{Decimal('40800000000.00000000000000'):.2E}"

(Я вважаю, це те саме, що Сез Тіммерман, трохи коротший)


3
Слід прийняти відповідь. f-string - це майбутнє форматування рядків python :)
Гандальф Сакс

1
Як фій для майбутніх читачів, як я: якщо вам не байдуже контролювати кількість цифр і не заперечуєте помилок з плаваючою комою, ви можете просто використовувати {num:E}, де, наприклад, число = 40800000000.00000000000000
Шаяан,


4

Мої десяткові знаки занадто великі, %Eщоб мені довелося імпровізувати:

def format_decimal(x, prec=2):
    tup = x.as_tuple()
    digits = list(tup.digits[:prec + 1])
    sign = '-' if tup.sign else ''
    dec = ''.join(str(i) for i in digits[1:])
    exp = x.adjusted()
    return '{sign}{int}.{dec}e{exp}'.format(sign=sign, int=digits[0], dec=dec, exp=exp)

Ось приклад використання:

>>> n = decimal.Decimal(4.3) ** 12314
>>> print format_decimal(n)
3.39e7800
>>> print '%e' % n
inf

3
Просто "{:.2e}".format(n)повертається '3.39e+7800'в Python 3.3.2 (v3.3.2: d047928ae3f6, 16 травня 2013, 00:06:53) [MSC v.1600 64 біт (AMD64)] у win32.
Cees Timmerman

4

Це найкраще працювало для мене:

import decimal
'%.2E' % decimal.Decimal('40800000000.00000000000000')
# 4.08E+10

4

Це зведений список "Простих" відповідей та коментарів.

ПІТОН 3

from decimal import Decimal

x = '40800000000.00000000000000'
# Converted to Float
x = Decimal(x)

# ===================================== # `Dot Format`
print("{0:.2E}".format(x))
# ===================================== # `%` Format
print("%.2E" % x)
# ===================================== # `f` Format
print(f"{x:.2E}")
# =====================================
# ALL Return: 4.08E+10
print((f"{x:.2E}") == ("%.2E" % x) == ("{0:.2E}".format(x)))
# True
print(type(f"{x:.2E}") == type("%.2E" % x) == type("{0:.2E}".format(x)))
# True
# =====================================

АБО Без IMPORT's

# NO IMPORT NEEDED FOR BASIC FLOATS
y = '40800000000.00000000000000'
y = float(y)

# ===================================== # `Dot Format`
print("{0:.2E}".format(y))
# ===================================== # `%` Format
print("%.2E" % y)
# ===================================== # `f` Format
print(f"{y:.2E}")
# =====================================
# ALL Return: 4.08E+10
print((f"{y:.2E}") == ("%.2E" % y) == ("{0:.2E}".format(y)))
# True
print(type(f"{y:.2E}") == type("%.2E" % y) == type("{0:.2E}".format(y)))
# True
# =====================================

Порівнюючи

# =====================================
x
# Decimal('40800000000.00000000000000')
y
# 40800000000.0

type(x)
# <class 'decimal.Decimal'>
type(y)
# <class 'float'>

x == y
# True
type(x) == type(y)
# False

x
# Decimal('40800000000.00000000000000')
y
# 40800000000.0

Тож для Python 3 ви можете зараз перемикатися між будь-яким із трьох.

Мій улюблений:

print("{0:.2E}".format(y))

3

Я віддаю перевагу Python 3.x спосіб.

cal = 123.4567
print(f"result {cal:.4E}")

4 вказує, скільки цифр показано на плаваючій частині.

cal = 123.4567
totalDigitInFloatingPArt = 4
print(f"result {cal:.{totalDigitInFloatingPArt}E} ")

2

Для перетворення десяткової в наукову нотацію без необхідності вказувати точність у рядку формату та без включення кінцевих нулів, я зараз використовую

def sci_str(dec):
    return ('{:.' + str(len(dec.normalize().as_tuple().digits) - 1) + 'E}').format(dec)

print( sci_str( Decimal('123.456000') ) )    # 1.23456E+2

Щоб зберегти будь-які задні нулі, просто видаліть normalize().


1

Ось найпростіший, який я міг знайти.

format(40800000000.00000000000000, '.2E')
#'4.08E+10'

("E" не відрізняється від регістру. Ви також можете використовувати ".2e")


0
def formatE_decimal(x, prec=2):
    """ Examples:
    >>> formatE_decimal('0.1613965',10)
    '1.6139650000E-01'
    >>> formatE_decimal('0.1613965',5)
    '1.61397E-01'
    >>> formatE_decimal('0.9995',2)
    '1.00E+00'
    """
    xx=decimal.Decimal(x) if type(x)==type("") else x 
    tup = xx.as_tuple()
    xx=xx.quantize( decimal.Decimal("1E{0}".format(len(tup[1])+tup[2]-prec-1)), decimal.ROUND_HALF_UP )
    tup = xx.as_tuple()
    exp = xx.adjusted()
    sign = '-' if tup.sign else ''
    dec = ''.join(str(i) for i in tup[1][1:prec+1])   
    if prec>0:
        return '{sign}{int}.{dec}E{exp:+03d}'.format(sign=sign, int=tup[1][0], dec=dec, exp=exp)
    elif prec==0:
        return '{sign}{int}E{exp:+03d}'.format(sign=sign, int=tup[1][0], exp=exp)
    else:
        return None
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.