【Python】一份非常好的Matplotlib教程


Matplotlib 教程

本文為譯文,原文載於此,譯文原載於此。本文歡迎轉載,但請保留本段文字,尊重作者和譯者的權益。謝謝。: )

介紹

Matplotlib 可能是 Python 2D-繪圖領域使用最廣泛的套件。它能讓使用者很輕松地將數據圖形化,並且提供多樣化的輸出格式。這里將會探索 matplotlib 的常見用法。

 

IPython 以及 pylab 模式

IPython 是 Python 的一個增強版本。它在下列方面有所增強:命名輸入輸出、使用系統命令(shell commands)、排錯(debug)能力。我們在命令行終端給 IPython 加上參數 -pylab (0.12 以后的版本是 --pylab)之后,就可以像 Matlab 或者 Mathematica 那樣以交互的方式繪圖。

pylab

pylab 是 matplotlib 面向對象繪圖庫的一個接口。它的語法和 Matlab 十分相近。也就是說,它主要的繪圖命令和 Matlab 對應的命令有相似的參數。

初級繪制

這一節中,我們將從簡到繁:先嘗試用默認配置在同一張圖上繪制正弦和余弦函數圖像,然后逐步美化它。

第一步,是取得正弦函數和預先函數的值:

1
2 3 4 
from pylab import *  X = np.linspace(-np.pi, np.pi, 256,endpoint=True) C,S = np.cos(X), np.sin(X) 

X 是一個 numpy 數組,包含了從  到  等間隔的 256 個值。C 和 S 則分別是這 256 個值對應的余弦和正弦函數值組成的numpy 數組。

你可以在 IPython 的交互模式下測試代碼,也可以下載代碼(下載鏈接就是這些示例圖),然后執行:

1
python exercise_1.py

使用默認配置

Matplotlib 的默認配置都允許用戶自定義。你可以調整大多數的默認配置:圖片大小和分辨率(dpi)、線寬、顏色、風格、坐標軸、坐標軸以及網格的屬性、文字與字體屬性等。不過,matplotlib 的默認配置在大多數情況下已經做得足夠好,你可能只在很少的情況下才會想更改這些默認配置。

1
2 3 4 5 6 7 8 9 
from pylab import *  X = np.linspace(-np.pi, np.pi, 256,endpoint=True) C,S = np.cos(X), np.sin(X)  plot(X,C) plot(X,S)  show() 

點擊圖片下載相應 Python 代碼

默認配置的具體內容

下面的代碼中,我們展現了 matplotlib 的默認配置並輔以注釋說明,這部分配置包含了有關繪圖樣式的所有配置。代碼中的配置與默認配置完全相同,你可以在交互模式中修改其中的值來觀察效果。

1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 
# 導入 matplotlib 的所有內容(nympy 可以用 np 這個名字來使用) from pylab import *  # 創建一個 8 * 6 點(point)的圖,並設置分辨率為 80 figure(figsize=(8,6), dpi=80)  # 創建一個新的 1 * 1 的子圖,接下來的圖樣繪制在其中的第 1 塊(也是唯一的一塊) subplot(1,1,1)  X = np.linspace(-np.pi, np.pi, 256,endpoint=True) C,S = np.cos(X), np.sin(X)  # 繪制余弦曲線,使用藍色的、連續的、寬度為 1 (像素)的線條 plot(X, C, color="blue", linewidth=1.0, linestyle="-")  # 繪制正弦曲線,使用綠色的、連續的、寬度為 1 (像素)的線條 plot(X, S, color="green", linewidth=1.0, linestyle="-")  # 設置橫軸的上下限 xlim(-4.0,4.0)  # 設置橫軸記號 xticks(np.linspace(-4,4,9,endpoint=True))  # 設置縱軸的上下限 ylim(-1.0,1.0)  # 設置縱軸記號 yticks(np.linspace(-1,1,5,endpoint=True))  # 以分辨率 72 來保存圖片 # savefig("exercice_2.png",dpi=72)  # 在屏幕上顯示 show() 

點擊圖片下載相應 Python 代碼

改變線條的顏色和粗細

首先,我們以藍色和紅色分別表示余弦和正弦函數,而后將線條變粗一點。接下來,我們在水平方向拉伸一下整個圖。

1
2 3 4 5 
...
figure(figsize=(10,6), dpi=80) plot(X, C, color="blue", linewidth=2.5, linestyle="-") plot(X, S, color="red", linewidth=2.5, linestyle="-") ... 

點擊圖片下載相應 Python 代碼

設置圖片邊界

當前的圖片邊界設置得不好,所以有些地方看得不是很清楚。

1
2 3 4 
...
xlim(X.min()*1.1, X.max()*1.1) ylim(C.min()*1.1, C.max()*1.1) ... 

更好的方式是這樣:

