Pythonmatplotlibの複数のバー


94

matplotlibで複数のバーをプロットする方法、バー関数を複数回呼び出そうとすると、それらは重なり合い、下の図に示すように、最も高い値の赤のみが表示されます。X軸に日付のある複数のバーをプロットするにはどうすればよいですか?

これまでのところ、私はこれを試しました:

import matplotlib.pyplot as plt
import datetime

x = [
    datetime.datetime(2011, 1, 4, 0, 0),
    datetime.datetime(2011, 1, 5, 0, 0),
    datetime.datetime(2011, 1, 6, 0, 0)
]
y = [4, 9, 2]
z = [1, 2, 3]
k = [11, 12, 13]

ax = plt.subplot(111)
ax.bar(x, y, width=0.5, color='b', align='center')
ax.bar(x, z, width=0.5, color='g', align='center')
ax.bar(x, k, width=0.5, color='r', align='center')
ax.xaxis_date()

plt.show()

私はこれを得た:

ここに画像の説明を入力してください

結果は次のようになりますが、日付はx軸上にあり、バーは互いに隣接しています。

ここに画像の説明を入力してください


x値を変更する必要があります
jterrace 2013年

2
どういう意味ですか ?X値は、日付...ある
ジョン・スミス

4
なぜこれは単にmatplotlibでサポートされていないのですか?!
ihadanny

回答:


118
import matplotlib.pyplot as plt
from matplotlib.dates import date2num
import datetime

x = [
    datetime.datetime(2011, 1, 4, 0, 0),
    datetime.datetime(2011, 1, 5, 0, 0),
    datetime.datetime(2011, 1, 6, 0, 0)
]
x = date2num(x)

y = [4, 9, 2]
z = [1, 2, 3]
k = [11, 12, 13]

ax = plt.subplot(111)
ax.bar(x-0.2, y, width=0.2, color='b', align='center')
ax.bar(x, z, width=0.2, color='g', align='center')
ax.bar(x+0.2, k, width=0.2, color='r', align='center')
ax.xaxis_date()

plt.show()

ここに画像の説明を入力してください

「y値も重複している」とはどういう意味かわかりませんが、次のコードで問題は解決しますか?

ax = plt.subplot(111)
w = 0.3
ax.bar(x-w, y, width=w, color='b', align='center')
ax.bar(x, z, width=w, color='g', align='center')
ax.bar(x+w, k, width=w, color='r', align='center')
ax.xaxis_date()
ax.autoscale(tight=True)

plt.show()

ここに画像の説明を入力してください


ありがとう、でも私が3つのバーを持っているなら、それはよさそうだ。40本くらい試してみるとめちゃくちゃです。ソリューションをよりスケーラブルに更新していただけますか?
ジョンスミス

「めちゃくちゃ」を定義しますか?オーバーラップしているXラベルautofmt_xdate()は、ラベルを自動回転するを使用して修正できます。
ジョンリヨン

問題はXラベルのオーバーラップではなく、y値もオーバーラップしていることです。それを修正する方法は?
ジョンスミス

また、width = 0,2は、長い期間には小さすぎます。より大きな値を使用すると、同じ結果が得られません。
ジョンスミス

もう一つは、最初と最後のスペースです。スペースを削除して最初の日付を直接開始し、同様にスペースなしまたはスペースなしで最後の日付を終了する方法。
ジョンスミス

61

日付をx値として使用する場合の問題は、2番目の画像のような棒グラフが必要な場合、それらが間違っていることです。積み上げ棒グラフ(色を重ねる)を使用するか、日付でグループ化する(x軸の「偽の」日付、基本的にはデータポイントをグループ化する)必要があります。

import numpy as np
import matplotlib.pyplot as plt

N = 3
ind = np.arange(N)  # the x locations for the groups
width = 0.27       # the width of the bars

fig = plt.figure()
ax = fig.add_subplot(111)

yvals = [4, 9, 2]
rects1 = ax.bar(ind, yvals, width, color='r')
zvals = [1,2,3]
rects2 = ax.bar(ind+width, zvals, width, color='g')
kvals = [11,12,13]
rects3 = ax.bar(ind+width*2, kvals, width, color='b')

ax.set_ylabel('Scores')
ax.set_xticks(ind+width)
ax.set_xticklabels( ('2011-Jan-4', '2011-Jan-5', '2011-Jan-6') )
ax.legend( (rects1[0], rects2[0], rects3[0]), ('y', 'z', 'k') )

def autolabel(rects):
    for rect in rects:
        h = rect.get_height()
        ax.text(rect.get_x()+rect.get_width()/2., 1.05*h, '%d'%int(h),
                ha='center', va='bottom')

autolabel(rects1)
autolabel(rects2)
autolabel(rects3)

plt.show()

ここに画像の説明を入力してください


x軸に100日のように表示したい場合、どのようにそれらを適合させますか?
ジョンスミス

1
numpyで必要な日付を簡単に生成できますdatetime64:例:1か月分:np.arange('2012-02', '2012-03', dtype='datetime64[D]')。100日を超える40個のデータセット(別のコメントによる)がある場合は、このデータを表すための最良の方法についてもっと考える必要があるかもしれません。
ジョンリヨン

また、ax.xaxis_date()を使用すると非常に有利です。これは、日付がx軸に収まるためです。
ジョンスミス

