【B.LEAGUE(B2)】Pythonデータ分析(茨城ROBOTS編)

今回は、プログラミング言語であるPythonを用いて、B.LEAGUE B2に所属している 「茨城ROBOTS」のスタッツをデータ分析してみました。

本ブログでは主に、チームの特徴や勝敗のポイントなどを記載していきます。

1. 2020年度B2リーグ順位表

まず、レギュラーシーズンのB2順位表を確認します。

表 B2順位表

茨城ROBOTSは全体第2位となります。とても好成績を残していますね。この強さはどこにあるのか、スタッツから分析していきます。

Baran

茨城ROBOTSのスタッツ、気になります。

2. 平均スタッツ

茨城ROBOTSのスタッツを確認していきます。以下のレーダーグラフはB2リーグの平均値を50とする偏差値方式で表示しています。具体的な数値や、平均値のランクはその都度記載していきます。

「Team」の項目は、本記事の対象チーム(今回は茨城ROBOTS)の値、「OppTeam」は本記事対象チームの対戦相手の値となります。「OppTeam」の値は小さい方が本記事の対象チームにとって値が高いということになります。

まずは基本スタッツを確認します。

図1 基本スタッツ
 表1-1 基本スタッツ
表1-2 基本スタッツ(対戦相手)

特筆すべきは、B2リーグ2位のPTS(得点)、TOVの少なさでB2リーグ1位、AST(アシスト)がB2全体で1位となっています。また、平均失点も79.4点のリーグ5位、Opp-STLが少なくのでミスが少ない中で、しっかり得点を稼いでいるチームであることが伺えます。

4 Factorも確認しておきます。

図2 4-Factor
表2-1 4-Factor
表2-2 4-Factor(対戦相手)

4-Factorを見ても、B2リーグ2位の得点力、B2リーグ一位のTOV%の低さですね。特にTOV%は偏差値70超えと圧倒的です。

3. 得点関連

得点の取得割合を見ていきます。まずはシュートアテンプト数です。

図3 シュートアテンプト数
表3-1 シュートアテンプト数
表3-2 シュートアテンプト数(対戦相手)

茨城ROBOTSは、FGアテンプト数がリーグ2位の平均67.7回です。また、FGAに対し、3PAの割合が高いため、3PAが多いチームであることが伺えます。

一方、Opp-2PA%がB2リーグ1位であることから、対戦相手は茨城ROBOTS相手に3Pを多用する(させられている)ようです。

また、得点ケースを見てみましょう。

図4 得点ケース
表4-1 得点ケース
表4-2 得点ケース(対戦相手)

得点ケースでは、得点パターンの偏差値がおおよそ50近くであるため、どのシュチュエーションでもまんべんなく点が取れているようです。

対する対戦相手は、Opp-PfT%や、Opp-PitP%が低いため、速攻やインサイドからのポイントが取りづらいようです。これは茨城ROBOTSのTOV%の低さと相関がありそうです。

4. アシスト関連

図5 アシスト
表5-1 アシスト
表5-2 アシスト(対戦相手)

TOVの少なさ、ASTの多さが目立ちます。チームでしっかりミスを減らしながら得点していることが伺えます。また、対戦相手のOpp-STKもB2リーグ最小で、ROBOTSのミスの少なさが伺えます。

5. リバウンド関連

図6 リバウンド
表6-1 リバウンド
表6-2 リバウンド(対戦相手)

リバウンドスタッツは総じてB2平均値となっています。

6. 勝敗時のスタッツ比較

ここからはROBOTSの勝利要因を探っていきますので、勝敗別にスタッツを見直してみます。

特に注目スべきスタッツは以下です。

6-1. 敗戦時は3Pアテンプト数が増えるが、3P%が上がらない

図7 勝敗時のシュートアテンプト数
図8 勝敗時のアシストとシュート確率

茨城は敗戦時に3Pのアテンプト数が上昇する傾向があり、また3P%が下がる傾向があります。そのため、得点効率が減少していると推察されます。

6-2. 敗戦時はDRBリバンドが取れていない

図9 勝敗時のリバウンド

勝利時はDRBリバウンドが取れていますが、敗戦時はDRBリバウンドが取れず、TRB%も大きく減少しています。相手のORBが多くなっているのが伺えます。

7. 考察

B2リーグ全体2位あり、TOVが最も少なく、得点も全体2位です。群馬とのB2決勝戦が楽しみです。

確率の低い3Pを打たされず、しっかりDRBを取れていけばROBOTSペースになっていくと予想します。

Baran

B1昇格おめでとうございます!

Sakura

GO!GO! ROBOTS!!

8. サンプルコード

以下は、Pythonでのレーダーチャートのコード例です。

df_teamは本記事対象チームの2020年度平均スタッツ、df_oppteamは対戦相手の2020年度平均スタッツとなります。

まず、polar_list関数で、表示したデータを抽出して辞書化します。その後、plot_polar関数でレーダーチャートを表示します。