1
2 3 4 5 6 7 8 
xmin ,xmax = X.min(), X.max()
ymin, ymax = Y.min(), Y.max()  dx = (xmax - xmin) * 0.2 dy = (ymax - ymin) * 0.2  xlim(xmin - dx, xmax + dx) ylim(ymin - dy, ymax + dy) 

點擊圖片下載相應 Python 代碼

設置記號

我們討論正弦和余弦函數的時候,通常希望知道函數在  和  的值。這樣看來,當前的設置就不那么理想了。

1
2 3 4 
...
xticks( [-np.pi, -np.pi/2, 0, np.pi/2, np.pi]) yticks([-1, 0, +1]) ... 

點擊圖片下載相應 Python 代碼

設置記號的標簽

記號現在沒問題了,不過標簽卻不大符合期望。我們可以把  當做是 ,但畢竟不夠精確。當我們設置記號的時候,我們可以同時設置記號的標簽。注意這里使用了 LaTeX。

1
2 3 4 5 6 7 
...
xticks([-np.pi, -np.pi/2, 0, np.pi/2, np.pi],  [r'$-\pi$', r'$-\pi/2$', r'$0$', r'$+\pi/2$', r'$+\pi$'])  yticks([-1, 0, +1],  [r'$-1$', r'$0$', r'$+1$']) ... 

點擊圖片下載相應 Python 代碼

移動脊柱

坐標軸線和上面的記號連在一起就形成了脊柱(Spines,一條線段上有一系列的凸起,是不是很像脊柱骨啊~),它記錄了數據區域的范圍。它們可以放在任意位置,不過至今為止,我們都把它放在圖的四邊。

實際上每幅圖有四條脊柱(上下左右),為了將脊柱放在圖的中間,我們必須將其中的兩條(上和右)設置為無色,然后調整剩下的兩條到合適的位置——數據空間的 0 點。

1
2 3 4 5 6 7 8 9 
...
ax = gca() ax.spines['right'].set_color('none') ax.spines['top'].set_color('none') ax.xaxis.set_ticks_position('bottom') ax.spines['bottom'].set_position(('data',0)) ax.yaxis.set_ticks_position('left') ax.spines['left'].set_position(('data',0)) ... 

點擊圖片下載相應 Python 代碼

添加圖例

我們在圖的左上角添加一個圖例。為此,我們只需要在 plot 函數里以「鍵 - 值」的形式增加一個參數。

1
2 3 4 5 6 
...
plot(X, C, color="blue", linewidth=2.5, linestyle="-", label="cosine") plot(X, S, color="red", linewidth=2.5, linestyle="-", label="sine")  legend(loc='upper left') ... 

點擊圖片下載相應 Python 代碼

給一些特殊點做注釋

我們希望在  的位置給兩條函數曲線加上一個注釋。首先,我們在對應的函數圖像位置上畫一個點;然后,向橫軸引一條垂線,以虛線標記;最后,寫上標簽。

1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 
...
 t = 2*np.pi/3 plot([t,t],[0,np.cos(t)], color ='blue', linewidth=2.5, linestyle="--") scatter([t,],[np.cos(t),], 50, color ='blue')  annotate(r'$\sin(\frac{2\pi}{3})=\frac{\sqrt{3}}{2}$',  xy=(t, np.sin(t)), xycoords='data',  xytext=(+10, +30), textcoords='offset points', fontsize=16,  arrowprops=dict(arrowstyle="->", connectionstyle="arc3,rad=.2"))  plot([t,t],[0,np.sin(t)], color ='red', linewidth=2.5, linestyle="--") scatter([t,],[np.sin(t),], 50, color ='red')  annotate(r'$\cos(\frac{2\pi}{3})=-\frac{1}{2}$',  xy=(t, np.cos(t)), xycoords='data',  xytext=(-90, -50), textcoords='offset points', fontsize=16,  arrowprops=dict(arrowstyle="->", connectionstyle="arc3,rad=.2")) ... 

點擊圖片下載相應 Python 代碼

精益求精

坐標軸上的記號標簽被曲線擋住了,作為強迫症患者(霧)這是不能忍的。我們可以把它們放大,然后添加一個白色的半透明底色。這樣可以保證標簽和曲線同時可見。

1
2 3 4 5 
...
for label in ax.get_xticklabels() + ax.get_yticklabels():  label.set_fontsize(16)  label.set_bbox(dict(facecolor='white', edgecolor='None', alpha=0.65 )) ... 

點擊圖片下載相應 Python 代碼

圖像、子圖、坐標軸和記號

