堆叠的matplotlib栏中的替代颜色

时间:2012-10-12 15:37:43

标签: python matplotlib

我在matplotlib中生成堆积条形图(在Windows 7上使用Python 2.7)。

由于我想用它来成对地比较来自2个数据集的数据,我想为每个第2个条使用不同的颜色。有谁能告诉我如何实现这个目标?

我的条形图看起来基本上是这样的:

import numpy
import matplotlib.pyplot as plt


fig = plt.figure()
ax1 = fig.add_subplot(1,1,1)

IDs = ["1","A","2","B","3","C","4","D","5","E"]
N = len(IDs)

property1 = numpy.array([1,3,4,2,3,5,6,7,3,2])
property2 = numpy.array(range(10))
property3 = numpy.array(range(10,0,-1))

ind = numpy.arange(N)
width = 0.8

p1 = ax1.bar(ind, property1, width, color='red')
p2 = ax1.bar(ind, property2, width, color='blue', bottom=property1)
p3 = ax1.bar(ind, property3, width, color='green', bottom=property1 + property2)
plt.xticks(ind+width/2., IDs )

plt.show()
plt.close()

所以我想对用字母标记的条形图使用一种颜色方案,对于标有数字的条形图使用另一种颜色方案(因为例如“1”和“A”形成一对 - 它们代表在2种不同实验条件下的相同样品,这就是我希望他们彼此相邻的原因。)

理想情况下,如果宽度可以调整为在一对条之间没有间隙(但是对之间的间隙),那将是非常好的。

但是现在,我不知道如何解决这个问题,所以任何建议都会很棒!

(我可以单独使用两组“数据”,这样更容易吗?也许可以做两个间隙较大的图,相互交叉?)

2 个答案:

答案 0 :(得分:5)

你可以通过切片更优雅地做到这一点。一个可能更好的解决方案是去隔行数据。

IDs = ["1","A","2","B","3","C","4","D","5","E"]

N = len(IDs)
fig = plt.figure()
ax1 = fig.add_subplot(1,1,1)

set_count = 2
set_label = np.vstack([j*ones(len(IDs)//set_count) for j in range(set_count)]).T.ravel()

property1 = numpy.array([1,3,4,2,3,5,6,7,3,2])
property2 = numpy.array(range(10))
property3 = numpy.array(range(10,0,-1))

props = [property1,property2,property3]

ind = numpy.arange(N/set_count)
width = 0.9
b_width = 0.9/set_count

color_sets = [['red','green','blue'],['black','magenta','yellow']]

for j in range(set_count):
    tmp_accum = np.zeros(len(props[0]))
    for k in range(len(props)):
        ax1.bar(ind +j*b_width, props[k][set_label==j], width=b_width, color=color_sets[j][k], bottom=tmp_accum[set_label==j])
        tmp_accum += props[k]


lab_ind = []
for j in range(set_count):

    lab_ind.append(ind + (j+.5)*b_width)

plt.xticks(np.vstack(lab_ind).T.ravel(), IDs )

它需要一些健全性检查,但它可以工作,并且应该可以轻松扩展到2个以上的类和3个以上的属性(您只需要确保所有长度同步,使用itertools.cycle颜色可能有帮助。)

image of results

答案 1 :(得分:1)

找到一种有效的方法,虽然看起来很不优雅:

import numpy
import matplotlib.pyplot as plt

fig = plt.figure()
ax1 = fig.add_subplot(1,1,1)

IDs = ["1","A","2","B","3","C","4","D","5","E"]

N = len(IDs)

property1_1 = numpy.array([1,0,4,0,3,0,6,0,3,0])
property1_2 = numpy.array([0,3,0,2,0,5,0,7,0,2])
property2_1 = numpy.array([1,0,2,0,3,0,4,0,5,0])
property2_2 = numpy.array([0,6,0,7,0,8,0,9,0,10])
property3_1 = numpy.array([10,0,9,0,8,0,7,0,6,0])
property3_2 = numpy.array([0,5,0,4,0,3,0,2,0,1])

ind = numpy.arange(N)
width = 0.8

p1 = ax1.bar(ind, property1_1, width, color='red')
p2 = ax1.bar(ind, property2_1, width, color='blue', bottom=property1_1)
p3 = ax1.bar(ind, property3_1, width, color='green', bottom=property1_1 + property2_1)

p4 = ax1.bar(ind-0.2, property1_2, width, color='#FF6666')
p5 = ax1.bar(ind-0.2, property2_2, width, color='#6699FF', bottom=property1_2)
p6 = ax1.bar(ind-0.2, property3_2, width, color='#33CC33', bottom=property1_2+property2_2)
plt.xticks(ind+width/2., IDs )

plt.show()
plt.close()

这是我为了解决这个问题的其他人而做的。我不认为这是最佳的,并希望得到更多答案。