6

In Python, scientific notation always gives me 2 digits in exponent:

print('%17.8E\n' % 0.0665745511651039)
6.65745512E-02

However, I badly want to have 3 digits like:

6.65745512E-002

Can we do this with a built-in configuration/function in Python?

I know my question is basically the same question as: Python - number of digits in exponent, but this question was asked 4 years ago and I don't want to call such a function thousand times. I hope there should be a better solution now.

Community
  • 1
  • 1
IanHacker
  • 541
  • 9
  • 27
  • 1
    Possible duplicate of [Python - number of digits in exponent](http://stackoverflow.com/questions/9910972/python-number-of-digits-in-exponent) – vsminkov Aug 27 '16 at 19:15

2 Answers2

4

Unfortunately, you can not change this default behavior since you can not override the str methods.

However, you can wrap the float, and use the __format__ method:

class MyNumber:
    def __init__(self, val):
        self.val = val

    def __format__(self,format_spec):
        ss = ('{0:'+format_spec+'}').format(self.val)
        if ( 'E' in ss):
            mantissa, exp = ss.split('E')            
            return mantissa + 'E'+ exp[0] + '0' + exp[1:]
        return ss


     print( '{0:17.8E}'.format( MyNumber(0.0665745511651039)))
napuzba
  • 6,033
  • 3
  • 21
  • 32
  • Yes, the wrapping-the-float method perfectly works for print(). Thank you. However, I would also like to write the output to a file and it works only half way: 'ValueError: invalid literal for float(): -4.10522747+00' I need your help. Let me edit my original question to explain this error. – IanHacker Aug 28 '16 at 04:53
  • When the number is between 1 <= n < 10, the format doesn't work and simply shows 2 digits. ` -6.08474553E-001 -4.10522747+00 -4.75716448+00 -1.58045329E-002 4.72555542+00` (It didn't work for print(), too.) Could you please fix this? – IanHacker Aug 28 '16 at 05:52
  • Updated for positive exponent – napuzba Aug 28 '16 at 06:26
  • It works like a charm. Now it shows all in 3 digits. ` -6.08474553E-001 -4.10522747E+000 -4.75716448E+000 -1.58045329E-002 4.72555542E+000` Thank you so much! – IanHacker Aug 28 '16 at 08:36
0

You can use your own formatter and override format_field:

import string
class MyFormatter(string.Formatter): 
    def format_field(self, value, format_spec):
        ss = string.Formatter.format_field(self,value,format_spec)
        if format_spec.endswith('E'):
            if ( 'E' in ss):
                mantissa, exp = ss.split('E')
                return mantissa + 'E'+ exp[0] + '0' + exp[1:]                  
        return ss

print( MyFormatter().format('{0:17.8E}',0.00665745511651039) )
napuzba
  • 6,033
  • 3
  • 21
  • 32