hachteja
hachteja

Reputation: 101

Scientific notation on each tick in the default font in pyplot

So what I want is to have my pyplot ticks in scientific notation. So each tick looks like 1x10^6 rather than 1 and then the 10^6 on the axis. The only way I've been able to do this so far is by manually setting each ticklabel as r'$1\times10^6$', but this puts it in the mathematical expression font and set_yticklabels wont listen if I try to pass a fontdict.

How would I accomplish this?

Upvotes: 5

Views: 6790

Answers (3)

Jakob
Jakob

Reputation: 20811

I'm not sure if I understand your question correctly, but do you want something like this?

import matplotlib.pyplot as plt
import numpy as np
plt.plot(np.logspace(1,10,10),np.logspace(1,5,10))
ax = plt.gca()
ax.get_xaxis().set_major_formatter(plt.LogFormatter(10,  labelOnlyBase=False))
ax.get_yaxis().set_major_formatter(plt.LogFormatter(10,  labelOnlyBase=False))

which gives
enter image description here

Update:

The approach shown above only works if the data range is big enough. If the scientific notation is wanted for a smaller range a custom Formatter can be used as

import numpy as np
import matplotlib.pyplot as plt
from matplotlib.ticker import FuncFormatter

def MyFormatter(x,lim):
    if x == 0:
        return 0
    return '{0:.2f}e{1:.2f}'.format(np.sign(x)*10**(-np.floor(np.log10(abs(x)))+np.log10(abs(x))),np.floor(np.log10(abs(x))))
    #The first argument of the format gives the first significant digits of the number with the sign preserved and brought to a range between [1-10), The next argument gives the  numbers integer exponent of 10
    #Both the first and second arguments are formatted to display only 2 decimal places due to the lack of space.

majorFormatter = FuncFormatter(MyFormatter)

t = np.arange(0.0, 100.0, 0.1)
s = np.sin(0.1*np.pi*t)*np.exp(-t*0.01)

fig, ax = plt.subplots()
plt.plot(t,s)

ax.xaxis.set_major_formatter(majorFormatter)

This gives a plot like
enter image description here

Upvotes: 5

lunguini
lunguini

Reputation: 991

building on the answer of Jakob but using python's built in scientific notation string formatting

from matplotlib.ticker import FuncFormatter
from matplotlib import pyplot as plt

def sci_format(x,lim):
    return '{:.1e}'.format(x)

major_formatter = FuncFormatter(sci_format)

t = np.arange(0.0, 100.0, 0.1)
s = np.sin(0.1*np.pi*t) * np.exp(-t*0.01)

fig, ax = plt.subplots()
plt.plot(t, s)

ax.xaxis.set_major_formatter(major_formatter)

Upvotes: 1

Ryan Farber
Ryan Farber

Reputation: 391

The (updated) answer did not work for me with negative values (since log(x) returns Nan for negative x).

Also, I think the following is much simpler:

import numpy as np
import matplotlib.pyplot as plt
from matplotlib.ticker import FuncFormatter

def MyFormatter(x,lim):
  if x == 0:
      return 0
  else:
    x = str(x).split("e")
    return x[0] + r"$\times 10^{" + x[1] + r"}$"
  # end if/else
# end def
majorFormatter = FuncFormatter(MyFormatter)

t = np.arange(0.0, 100.0, 0.1)
s = np.sin(0.1*np.pi*t)*np.exp(-t*0.01)

fig, ax = plt.subplots()
plt.plot(t,s)

ax.xaxis.set_major_formatter(majorFormatter)

Upvotes: 0

Related Questions