NBA 戰績表系統:從 API 抓數據到生成戰績表

更新 發佈閱讀 18 分鐘

這篇教學會帶你建立一個完整的 NBA 數據分析系統,包含兩個核心程式:

  1. 數據抓取器 (fetch_games.py) - 從 API 抓取比賽數據
  2. 戰績生成器 (standings_generator.py) - 計算並顯示戰績表

整體架構

API (balldontlie.io)

fetch_games.py → games_2024.json

standings_generator.py → 戰績表 + CSV

Part 1: 數據抓取器解析

使用的 API:balldontlie.io

你的程式用的是 balldontlie.io 這個第三方 NBA API。它的特點:

  • 免費但有速率限制
  • 需要 API Key
  • 資料結構清楚且穩定

核心功能解析

1. API 請求設定

API_KEY = "3****1-****-****-****-d4****33"
BASE_URL = "https://api.balldontlie.io/v1/games"

headers = {"Authorization": API_KEY}
params = {
"seasons[]": season_year, # 例如 2024
"per_page": 100, # 每次抓 100
"start_date": "2024-10-22" # 可選:從特定日期開始
}

2. 分頁抓取機制(重要!)

cursor = None  # 用來追蹤目前位置

while True:
if cursor:
params["cursor"] = cursor # 下一頁的游標

response = requests.get(BASE_URL, headers=headers, params=params)
data = response.json()

# 取得下一頁的游標
cursor = data.get("meta", {}).get("next_cursor")
if not cursor:
break # 沒有下一頁了

time.sleep(12) # 避免觸發速率限制

為什麼用 cursor 而不是 page number?


Cursor-based pagination 更穩定,即使在抓取過程中有新資料進來,也不會漏掉或重複。


3. 聰明的過濾邏輯

played_games_on_page = [
game for game in page_games
if game.get("home_team_score", 0) > 0 or game.get("visitor_team_score", 0) > 0
]

if page_games and not played_games_on_page:
print("已到達未開打的賽程,停止抓取。")
break

這段很聰明!它會:

  • 只保留已開打的比賽(分數大於 0)
  • 一旦碰到整頁都是未開打的比賽,就停止抓取
  • 節省時間和 API 配額

4. 增量更新功能(超實用!)

if output_path.exists():
# 讀取現有資料
existing_games = data.get("games", [])

# 找出最後一場比賽的日期
latest_date = max(datetime.fromisoformat(g["date"]) for g in existing_games)

# 只抓新的比賽
start_date_for_fetch = (latest_date.date() + timedelta(days=1)).strftime('%Y-%m-%d')
print(f"將從 {start_date_for_fetch} 開始搜尋新比賽")

這表示:

  • 第一次執行:抓整個賽季
  • 之後執行:只抓新的比賽
  • 超級省時間和 API 配額!

5. 去重與排序

# 合併舊資料和新資料
all_games = existing_games + newly_fetched_games

# 用 dict 去重(以 game ID 為 key)
unique_games = {game['id']: game for game in all_games}

# 按日期排序
sorted_games = sorted(unique_games.values(), key=lambda g: g['date'])

Part 2: API 回傳的 JSON 格式

balldontlie.io 回傳的資料長這樣:

{
"data": [
{
"id": 1234567,
"date": "2024-10-22T23:00:00.000Z",
"season": 2024,
"status": "Final",
"period": 4,
"time": "Final",
"postseason": false,
"home_team": {
"id": 2,
"conference": "East",
"division": "Atlantic",
"city": "Boston",
"name": "Celtics",
"full_name": "Boston Celtics",
"abbreviation": "BOS"
},
"visitor_team": {
"id": 20,
"conference": "East",
"division": "Atlantic",
"city": "New York",
"name": "Knicks",
"full_name": "New York Knicks",
"abbreviation": "NYK"
},
"home_team_score": 108,
"visitor_team_score": 104
}
],
"meta": {
"next_cursor": "eyJpZCI6MTIzNDU2Nywic2Vhc29uIjoyMDI0fQ==",
"per_page": 100
}
}

Part 3: 完整使用流程

步驟 1: 抓取賽季數據

# 抓取 2024-25 賽季(會自動判斷當前賽季)
python fetch_games.py

# 指定賽季
python fetch_games.py --season 2023