到目前為止,我們都用隱式的方法來繪制圖像和坐標軸。快速繪圖中,這是很方便的。我們也可以顯式地控制圖像、子圖、坐標軸。Matplotlib 中的「圖像」指的是用戶界面看到的整個窗口內容。在圖像里面有所謂「子圖」。子圖的位置是由坐標網格確定的,而「坐標軸」卻不受此限制,可以放在圖像的任意位置。我們已經隱式地使用過圖像和子圖:當我們調用 plot 函數的時候,matplotlib 調用 gca() 函數以及 gcf() 函數來獲取當前的坐標軸和圖像;如果無法獲取圖像,則會調用 figure() 函數來創建一個——嚴格地說,是用 subplot(1,1,1) 創建一個只有一個子圖的圖像。

圖像

所謂「圖像」就是 GUI 里以「Figure #」為標題的那些窗口。圖像編號從 1 開始,與 MATLAB 的風格一致,而於 Python 從 0 開始編號的風格不同。以下參數是圖像的屬性:

參數 默認值 描述
num 1 圖像的數量
figsize figure.figsize 圖像的長和寬(英寸)
dpi figure.dpi 分辨率(點/英寸)
facecolor figure.facecolor 繪圖區域的背景顏色
edgecolor figure.edgecolor 繪圖區域邊緣的顏色
frameon True 是否繪制圖像邊緣

這些默認值可以在源文件中指明。不過除了圖像數量這個參數,其余的參數都很少修改。

你在圖形界面中可以按下右上角的 X 來關閉窗口(OS X 系統是左上角)。Matplotlib 也提供了名為 close 的函數來關閉這個窗口。close 函數的具體行為取決於你提供的參數:

  1. 不傳遞參數:關閉當前窗口;
  2. 傳遞窗口編號或窗口實例(instance)作為參數:關閉指定的窗口;
  3. all:關閉所有窗口。

和其他對象一樣,你可以使用 setp 或者是 set_something 這樣的方法來設置圖像的屬性。

子圖

你可以用子圖來將圖樣(plot)放在均勻的坐標網格中。用 subplot 函數的時候,你需要指明網格的行列數量,以及你希望將圖樣放在哪一個網格區域中。此外,gridspec 的功能更強大,你也可以選擇它來實現這個功能。

點擊圖片下載相應 Python 代碼

點擊圖片下載相應 Python 代碼

點擊圖片下載相應 Python 代碼

點擊圖片下載相應 Python 代碼

坐標軸

坐標軸和子圖功能類似,不過它可以放在圖像的任意位置。因此,如果你希望在一副圖中繪制一個小圖,就可以用這個功能。

點擊圖片下載相應 Python 代碼

點擊圖片下載相應 Python 代碼

記號

良好的記號是圖像的重要組成部分。Matplotlib 里的記號系統里的各個細節都是可以由用戶個性化配置的。你可以用 Tick Locators 來指定在那些位置放置記號,用 Tick Formatters 來調整記號的樣式。主要和次要的記號可以以不同的方式呈現。默認情況下,每一個次要的記號都是隱藏的,也就是說,默認情況下的次要記號列表是空的——NullLocator

Tick Locators

下面有為不同需求設計的一些 Locators。

類型 說明
NullLocator

No ticks.

IndexLocator

Place a tick on every multiple of some base number of points plotted.

FixedLocator

Tick locations are fixed.

LinearLocator

Determine the tick locations.

MultipleLocator

Set a tick on every integer that is multiple of some base.

AutoLocator

Select no more than n intervals at nice locations.

LogLocator

Determine the tick locations for log axes.

這些 Locators 都是 matplotlib.ticker.Locator 的子類,你可以據此定義自己的 Locator。以日期為 ticks 特別復雜,因此 Matplotlib 提供了 matplotlib.dates 來實現這一功能。

其他類型的圖

接下來的內容是練習。請運用你學到的知識,從提供的代碼開始,實現配圖所示的效果。具體的答案可以點擊配圖下載。

普通圖

1
2 3 4 5 6 7 8 9 
from pylab import *  n = 256 X = np.linspace(-np.pi,np.pi,n,endpoint=True) Y = np.sin(2*X)  plot (X, Y+1, color='blue', alpha=1.00) plot (X, Y-1, color='blue', alpha=1.00) show() 

點擊圖片下載相應 Python 代碼

散點圖

1
2 3 4 5 6 7 8 
from pylab import *  n = 1024 X = np.random.normal(0,1,n) Y = np.random.normal(0,1,n)  scatter(X,Y) show() 

點擊圖片下載相應 Python 代碼

條形圖

1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 
from pylab import *  n = 12 X = np.arange(n) Y1 = (1-X/float(n)) * np.random.uniform(0.5,1.0,n) Y2 = (1-X/float(n)) * np.random.uniform(0.5,1.0,n)  bar(X, +Y1, facecolor='#9999ff', edgecolor='white') bar(X, -Y2, facecolor='#ff9999', edgecolor='white')  for x,y in zip(X,Y1):  text(x+0.4, y+0.05, '%.2f' % y, ha='center', va= 'bottom')  ylim(-1.25,+1.25) show() 