3
最初に行ってみませんか?私はあなたがあなたのためにあなたのコードを書くのではなく、あなたが学ぶのを手伝おうとしています。私はあなたがそれを行うことができると確信してxaxis_dateいますが、あなたはtimedeltaそれらが重ならないように各シリーズのあなたの日付値をオフセットするために私が書いたものを適応させる必要があります(例えばを使用する時間数によって)。もう1つの答えはこれだけですが、後でラベルをいじる必要があるかもしれません。
ジョンリヨン

わかりましたが、np.arange( '2012-02'、 '2012-03、dtype =' datetime64 [D] ')を実行すると、次のようになります:' str 'および'のサポートされていないオペランドタイプstr」は
ジョン・スミス

25

私はこれが約matplotlibであることを知っていますが、を使用するpandasseaborn多くの時間を節約できます:

df = pd.DataFrame(zip(x*3, ["y"]*3+["z"]*3+["k"]*3, y+z+k), columns=["time", "kind", "data"])
plt.figure(figsize=(10, 6))
sns.barplot(x="time", hue="kind", y="data", data=df)
plt.show()

ここに画像の説明を入力してください


すばらしい答えですが、x軸のせいでやや不完全です。見栄えを良くできますか?
Spinor8

パンダとmatplotlibを使って彼をやることもできると思います
VickiB19年

19

同様の解決策を探したが、十分な柔軟性が見つからなかったため、独自の関数を作成することにしました。これにより、グループごとに必要な数のバーを作成し、グループの幅とグループ内のバーの個々の幅の両方を指定できます。

楽しい:

from matplotlib import pyplot as plt


def bar_plot(ax, data, colors=None, total_width=0.8, single_width=1, legend=True):
    """Draws a bar plot with multiple bars per data point.

    Parameters
    ----------
    ax : matplotlib.pyplot.axis
        The axis we want to draw our plot on.

    data: dictionary
        A dictionary containing the data we want to plot. Keys are the names of the
        data, the items is a list of the values.

        Example:
        data = {
            "x":[1,2,3],
            "y":[1,2,3],
            "z":[1,2,3],
        }

    colors : array-like, optional
        A list of colors which are used for the bars. If None, the colors
        will be the standard matplotlib color cyle. (default: None)

    total_width : float, optional, default: 0.8
        The width of a bar group. 0.8 means that 80% of the x-axis is covered
        by bars and 20% will be spaces between the bars.

    single_width: float, optional, default: 1
        The relative width of a single bar within a group. 1 means the bars
        will touch eachother within a group, values less than 1 will make
        these bars thinner.

    legend: bool, optional, default: True
        If this is set to true, a legend will be added to the axis.
    """

    # Check if colors where provided, otherwhise use the default color cycle
    if colors is None:
        colors = plt.rcParams['axes.prop_cycle'].by_key()['color']

    # Number of bars per group
    n_bars = len(data)

    # The width of a single bar
    bar_width = total_width / n_bars

    # List containing handles for the drawn bars, used for the legend
    bars = []

    # Iterate over all data
    for i, (name, values) in enumerate(data.items()):
        # The offset in x direction of that bar
        x_offset = (i - n_bars / 2) * bar_width + bar_width / 2

        # Draw a bar for every value of that type
        for x, y in enumerate(values):
            bar = ax.bar(x + x_offset, y, width=bar_width * single_width, color=colors[i % len(colors)])

        # Add a handle to the last drawn bar, which we'll need for the legend
        bars.append(bar[0])

    # Draw legend if we need
    if legend:
        ax.legend(bars, data.keys())


if __name__ == "__main__":
    # Usage example:
    data = {
        "a": [1, 2, 3, 2, 1],
        "b": [2, 3, 4, 3, 1],
        "c": [3, 2, 1, 4, 2],
        "d": [5, 9, 2, 1, 8],
        "e": [1, 3, 2, 2, 3],
        "f": [4, 3, 1, 1, 4],
    }

    fig, ax = plt.subplots()
    bar_plot(ax, data, total_width=.8, single_width=.9)
    plt.show()

出力:

ここに画像の説明を入力してください


これを変更してx軸にラベルを追加するにはどうすればよいですか?バーの各グループのように?
x 8920

変更xticks例えば、プロットのをplt.xticks(range(5), ["one", "two", "three", "four", "five"])
pascscha

素晴らしい機能、とても役に立ちました、ありがとう。私が変更した唯一のことは、barplot呼び出しにlabel = data.keys [i]を入力するだけで、バーリストを作成する必要がない場合に、凡例が簡単になると思うことです。
エイドリアントンプキンス

0

私はこの解決策を実行しました。1つの図に複数のプロットをプロットする場合は、次のプロットをプロットする前に matplotlib.pyplot.hold(True) 、別のプロットを追加できるように正しく設定していることを確認してください。

X軸の日時の値に関しては、バーの配​​置を使用したソリューションが有効です。で別の棒グラフを作成する場合matplotlib.pyplot.bar()、を使用align='edge|center'して設定するだけwidth='+|-distance'です。

すべてのバー(プロット)を正しく設定すると、バーが正常に表示されます。


ドキュメントに記載matplotlib.pyplot.holdされているように、v2.0以降非推奨になっているようです
engineervix
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.