def poloar_list(df_team, df_oppteam, b_team):
    polar_list1 = ['PTS', '3PM', '2PM', 'FTM', 'AST', 'TOV', 'STL', 'ORB', 'TRB', 'DRB', 'BLK']
    polar_list2 = ['eFG%', 'ORB%', 'TOV%', 'FT/FGA']
    polar_list3 = ['FGA', '3PA', '3PA%', 'OPA', 'OPA%', '2PA', '2PA%', 'IPA', 'IPA%', 'FTA']
    polar_list4 = ['PTS', '3P/PTS', '2P/PTS', 'PfT%', 'PitP%', 'SCP%', 'FBP%', 'FT/PTS']
    polar_list5 = ['AST', 'TOV', 'AST/TOV', 'STL', '3P%', '2P%', 'FG%']
    polar_list6 = ['ORB', 'ORB%', 'Opp_DRB', 'DRB', 'DRB%', 'Opp_ORB', 'TRB', 'TRB%', 'Opp_TRB']

    df_team = df_team.loc["{}".format(b_team)]
    df_oppteam = df_oppteam.loc["{}".format(b_team)]
    df_polar_team = df_team[polar_list1]
    df_polar_oppteam = df_oppteam[polar_list1]
    df_polar_team_4factor = df_team[polar_list2]
    df_polar_oppteam_4factor = df_oppteam[polar_list2]
    df_polar_team_attempt = df_team[polar_list3]
    df_polar_oppteam_attempt = df_oppteam[polar_list3]
    df_polar_team_point = df_team[polar_list4]
    df_polar_oppteam_point = df_oppteam[polar_list4]
    df_polar_team_ast = df_team[polar_list5]
    df_polar_oppteam_ast = df_oppteam[polar_list5]
    df_polar_team_rb = df_team[polar_list6]
    df_polar_oppteam_rb = df_oppteam[polar_list6]
    df_dict = {
                0: ["Team", df_polar_team], 1: ["Oppteam", df_polar_oppteam],
                2: ["Team_4factor", df_polar_team_4factor], 3: ["Oppteam_4factor", df_polar_oppteam_4factor],
                4: ["Team_Attempt", df_polar_team_attempt], 5: ["Oppteam_Attempt", df_polar_oppteam_attempt],
                6: ["Team_point", df_polar_team_point], 7: ["Oppteam_point", df_polar_oppteam_point],
                8: ["Team_AST", df_polar_team_ast], 9: ["Oppteam_AST", df_polar_oppteam_ast],
                10: ["Team_RB", df_polar_team_rb], 11: ["Oppteam_RB ", df_polar_oppteam_rb]
    }
    return df_dict


def plot_polar(df_dict, b_team, N=2):
    sns.set(font_scale=1.0)
    fig, ax = plt.subplots(6, N, figsize=(12, 48), subplot_kw={'projection': 'polar'})
    for i, df in df_dict.items():
        x, y = divmod(i, N)
        labels = df[1].index
        values = list(df[1])
        angles = np.linspace(0, 2 * np.pi, len(labels) + 1, endpoint=True)
        values = np.concatenate((values, [values[0]]))
        ax[x, y].plot(angles, values, 'o-', color="blue", label="Overall: {}".format(round(df[1].mean(), 1)))
        ax[x, y].fill(angles, values, alpha=0.25, color="blue")  

        ax[x, y].set_rlim(20, 80)

        ax[x, y].set_thetalim([-np.pi, np.pi])
        ax[x, y].set_thetagrids(angles[:-1] * 180 / np.pi, labels, fontsize=15)
        ax[x, y].set_theta_zero_location('N')

        ind = 2
        gridlines = ax[x, y].yaxis.get_gridlines()
        gridlines[2].set_color("black")
        gridlines[ind].set_linewidth(1)
        gridlines[ind].set_linestyle("--")
        ax[x, y].set_title("{0}: {1} ".format(df[0], b_team), fontsize=15)
        ax[x, y].legend(bbox_to_anchor=(0.5, -0.18), loc= 'lower center')

    plt.tight_layout()
    plt.show()

こちらの記事を参照させていただきました↓

公式のサンプルはうまくプロットできないし、コードが長くて死ぬ。

シンプルにレーダーチャートを描きたいだけなのに…。

Matplotlibでレーダーチャートを描く(16行)
Baran

以下の記事も書いています。よかったらご覧ください。

Twitterもやってまーす。

コメントを残す

メールアドレスが公開されることはありません。 が付いている欄は必須項目です

ABOUT US

Baran-gizagiza
経歴:浪人→理系大学院卒業→大手製造業に就職(技術職)→アメリカ赴任中 仕事は、研究・設計など上流工程の仕事に携わっています。企業勤務を継続しながら、新しいことにチャレンジしたいと思い、ブログを下記はじめました。 このブログでは、趣味である 筋トレ(健康、ダイエット) AIとデータ(高校数学、プログラミング) 読書(主に自己啓発系) を中心に、人生経験やおすすめ情報の発信しています。