點擊圖片下載相應 Python 代碼

等高線圖

1
2 3 4 5 6 7 8 9 10 11 12 
from pylab import *  def f(x,y): return (1-x/2+x**5+y**3)*np.exp(-x**2-y**2)  n = 256 x = np.linspace(-3,3,n) y = np.linspace(-3,3,n) X,Y = np.meshgrid(x,y)  contourf(X, Y, f(X,Y), 8, alpha=.75, cmap='jet') C = contour(X, Y, f(X,Y), 8, colors='black', linewidth=.5) show() 

點擊圖片下載相應 Python 代碼

灰度圖(Imshow)

1
2 3 4 5 6 7 8 9 
from pylab import *  def f(x,y): return (1-x/2+x**5+y**3)*np.exp(-x**2-y**2)  n = 10 x = np.linspace(-3,3,4*n) y = np.linspace(-3,3,3*n) X,Y = np.meshgrid(x,y) imshow(f(X,Y)), show() 

點擊圖片下載相應 Python 代碼

餅狀圖

1
2 3 4 5 
from pylab import *  n = 20 Z = np.random.uniform(0,1,n) pie(Z), show() 

點擊圖片下載相應 Python 代碼

量場圖(Quiver Plots)

1
2 3 4 5 
from pylab import *  n = 8 X,Y = np.mgrid[0:n,0:n] quiver(X,Y), show() 

點擊圖片下載相應 Python 代碼

網格

1
2 3 4 5 6 7 8 9 
from pylab import *  axes = gca() axes.set_xlim(0,4) axes.set_ylim(0,3) axes.set_xticklabels([]) axes.set_yticklabels([])  show() 

點擊圖片下載相應 Python 代碼

多重網格

1
2 3 4 5 6 7 
from pylab import *  subplot(2,2,1) subplot(2,2,3) subplot(2,2,4)  show() 

點擊圖片下載相應 Python 代碼

極軸圖

1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 
from pylab import *  axes([0,0,1,1])  N = 20 theta = np.arange(0.0, 2*np.pi, 2*np.pi/N) radii = 10*np.random.rand(N) width = np.pi/4*np.random.rand(N) bars = bar(theta, radii, width=width, bottom=0.0)  for r,bar in zip(radii, bars):  bar.set_facecolor( cm.jet(r/10.))  bar.set_alpha(0.5)  show() 

點擊圖片下載相應 Python 代碼

3D 圖

1
2 3 4 5 6 7 8 9 10 11 12 13 14 
from pylab import * from mpl_toolkits.mplot3d import Axes3D  fig = figure() ax = Axes3D(fig) X = np.arange(-4, 4, 0.25) Y = np.arange(-4, 4, 0.25) X, Y = np.meshgrid(X, Y) R = np.sqrt(X**2 + Y**2) Z = np.sin(R)  ax.plot_surface(X, Y, Z, rstride=1, cstride=1, cmap='hot')  show() 

點擊圖片下載相應 Python 代碼

手稿

點擊圖片下載相應 Python 代碼

結束篇

Matplotlib 能有今天這樣強大的功能和廣泛的使用得益於大量的文檔和社區開發者。這里提供一些有益的鏈接。

入門教程

Matplotlib 文檔

隨代碼分發的文檔

Matplotlib 的代碼是自文檔(Self-documented)的。你可以在 Python 中快速查詢某個命令的用法。

1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 
>>> from pylab import * >>> help(plot) Help on function plot in module matplotlib.pyplot:  plot(*args, **kwargs)  Plot lines and/or markers to the  :class:`~matplotlib.axes.Axes`. *args* is a variable length  argument, allowing for multiple *x*, *y* pairs with an  optional format string. For example, each of the following is  legal::   plot(x, y) # plot x and y using default line style and color  plot(x, y, 'bo') # plot x and y using blue circle markers  plot(y) # plot y using x as index array 0..N-1  plot(y, 'r+') # ditto, but with red plusses   If *x* and/or *y* is 2-dimensional, then the corresponding columns  will be plotted.  ... 

畫廊

Matplotlib 畫廊 也非常有用。其中的例子都有配圖和對應的代碼,當你不知道某一個效果如何實現的時候,你可以在這里找找。

哦,這里還有一個小一點的畫廊

郵件列表

你可以在用戶郵件列表提問,或者在開發者郵件列表里交流技術。

譯注:郵件列表是一個很正式的交流平台,其內的每一封郵件,列表的收聽者都會看到,所以請不要在郵件列表灌水或發表質量低劣的文章。

快速查詢

你可以在這里找到 Matplotlib 主要的屬性表,以便按照需求個性化配置你的輸出圖樣。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM