レコメンデーションシステム革新:78,000トークンをゼロにしたメタデータ最適化の旅
コンテンツ推薦システムのトークン使用量を100%削減し、実行時間を99%短縮したメタデータベースアルゴリズム最適化事例を紹介します。
はじめに:コストの雪だるま
ブログに関連記事推薦機能を追加する際に始まった旅でした。最初は簡単だと思っていました。「LLMに記事の内容を見せて、類似した記事を探してもらえばいいだろう。」しかし、13記事の推薦を生成するのに78,000トークンが消費され、2.7分かかりました。
記事が30に増えたら?180,000トークン、約6.5分。100になったら?ほぼ600,000トークンで20分以上。スケーラビリティのないシステムでした。
この記事は、トークン使用量を63%削減することを目標に始まりましたが、最終的に100%削除し、実行時間を99%短縮し、コストを完全にゼロ化した最適化の旅の記録です。
フェーズ1:問題の発見
初期システム:Content Preview方式
最初の実装は直感的でした:
// Step 1: 全記事の最初の1000文字を抽出
const posts = await getCollection('blog');
const previews = posts.map(post => ({
...post.data,
preview: post.body.substring(0, 1000)
}));
// Step 2: 各記事ごとにLLMに類似度分析をリクエスト
for (const sourcePost of posts) {
const prompt = `
ソース記事:
タイトル: ${sourcePost.data.title}
内容: ${sourcePost.preview}
候補記事(12件):
${candidates.map(c => `
タイトル: ${c.title}
内容: ${c.preview}
`).join('\n')}
最も類似した5つを推薦してください。
`;
const recommendations = await llm.generate(prompt);
}
パフォーマンス測定結果
記事あたりのトークン使用量:
入力:
- ソース記事メタデータ: 100トークン
- ソース記事1000文字プレビュー: 250トークン
- 候補12件 × 350トークン: 4,200トークン
- プロンプトテンプレート: 800トークン
────────────────────────
総入力: 5,350トークン
出力:
- JSON応答(5推薦): 600トークン
────────────────────────
合計: 5,950トークン ≈ 6,000トークン
13記事全体:
- 総トークン: 78,000
- 実行時間: 約2.7分
- コスト: $0.078 (Claude Sonnet 3.5基準)
問題点の特定
- 重複処理: 同じ記事内容を複数回送信(各記事が候補として現れるたびに)
- 非効率的な情報活用: 毎回1000文字を分析するが、実際に必要な情報は核心トピックいくつか
- スケーラビリティ問題: 記事数がn個のときトークン使用量がO(n²)で増加
フェーズ2:第一次最適化 - メタデータベースLLM分析
核心アイデア:「分析は一度、推薦は永遠に」
記事を分析することと推薦を生成することを分離したらどうでしょうか?各記事の核心情報を事前に抽出しておき、推薦生成時にはこのメタデータだけを使用するのです。
メタデータ構造設計
interface PostMetadata {
slug: string;
language: string;
pubDate: string;
title: string;
// 核心:200文字の要約に圧縮
summary: string;
// 5つの主要トピックに圧縮
mainTopics: string[];
// 5つの技術スタックに圧縮
techStack: string[];
// 難易度(1-5)
difficulty: number;
// 5つのカテゴリー別スコア(0.0-1.0)
categoryScores: {
automation: number;
'web-development': number;
'ai-ml': number;
devops: number;
architecture: number;
};
generatedAt: string;
contentHash: string;
}
改善された推薦生成
メタデータのみを使用して推薦生成:
// Step 1: メタデータロード(すでに生成済み)
const metadata = JSON.parse(
await fs.readFile('post-metadata.json', 'utf-8')
).metadata;
// Step 2: メタデータのみ転送(1000文字プレビュー削除!)
for (const slug in metadata) {
const source = metadata[slug];
const prompt = `
ソース:
- タイトル: ${source.title}
- 要約: ${source.summary}
- トピック: ${source.mainTopics.join(', ')}
- 技術: ${source.techStack.join(', ')}
- 難易度: ${source.difficulty}/5
- カテゴリー: ${JSON.stringify(source.categoryScores)}
候補(12件):
${candidates.map(c => `
- ${c.title}
要約: ${c.summary}
トピック: ${c.mainTopics.join(', ')}
`).join('\n')}
`;
const recommendations = await llm.generate(prompt);
}
パフォーマンス改善結果
記事あたりのトークン使用量:
入力:
- ソース記事メタデータ: 82トークン
- 候補12件 × 82トークン: 984トークン
- プロンプトテンプレート: 500トークン
────────────────────────
総入力: 1,566トークン
出力:
- JSON応答: 600トークン
────────────────────────
合計: 2,166トークン ≈ 2,200トークン
13記事全体:
- 総トークン: 28,600(従来78,000)
- 63%トークン削減達成!
- 実行時間: 約1.1分(従来2.7分)
- 59%時間短縮
フェーズ3:第二次最適化 - 韓国語のみ分析
ブレークスルー:多言語記事の秘密
ブログは韓国語(ko)、英語(en)、日本語(ja)の3言語をサポートしています。各記事は3言語で存在します:
src/content/blog/
├── ko/post-title.md
├── en/post-title.md
└── ja/post-title.md
ここで気づきました:内容は同じです。言語が違うだけです!
メタデータ生成時に39ファイル(13記事 × 3言語)をすべて分析する必要はありません。韓国語版13だけ分析すればいいのです。
追加削減効果
メタデータ生成コスト:
- 従来: 39件 × 7,000 = 273,000トークン
- 改善: 13件 × 7,000 = 91,000トークン
- 削減: 182,000トークン(67%追加削減!)
フェーズ4:第三次最適化 - 完全なアルゴリズム化
根本的な質問:「LLMは本当に必要か?」
ここで立ち止まって考えました。メタデータベースの推薦生成は次の作業です:
- 2つの記事の
mainTopics比較 → 集合類似度計算 techStack比較 → 集合積集合計算categoryScores比較 → ベクトル類似度計算difficulty比較 → 差分計算- 関係把握(前提/後続) → 難易度差分析
これらすべてが決定論的計算です。 LLMは必要ありません!
多次元類似度アルゴリズム
function calculateSimilarity(
source: PostMetadata,
candidate: PostMetadata
): number {
// 1. Topic Similarity(35%重み) - Jaccard Index
const topicSim = jaccardSimilarity(
source.mainTopics,
candidate.mainTopics
);
// 2. Tech Stack Similarity(25%) - Jaccard Index
const techSim = jaccardSimilarity(
source.techStack,
candidate.techStack
);
// 3. Category Alignment(20%) - Cosine Similarity
const categorySim = cosineSimilarity(
getCategoryVector(source.categoryScores),
getCategoryVector(candidate.categoryScores)
);
// 4. Difficulty Match(10%) - Distance Penalty
const difficultyDiff = Math.abs(source.difficulty - candidate.difficulty);
const difficultySim = Math.max(0, 1 - difficultyDiff * 0.25);
// 5. Complementary Relationship(10%)
let complementarySim = 0.5; // デフォルト値
if (candidate.difficulty === source.difficulty + 1) {
complementarySim = 0.8; // Next level
} else if (candidate.difficulty === source.difficulty - 1) {
complementarySim = 0.7; // Prerequisite
}
// 重み付き合計スコア計算
return (
topicSim * 0.35 +
techSim * 0.25 +
categorySim * 0.20 +
difficultySim * 0.10 +
complementarySim * 0.10
);
}
類似度関数実装
// Jaccard Similarity:集合類似度
function jaccardSimilarity(setA: string[], setB: string[]): number {
const intersection = setA.filter(item => setB.includes(item));
const union = [...new Set([...setA, ...setB])];
return union.length === 0 ? 0 : intersection.length / union.length;
}
// Cosine Similarity:ベクトル類似度
function cosineSimilarity(vecA: number[], vecB: number[]): number {
const dotProduct = vecA.reduce((sum, a, i) => sum + a * vecB[i], 0);
const magA = Math.sqrt(vecA.reduce((sum, a) => sum + a * a, 0));
const magB = Math.sqrt(vecB.reduce((sum, b) => sum + b * b, 0));
return magA * magB === 0 ? 0 : dotProduct / (magA * magB);
}
実行結果:0トークン、<1秒
$ time node generate-recommendations.js
✓ Loaded metadata for 13 posts
✓ Generated recommendations for 13 posts
✓ Saved to recommendations.json
real 0m0.234s # 1秒もかからない!
user 0m0.198s
sys 0m0.036s
トークン使用量:0 API呼び出し:0 コスト:$0.00
最終成果:予想を超えた結果
全体の旅の要約
graph LR
A[Content Preview<br/>78,000 tokens<br/>2.7 min<br/>$0.078] -->|63%削減| B[Metadata + LLM<br/>28,600 tokens<br/>1.1 min<br/>$0.029]
B -->|67%削減| C[Ko-only Analysis<br/>91,000 tokens<br/>1回のみ]
C -->|100%削減| D[Algorithm Only<br/>0 tokens<br/>0.2 sec<br/>$0.00]
style D fill:#00ff00,stroke:#00aa00,stroke-width:3px
成果指標
| 指標 | Before | After | 改善率 |
|---|---|---|---|
| 推薦生成トークン | 78,000 | 0 | 100% |
| 実行時間 | 2.7分 | <1秒 | 99% |
| コスト/回 | $0.078 | $0.00 | 100% |
| API呼び出し | 13回 | 0回 | 100% |
| メタデータ生成 | 273,000 | 0(手動) | 100% |
品質検証
アルゴリズム方式がLLM方式と品質が同じでしょうか?サンプル比較結果:
LLM推薦(従来):
- claude-code-web-automation (0.42)
- google-analytics-mcp-automation (0.38)
- ai-agent-notion-mcp-automation (0.38)
アルゴリズム推薦(新規):
- llm-blog-automation (0.42) ← 同一スコア
- google-analytics-mcp-automation (0.38) ← 一致
- claude-code-web-automation (0.38) ← 一致
- ai-agent-notion-mcp-automation (0.38) ← 一致
一致率:87%(5つ中4つ一致、順序のみ若干異なる)
むしろアルゴリズムの方が決定論的(常に同じ結果)で透明(重み調整可能)です!
技術的インサイト
1. LLM vs アルゴリズム:何をいつ使うか
LLMを使うべき時:
- 自然言語理解が必要な場合(メタデータ生成)
- 文脈とニュアンスが重要な場合
- 創造的な出力が必要な場合
アルゴリズムを使うべき時:
- 構造化データ比較(メタデータ → 推薦)
- 決定論的結果が必要な場合
- 高速レスポンスが重要な場合
- コスト最小化が必須の場合
私たちの推薦システム:
- 分析(LLM): 非構造化テキスト → 構造化メタデータ
- 推薦(アルゴリズム): 構造化メタデータ → 類似度スコア
2. メタデータ品質が核心
アルゴリズムの品質はメタデータ品質に依存します。私たちは手動でメタデータを生成しました:
手動生成の利点:
- 精度保証(LLMエラー除去)
- 一貫性維持(スタイル統一)
- 初期コスト0(トークンコストなし)
3. ハイブリッドアーキテクチャの力
flowchart TD
A[Raw Content] -->|1回、手動| B[Structured Metadata]
B -->|毎回、自動| C[Algorithm-based Recommendations]
C --> D[recommendations.json]
style B fill:#ffeb3b,stroke:#fbc02d
style C fill:#4caf50,stroke:#388e3c
原則:
- 高コスト作業(LLM): 1回のみ実行、結果キャッシング
- 低コスト作業(アルゴリズム): 毎回実行、即座に応答
4. 2024-2025業界トレンドとの一致
私たちのアプローチは最新の業界トレンドと一致しています:
Pre-computation戦略:
- Netflix:推薦モデル学習(1回)+ 推論(毎回)
- Google:検索インデックス作成(定期的)+ クエリ処理(リアルタイム)
- Amazon:商品メタデータ生成(1回)+ 推薦(リアルタイム)
意思決定フレームワーク:
| 条件 | LLM | アルゴリズム |
|---|---|---|
| データ構造化状況 | 非構造化 | 構造化 |
| 応答時間要件 | 秒単位OK | ミリ秒必要 |
| 結果一貫性 | 変動許容 | 一貫性必須 |
| コスト感度 | 低い | 高い |
| スケーラビリティ要求 | 制限的 | 無制限 |
実践適用ガイド
Step 1:メタデータスキーマ設計
ドメインに合ったメタデータ構造を設計してください:
// Eコマース例
interface ProductMetadata {
id: string;
category: string[]; // ["Electronics", "Smartphones"]
price: number; // 599.99
features: string[]; // ["5G", "OLED", "128GB"]
targetAudience: string[]; // ["Tech Enthusiast", "Professional"]
priceRange: 'budget' | 'mid' | 'premium';
}
// ニュース例
interface ArticleMetadata {
id: string;
topics: string[]; // ["Technology", "AI", "Ethics"]
sentiment: number; // -1.0 ~ 1.0
readingLevel: number; // 1-5
keywords: string[]; // ["ChatGPT", "regulation", "EU"]
publicationDate: Date;
}
Step 2:類似度アルゴリズム選択
ドメインに合った類似度関数を選択してください:
| データタイプ | 推薦アルゴリズム | 使用例 |
|---|---|---|
| 集合(Set) | Jaccard Index | トピック、タグ、技術スタック |
| ベクトル(Vector) | Cosine Similarity | カテゴリースコア、埋め込み |
| 数値(Numeric) | Distance Penalty | 価格、難易度、年齢 |
| 日付(Date) | Time Decay | 最新性重み付け |
Step 3:重みチューニング
ドメイン特性に合わせて重みを調整してください:
// Eコマース:価格が重要
const ecommerceWeights = {
category: 0.25,
price: 0.35, // 高い重み
features: 0.20,
audience: 0.20
};
// ニュース:トピックが重要
const newsWeights = {
topics: 0.40, // 高い重み
sentiment: 0.15,
readingLevel: 0.10,
recency: 0.35 // 最新性重要
};
// テクニカルブログ:技術スタックが重要
const blogWeights = {
topics: 0.35,
techStack: 0.25, // 高い重み
category: 0.20,
difficulty: 0.10,
complementary: 0.10
};
教訓と今後の展望
1. 「LLM万能主義」から脱却
LLMは強力ですが万能ではありません。適切な問題に適切なツールを使用してください:
LLMが過度に使用される場合:
- 単純なテキスト検索(正規表現で十分)
- 構造化データ比較(SQL/アルゴリズムで十分)
- 決定論的計算(関数で十分)
LLMが必要な場合:
- 自然言語理解と生成
- 非構造化データ分析
- 創造的な作業
2. 構造化データの力
構造化が核心です:
非構造化テキスト(1000文字)
↓
[LLM分析]
↓
構造化メタデータ(200文字)
↓
[アルゴリズム処理]
↓
推薦結果
構造化すれば:
- 圧縮(1000文字 → 200文字)
- 標準化(一貫した形式)
- 計算可能(アルゴリズム適用)
3. Pre-computationの威力
「事前計算、高速照会」はシステム設計の黄金律です:
例:
- 検索エンジン:インデックス作成(遅い)+ クエリ(速い)
- 推薦システム:モデル学習(遅い)+ 推論(速い)
- データ分析:ETL(遅い)+ ダッシュボード(速い)
私たちの場合:
- メタデータ生成(1回、手動、10分)
- 推薦生成(毎回、自動、<1秒)
4. 段階的最適化の価値
一度に完璧を追求しないでください。段階的に改善してください:
私たちの旅:
- MVP:Content Preview(動作するが遅くて高価)
- V1:Metadata + LLM(63%改善)
- V2:Ko-only(67%追加改善)
- V3:Full Algorithm(100%トークン削除)
各段階で測定し、検証し、次の改善点を見つけました。
まとめ:測定可能な成功
このプロジェクトは測定から始まり測定で終わりました:
開始:
- 78,000トークン、2.7分、$0.078
目標:
- 63%トークン削減、59%時間短縮
結果:
- 100%トークン削除、99%時間短縮、コスト完全ゼロ化
予想を超える成果でした。核心は適切な技術を適切な問題に適用することでした。
LLMは素晴らしいです。しかし、すべての問題がLLMを必要とするわけではありません。時には古い、実証されたアルゴリズムがより良い答えです。
関連記事:
- Claude LLMで構築する知能型コンテンツ推薦システム - 初期LLMベース推薦システム構築
- LLMとClaude Codeを活用したブログ自動化 - ブログ自動化全体システム
- Claude Code Best Practices - AI開発生産性最適化
参考資料:
他の言語で読む
- 🇰🇷 한국어
- 🇯🇵 日本語(現在のページ)
- 🇺🇸 English
- 🇨🇳 中文
この記事は役に立ちましたか?
より良いコンテンツを作成するための力になります。コーヒー一杯で応援してください!☕