# 指定輸出檔案
python fetch_games.py --season 2024 --output my_games.json

第一次執行

開始抓取 2024-2025 賽季的比賽資料...
正在抓取第 1...
已抓取 87 場新的已開打比賽。
等待 12 秒以避免觸發頻率限制...
正在抓取第 2...
已抓取 187 場新的已開打比賽。
...
資料庫中總共有 4562024-2025 賽季比賽。
已將最新的比賽資料寫入 games_2024.json

之後執行(增量更新):

找到現有檔案 games_2024.json,將從上次進度開始更新...
上次抓取到 2024-11-15,將從 2024-11-16 開始搜尋新比賽。
開始抓取 2024-2025 賽季的比賽資料...
搜尋開始日期: 2024-11-16
正在抓取第 1...
已抓取 12 場新的已開打比賽。
資料庫中總共有 4682024-2025 賽季比賽。

步驟 2: 生成戰績表

# 使用剛抓的資料
python standings_generator.py games_2024.json

# 同時匯出 CSV
python standings_generator.py games_2024.json --csv standings.csv

輸出:

目標賽季: 2024

正在讀取檔案: games_2024.json...
找到 468 場比賽,過濾後剩下 4682024 賽季的有效例行賽。
已初始化 30 支球隊的統計資料。
已完成所有比賽的數據統計。

EASTERN CONFERENCE
TEAM W L PCT GB CONF DIV HOME ROAD L10 STR PF PA DIFF
1 Cleveland Cava 15 0 1.00010-0 5-0 8-0 7-0 10-0 W15 119.7 109.2 10.5
2 Boston Celtics 12 3 .800 3.0 9-2 4-1 7-1 5-2 7-3 W2 120.4 110.8 9.6
...

Part 4: 自動化腳本

寫一個 bash 腳本 update_standings.sh

#!/bin/bash

echo "=== NBA 戰績表自動更新 ==="
echo ""

# 步驟 1: 更新比賽數據
echo "步驟 1: 從 API 抓取最新比賽..."
python fetch_games.py

# 檢查是否成功
if [ $? -ne 0 ]; then
echo "❌ 數據抓取失敗"
exit 1
fi

echo ""
echo "✓ 數據抓取完成"
echo ""

# 步驟 2: 生成戰績表
echo "步驟 2: 生成戰績表..."
python standings_generator.py --csv standings.csv

if [ $? -ne 0 ]; then
echo "❌ 戰績表生成失敗"
exit 1
fi

echo ""
echo "✓ 戰績表生成完成"
echo ""
echo "=== 更新完成! ==="
echo "- 比賽數據: games_2024.json"
echo "- 戰績表: standings.csv"

使用方法:

chmod +x update_standings.sh
./update_standings.sh

Part 5: 進階技巧

技巧 1: 定時自動更新

cron 設定每天自動更新:

# 編輯 crontab
crontab -e

# 加入這行(每天早上 8 點執行)
0 8 * * * cd /path/to/project && ./update_standings.sh

技巧 2: 錯誤處理與通知

def fetch_with_retry(url, headers, params, max_retries=3):
"""帶重試機制的請求"""
for attempt in range(max_retries):
try:
response = requests.get(url, headers=headers, params=params, timeout=10)
response.raise_for_status()
return response.json()
except requests.exceptions.RequestException as e:
if attempt == max_retries - 1:
raise
print(f"請求失敗,{3-attempt} 秒後重試... ({e})")
time.sleep(3)

技巧 3: 速率限制監控

def check_rate_limit(response):
"""檢查 API 配額"""
remaining = response.headers.get('X-RateLimit-Remaining')
if remaining and int(remaining) < 10:
print(f"⚠️ API 配額剩餘: {remaining}")

技巧 4: 數據驗證

def validate_game_data(game):
"""驗證比賽資料完整性"""
required_fields = ['id', 'date', 'home_team', 'visitor_team']
for field in required_fields:
if field not in game:
return False, f"缺少欄位: {field}"

if game.get('home_team_score', 0) == 0 and game.get('visitor_team_score', 0) == 0:
return False, "比賽尚未開打"

return True, "OK"

Part 6: 常見問題排解

Q1: API Key 失效怎麼辦?

