今回は、プログラミング言語であるPythonを用いて、B.LEAGUE B2に所属している 「茨城ROBOTS」のスタッツをデータ分析してみました。
本ブログでは主に、チームの特徴や勝敗のポイントなどを記載していきます。
▼ Contents
1. 2020年度B2リーグ順位表
まず、レギュラーシーズンのB2順位表を確認します。

茨城ROBOTSは全体第2位となります。とても好成績を残していますね。この強さはどこにあるのか、スタッツから分析していきます。
2. 平均スタッツ
茨城ROBOTSのスタッツを確認していきます。以下のレーダーグラフはB2リーグの平均値を50とする偏差値方式で表示しています。具体的な数値や、平均値のランクはその都度記載していきます。
「Team」の項目は、本記事の対象チーム(今回は茨城ROBOTS)の値、「OppTeam」は本記事対象チームの対戦相手の値となります。「OppTeam」の値は小さい方が本記事の対象チームにとって値が高いということになります。
まずは基本スタッツを確認します。



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



4-Factorを見ても、B2リーグ2位の
得点力、B2リーグ一位のTOV%の低さですね。特にTOV%は偏差値70超えと圧倒的です。
3. 得点関連
得点の取得割合を見ていきます。まずはシュートアテンプト数です。



茨城ROBOTSは、FGアテンプト数がリーグ2位の平均67.7回です。また、FGAに対し、3PAの割合が高いため、3PAが多いチームであることが伺えます。
一方、Opp-2PA%がB2リーグ1位であることから、対戦相手は茨城ROBOTS相手に3Pを多用する(させられている)ようです。
また、得点ケースを見てみましょう。



得点ケースでは、得点パターンの偏差値がおおよそ50近くであるため、どのシュチュエーションでもまんべんなく点が取れているようです。
対する対戦相手は、Opp-PfT%や、Opp-PitP%が低いため、速攻やインサイドからのポイントが取りづらいようです。これは茨城ROBOTSのTOV%の低さと相関がありそうです。
4. アシスト関連



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



リバウンドスタッツは総じてB2平均値となっています。
6. 勝敗時のスタッツ比較
ここからはROBOTSの勝利要因を探っていきますので、勝敗別にスタッツを見直してみます。
特に注目スべきスタッツは以下です。
6-1. 敗戦時は3Pアテンプト数が増えるが、3P%が上がらない


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

勝利時はDRBリバウンドが取れていますが、敗戦時はDRBリバウンドが取れず、TRB%も大きく減少しています。相手のORBが多くなっているのが伺えます。
7. 考察
B2リーグ全体2位あり、TOVが最も少なく、得点も全体2位です。群馬とのB2決勝戦が楽しみです。
確率の低い3Pを打たされず、しっかりDRBを取れていけばROBOTSペースになっていくと予想します。

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

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行)

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

Twitterもやってまーす。
茨城ROBOTSのスタッツ、気になります。