Как правильно сгенерировать трехмерную гистограмму, используя встроенные функции numpy или matplotlib в python? - PullRequest
3 голосов
/ 09 декабря 2011

Это более общий вопрос о создании трехмерной гистограммы в python.

Я попытался создать трехмерную гистограмму, используя массивы X и Y в следующем коде

import matplotlib
import pylab
import numpy as np
import matplotlib.pyplot as plt
from mpl_toolkits.mplot3d.axes3d import Axes3D
from matplotlib import cm

def threedhist():
    X = [1, 3, 5, 8, 6, 7, 1, 2, 4, 5]
    Y = [3, 4, 3, 6, 5, 3, 1, 2, 3, 8]
    fig = pylab.figure()
    ax = Axes3D(fig)
    ax.hist([X, Y], bins=10, range=[[0, 10], [0, 10]])
    plt.xlabel('X')
    plt.ylabel('Y')
    plt.zlabel('Frequency')
    plt.title('Histogram')
    plt.show()

Однако я получаю следующую ошибку

Traceback (most recent call last):
  File "<pyshell#0>", line 1, in <module>
    a3dhistogram()
  File "C:/Users/ckiser/Desktop/Projects/Tom/Python Files/threedhistogram.py", line 24, in a3dhistogram
    ax.hist([X, Y], bins=10, range=[[0, 10], [0, 10]])
  File "C:\Python27\lib\site-packages\matplotlib\axes.py", line 7668, in hist
    m, bins = np.histogram(x[i], bins, weights=w[i], **hist_kwargs)
  File "C:\Python27\lib\site-packages\numpy\lib\function_base.py", line 169, in histogram
    mn, mx = [mi+0.0 for mi in range]
TypeError: can only concatenate list (not "float") to list

Я пробовал код с и без «[» в строке ax.hist ([X, Y], bin = 10, range = [[0, 10], [0, 10]]) Я также пробовалфункция от numpy без успеха H, xedges, yedges = np.histogram2d (x, y, bins = (10, 10)) Я пропустил шаг или параметр?Любой совет будет принята с благодарностью.

Ответы [ 4 ]

4 голосов
/ 01 августа 2018

Я опубликовал это в связанной теме о цветных 3d графиках, но я думаю, что это также актуально здесь, так как я не мог найти полный ответ на то, что мне было нужно в любой теме. Этот код генерирует диаграмму рассеяния гистограммы для любого вида данных x-y. Высота представляет частоту значений в этом контейнере. Так, например, если бы у вас было много точек данных, где (x, y) = (20,20) оно было бы высоким и красным. Если бы у вас было мало точек данных в корзине, где (x, y) = (100,100), это было бы низким и синим.

Примечание: результат будет существенно различаться в зависимости от того, сколько данных у вас есть и сколько бинов вы выбираете для своей гистограммы. Отрегулируйте соответственно!

xAmplitudes = #your data here
yAmplitudes = #your other data here

x = np.array(xAmplitudes)   #turn x,y data into numpy arrays
y = np.array(yAmplitudes)

fig = plt.figure()          #create a canvas, tell matplotlib it's 3d
ax = fig.add_subplot(111, projection='3d')

#make histogram stuff - set bins - I choose 20x20 because I have a lot of data
hist, xedges, yedges = np.histogram2d(x, y, bins=(20,20))
xpos, ypos = np.meshgrid(xedges[:-1]+xedges[1:], yedges[:-1]+yedges[1:])

xpos = xpos.flatten()/2.
ypos = ypos.flatten()/2.
zpos = np.zeros_like (xpos)

dx = xedges [1] - xedges [0]
dy = yedges [1] - yedges [0]
dz = hist.flatten()

cmap = cm.get_cmap('jet') # Get desired colormap - you can change this!
max_height = np.max(dz)   # get range of colorbars so we can normalize
min_height = np.min(dz)
# scale each z to [0,1], and get their rgb values
rgba = [cmap((k-min_height)/max_height) for k in dz] 

ax.bar3d(xpos, ypos, zpos, dx, dy, dz, color=rgba, zsort='average')
plt.title("X vs. Y Amplitudes for ____ Data")
plt.xlabel("My X data source")
plt.ylabel("My Y data source")
plt.savefig("Your_title_goes_here")
plt.show()

Результаты для около 75 тысяч точек данных ниже. Обратите внимание, что вы можете перетаскивать объекты в разные ракурсы и, возможно, захотите сохранить несколько представлений для презентаций, потомков.

3d histogram side view 3d histogram perspective 2

1 голос
/ 11 мая 2013

В этом ответе есть решение для 2D и 3D гистограмм рассеянных точек.Использование простое:

points, sub = hist2d_scatter( radius, density, bins=4 )

points, sub = hist3d_scatter( temperature, density, radius, bins=4 )

Где sub - это экземпляр matplotlib "Subplot" (3D или нет), а points содержит точки, используемые для точечной диаграммы.

1 голос
/ 08 апреля 2015

Я добавил к ответу @ lxop, чтобы учесть произвольный размер сегментов:

from mpl_toolkits.mplot3d import Axes3D
import matplotlib.pyplot as plt
import numpy as np

fig = plt.figure()
ax = fig.add_subplot(111, projection='3d')

x = np.array([0, 2, 5, 10, 2, 3, 5, 2, 8, 10, 11])
y = np.array([0, 2, 5, 10, 6, 4, 2, 2, 5, 10, 11])
# This example actually counts the number of unique elements.
binsOne = sorted(set(x))
binsTwo = sorted(set(y))
# Just change binsOne and binsTwo to lists.
hist, xedges, yedges = np.histogram2d(x, y, bins=[binsOne, binsTwo])

# The start of each bucket.
xpos, ypos = np.meshgrid(xedges[:-1], yedges[:-1])

xpos = xpos.flatten()
ypos = ypos.flatten()
zpos = np.zeros_like(xpos)

# The width of each bucket.
dx, dy = np.meshgrid(xedges[1:] - xedges[:-1], yedges[1:] - yedges[:-1])

dx = dx.flatten()
dy = dy.flatten()
dz = hist.flatten()

ax.bar3d(xpos, ypos, zpos, dx, dy, dz, color='b', zsort='average')
1 голос
/ 20 апреля 2012

Посмотрите на http://matplotlib.sourceforge.net/examples/mplot3d/hist3d_demo.html, это рабочий пример сценария.

Я улучшил код по этой ссылке, чтобы он стал гистограммой:

from mpl_toolkits.mplot3d import Axes3D
import matplotlib.pyplot as plt
import numpy as np

fig = plt.figure()
ax = fig.add_subplot(111, projection='3d')
x = [1, 3, 5, 8, 6, 7, 1, 2, 4, 5]
y = [3, 4, 3, 6, 5, 3, 1, 2, 3, 8]

hist, xedges, yedges = np.histogram2d(x, y, bins=(4,4))
xpos, ypos = np.meshgrid(xedges[:-1]+xedges[1:], yedges[:-1]+yedges[1:])

xpos = xpos.flatten()/2.
ypos = ypos.flatten()/2.
zpos = np.zeros_like (xpos)

dx = xedges [1] - xedges [0]
dy = yedges [1] - yedges [0]
dz = hist.flatten()

ax.bar3d(xpos, ypos, zpos, dx, dy, dz, color='b', zsort='average')
plt.xlabel ("X")
plt.ylabel ("Y")

plt.show()

Я не уверен, как это сделать с помощью Axes3D.hist ().

...