balldontlie.io 註冊新的 API Key,然後替換程式碼中的:

API_KEY = "你的新 API Key"

Q2: 速率限制問題

balldontlie.io 免費版限制:每分鐘 30 次請求

如果太頻繁會收到 HTTP 429 錯誤。你的程式已經處理了(每次等 12 秒),但如果還是有問題:

# 增加等待時間
time.sleep(15) # 改成 15

Q3: JSON 檔案損壞

如果 games_2024.json 損壞,程式會自動重新抓取:

except (json.JSONDecodeError, KeyError) as e:
print(f"讀取或解析現有檔案時出錯: {e}。將重新抓取整個賽季。")
existing_games = []

Q4: 比賽數據不一致

有時候 API 會回傳重複的比賽,你的程式已經處理了:

# 用 game ID 去重
unique_games = {game['id']: game for game in all_games}

Part 7: 完整專案結構

nba-standings/
├── fetch_games.py # API 抓取器
├── standings_generator.py # 戰績表生成器
├── update_standings.sh # 自動化腳本
├── games_2024.json # 比賽數據(自動生成)
├── standings.csv # 戰績表(自動生成)
├── requirements.txt # 相依套件
└── README.md # 說明文件

requirements.txt:

requests>=2.31.0

總結

這個系統的設計很優秀:

增量更新 - 不會重複抓取已有的資料

錯誤處理 - API 失敗時保存現有資料

速率限制 - 自動等待避免被封鎖

資料驗證 - 過濾未開打的比賽

去重排序 - 確保資料品質

模組化設計 - 兩個程式各司其職


你可以每天執行一次 update_standings.sh,就能得到最新的 NBA 戰績表!

🏀有任何問題不要問我 ,程式AI生成的,文章也是AI生成的,我只負責叫他串API跟不斷地抱怨而已🏀

留言
avatar-img
電光文辭 neon prose
7會員
54內容數
放一些AI亂生的東西
2025/11/24
最近玩了一陣子 nanobanana pro,想說來分享一下使用體驗。 先說好的地方吧,這工具對中文的理解真的蠻準的,不用像其他工具還要想半天怎麼用英文描述或是繞一大圈。直接輸入中文,AI 就能理解你要的場景和氛圍,生圖速度也快,整體用起來算是順手。 然後就是版權問題,這工具基本上完全不管版權的
Thumbnail
含有成人內容
2025/11/24
最近玩了一陣子 nanobanana pro,想說來分享一下使用體驗。 先說好的地方吧,這工具對中文的理解真的蠻準的,不用像其他工具還要想半天怎麼用英文描述或是繞一大圈。直接輸入中文,AI 就能理解你要的場景和氛圍,生圖速度也快,整體用起來算是順手。 然後就是版權問題,這工具基本上完全不管版權的
Thumbnail
含有成人內容
2025/11/17
本文探討了現今 AI 模型在辨識手部照片時,容易將額外的手指誤判為皮膚摺痕或陰影的現象。本文透過設計結構化的「三段式提示流程」,引導 AI 逐步觀察並最終正確辨識出第六根手指的實驗過程。
Thumbnail
2025/11/17
本文探討了現今 AI 模型在辨識手部照片時,容易將額外的手指誤判為皮膚摺痕或陰影的現象。本文透過設計結構化的「三段式提示流程」,引導 AI 逐步觀察並最終正確辨識出第六根手指的實驗過程。
Thumbnail
2025/11/09
本文分享如何利用 Google Gemini 的 Canvas 功能,將傳統的旅行行程表轉化為視覺化、可互動的心智圖,並透過 AI 提示工程與協作功能,提升行程規劃的效率與樂趣。教你如何從行程概念的心智圖化、Canvas 的操作實務,到互動協作與發布,一步步打造你的專屬「旅行作戰室」。
Thumbnail
2025/11/09
本文分享如何利用 Google Gemini 的 Canvas 功能,將傳統的旅行行程表轉化為視覺化、可互動的心智圖,並透過 AI 提示工程與協作功能,提升行程規劃的效率與樂趣。教你如何從行程概念的心智圖化、Canvas 的操作實務,到互動協作與發布,一步步打造你的專屬「旅行作戰室」。
Thumbnail
看更多
你可能也想看
Thumbnail
賽勒布倫尼科夫以流亡處境回望蘇聯電影導演帕拉贊諾夫的舞台作品,以十段寓言式殘篇,重新拼貼記憶、暴力與美學,並將審查、政治犯、戰爭陰影與「形式即政治」的劇場傳統推到台前。本文聚焦於《傳奇:帕拉贊諾夫的十段殘篇》的舞台美術、音樂與多重扮演策略,嘗試解析極權底下不可言說之事,將如何成為可被觀看的公共發聲。
Thumbnail
賽勒布倫尼科夫以流亡處境回望蘇聯電影導演帕拉贊諾夫的舞台作品,以十段寓言式殘篇,重新拼貼記憶、暴力與美學,並將審查、政治犯、戰爭陰影與「形式即政治」的劇場傳統推到台前。本文聚焦於《傳奇:帕拉贊諾夫的十段殘篇》的舞台美術、音樂與多重扮演策略,嘗試解析極權底下不可言說之事,將如何成為可被觀看的公共發聲。
Thumbnail
柏林劇團在 2026 北藝嚴選,再次帶來由布萊希特改編的經典劇目《三便士歌劇》(The Threepenny Opera),導演巴里・柯斯基以舞台結構與舞台調度,重新向「疏離」進行提問。本文將從觀眾慾望作為戲劇內核,藉由沉浸與疏離的辯證,解析此作如何再次照見觀眾自身的位置。
Thumbnail
柏林劇團在 2026 北藝嚴選,再次帶來由布萊希特改編的經典劇目《三便士歌劇》(The Threepenny Opera),導演巴里・柯斯基以舞台結構與舞台調度,重新向「疏離」進行提問。本文將從觀眾慾望作為戲劇內核,藉由沉浸與疏離的辯證,解析此作如何再次照見觀眾自身的位置。
Thumbnail
本文深入解析臺灣劇團「晃晃跨幅町」對易卜生經典劇作《海妲.蓋柏樂》的詮釋,從劇本歷史、聲響與舞臺設計,到演員的主體創作方法,探討此版本如何讓經典劇作在當代劇場語境下煥發新生,滿足現代觀眾的觀看慾望。
Thumbnail
本文深入解析臺灣劇團「晃晃跨幅町」對易卜生經典劇作《海妲.蓋柏樂》的詮釋,從劇本歷史、聲響與舞臺設計,到演員的主體創作方法,探討此版本如何讓經典劇作在當代劇場語境下煥發新生,滿足現代觀眾的觀看慾望。
Thumbnail
《轉轉生》為奈及利亞編舞家庫德斯.奧尼奎庫與 Q 舞團創作的當代舞蹈作品,融合舞蹈、音樂、時尚和視覺藝術,透過身體、服裝與群舞結構,回應殖民歷史、城市經驗與祖靈記憶的交錯。本文將從服裝設計、身體語彙與「輪迴」的「誕生—死亡—重生」結構出發,分析《轉轉生》如何以當代目光,形塑去殖民視角的奈及利亞歷史。
Thumbnail
《轉轉生》為奈及利亞編舞家庫德斯.奧尼奎庫與 Q 舞團創作的當代舞蹈作品,融合舞蹈、音樂、時尚和視覺藝術,透過身體、服裝與群舞結構,回應殖民歷史、城市經驗與祖靈記憶的交錯。本文將從服裝設計、身體語彙與「輪迴」的「誕生—死亡—重生」結構出發,分析《轉轉生》如何以當代目光,形塑去殖民視角的奈及利亞歷史。
Thumbnail
明天的 NBA賽程 真的是滿到太扯,根本像提前上演 NBA季後賽的大亂鬥。 從早上 8:30 開打一路打到 11:00,十場比賽+多場東西區強權對決, 完全是 2025NBA賽程 裡最刺激的巨量賽日之一。每一場都有故事: 🔥 老鷹對騎士:雙後衛互拚 🔥 公鹿 vs 尼克:東區強權
Thumbnail
明天的 NBA賽程 真的是滿到太扯,根本像提前上演 NBA季後賽的大亂鬥。 從早上 8:30 開打一路打到 11:00,十場比賽+多場東西區強權對決, 完全是 2025NBA賽程 裡最刺激的巨量賽日之一。每一場都有故事: 🔥 老鷹對騎士:雙後衛互拚 🔥 公鹿 vs 尼克:東區強權
Thumbnail
今日NBA戰火全面開打!超過六場激戰結果曝光 今天的 nba賽程 可說是爆點連連!湖人與國王的對決最受矚目,最終湖人以 127:120 逆轉勝出,LeBron James 在第四節單節攻下15分,再次證明了他仍是聯盟最穩定的終結者。而灰狼與溜馬的比賽也相當刺激,灰狼在 Anthony Ed
Thumbnail
今日NBA戰火全面開打!超過六場激戰結果曝光 今天的 nba賽程 可說是爆點連連!湖人與國王的對決最受矚目,最終湖人以 127:120 逆轉勝出,LeBron James 在第四節單節攻下15分,再次證明了他仍是聯盟最穩定的終結者。而灰狼與溜馬的比賽也相當刺激,灰狼在 Anthony Ed
Thumbnail
想在NBA季後賽押寶?別再盲猜了!本文將揭露我的獨家選秀心法,從冠軍魔咒、數據分析到現代籃球戰術演變,一步步帶你解析為何紐約尼克隊才是今年的黑馬!防守、三分、明星陣容…所有致勝關鍵都在這裡!點擊了解,讓你從「看熱鬧」晉升「看門道」,還有機會抱走獎金!
Thumbnail
想在NBA季後賽押寶?別再盲猜了!本文將揭露我的獨家選秀心法,從冠軍魔咒、數據分析到現代籃球戰術演變,一步步帶你解析為何紐約尼克隊才是今年的黑馬!防守、三分、明星陣容…所有致勝關鍵都在這裡!點擊了解,讓你從「看熱鬧」晉升「看門道」,還有機會抱走獎金!
Thumbnail
NBA季後賽戰況激烈!在近期關鍵的第三戰中,東區的克利夫蘭騎士隊展現韌性,在客場以 126-104 擊敗印第安納步行者隊,成功將系列賽追至 1-2,暫時擺脫了 0-3 落後的絕境 [NBA官網賽事數據: https://www.youtube.com/watch?v=tr2jjqDGrWk]。多諾萬
Thumbnail
NBA季後賽戰況激烈!在近期關鍵的第三戰中,東區的克利夫蘭騎士隊展現韌性,在客場以 126-104 擊敗印第安納步行者隊,成功將系列賽追至 1-2,暫時擺脫了 0-3 落後的絕境 [NBA官網賽事數據: https://www.youtube.com/watch?v=tr2jjqDGrWk]。多諾萬
Thumbnail
2024–25 賽季,誰才是真正的 NBA MVP?當你同時看到 Jokić 再次打出歷史級數據,與 Shai Gilgeous-Alexander(SGA)率隊創下隊史最佳戰績與歷史級防守表現,你會怎麼選?本文將帶你從數據與戰績角度,深入解析 SGA 為何成為本季 MVP 的最佳人選。
Thumbnail
2024–25 賽季,誰才是真正的 NBA MVP?當你同時看到 Jokić 再次打出歷史級數據,與 Shai Gilgeous-Alexander(SGA)率隊創下隊史最佳戰績與歷史級防守表現,你會怎麼選?本文將帶你從數據與戰績角度,深入解析 SGA 為何成為本季 MVP 的最佳人選。
Thumbnail
大家平常在察看美國職籃賽程是否注意到一件事,那就是儘管nba賽程競爭越來越激烈 但nba的隊伍還是會繼續補強不會放過任何機會,其實這就是因為nba附加賽導致的正面效應 今天winbet小編就整理出nba球隊名單陣容,讓好奇開季nba先發名單的球迷都能一窺開季陣容 如果想知道哪裡有nba直播推薦
Thumbnail
大家平常在察看美國職籃賽程是否注意到一件事,那就是儘管nba賽程競爭越來越激烈 但nba的隊伍還是會繼續補強不會放過任何機會,其實這就是因為nba附加賽導致的正面效應 今天winbet小編就整理出nba球隊名單陣容,讓好奇開季nba先發名單的球迷都能一窺開季陣容 如果想知道哪裡有nba直播推薦
追蹤感興趣的內容從 Google News 追蹤更多 vocus 的最新精選內容追蹤 Google News