5
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

PythonとOpenAI APIで実践!MCP開発入門【第5回】AIを操る「呪文」? プロンプトエンジニアリングの基礎と効果的な指示の黄金律(ゼロ/ワン/フューショットの概念も)

Last updated at Posted at 2025-05-20

はじめに:AIのポテンシャルを解放する「プロンプト」という鍵

皆さん、こんにちは! AI開発の冒険、第5回へようこそ!前回(第4回)では、Pythonコードを使ってOpenAI APIに初めてのリクエストを送信し、AIからの応答(JSON形式)を受け取るという、記念すべき「ファーストコンタクト」を果たしましたね。AIがあなたの指示に応じてテキストを生成する様は、まさに未来を感じさせる体験だったのではないでしょうか。(2025年5月、ここ東京をはじめ世界中で、LLMを活用したサービス開発が熱を帯びています!)

しかし、AIから返ってきた応答を見て、「うーん、ちょっと期待と違うな…」「もっとこうして欲しいんだけど…」と感じた方もいるかもしれません。実は、AI(特にGPTのような大規模言語モデル - LLM)の性能を最大限に引き出し、私たちが本当に望む結果を得るためには、AIへの「指示の出し方」が極めて重要になります。

この「AIへの効果的な指示や質問を設計し、最適化する技術・ノウハウ」こそが、今、AI開発の世界で最も注目されているスキルの一つ、「プロンプトエンジニアリング」 です。

今回の記事では、このプロンプトエンジニアリングの基本的な考え方と、AIに「伝わる」指示を出すための黄金律、そして具体的なプロンプト作成テクニック(ゼロショット、ワンショット、フューショットプロンプティングの概念を含む)を、Pythonのコード例と共に詳しく解説していきます。この記事を読めば、あなたもAIをより巧みに操る「呪文(プロンプト)」の使い手へと一歩近づけるはずです!

はじめに:AIのポテンシャルを解放する「プロンプト」という鍵 - visual selection.png

なぜプロンプトが重要なのか? LLMの「取扱説明書」としてのプロンプト

LLMは、膨大なテキストデータで事前学習されており、驚くほど多様なタスクをこなす潜在能力を持っています。しかし、それはあくまで「潜在的」な能力。その力を特定の方向に導き、具体的な成果物として結実させるためには、ユーザーからの明確な指示、すなわち 「プロンプト」 が不可欠です。

  • LLMの動作原理(超簡略版): LLMは、入力されたプロンプト(テキストシーケンス)に続く、最も「ありえそうな」次の単語(より正確にはトークン)を予測し、それを繋げていくことで応答を生成します。つまり、プロンプトの質が、AIの思考の出発点と方向性を決定づけるのです。
    プロンプトの役割:
  • タスク定義 (Task Definition): AIに何をしてほしいのか(例:翻訳、要約、質問応答、コード生成など)を明確に伝えます。
  • コンテキスト提供 (Context Provisioning): タスクを遂行する上で必要な背景情報、制約条件、入力データなどを与えます。(これは、後の回で詳述するMCPによる広範なコンテキスト提供とは別に、プロンプト内で直接的に与える短期的なコンテキストです。)
  • 出力形式の指定 (Output Formatting): どのような形式(例:箇条書き、JSON、特定の文体、特定の言語)で応答してほしいかを指示します。

質の高いプロンプトは、AIの能力を最大限に引き出し、期待通りの、あるいは期待以上の結果をもたらします。逆に、曖昧で不適切なプロンプトは、AIを混乱させ、的外れな応答や、時には望ましくない結果(ハルシネーションと呼ばれる、もっともらしい嘘の情報を生成するなど)を生み出す原因ともなります。

効果的なプロンプトの黄金律:AIに「伝わる」指示の共通原則

具体的なテクニックに入る前に、あらゆるプロンプト作成に共通する「黄金律」をいくつか押さえておきましょう。

  1. 明確かつ具体的に (Be Clear and Specific):
    AIは「空気を読む」ことができません。曖昧な指示は曖昧な結果しか生みません。「いい感じの文章を書いて」ではなく、「[ターゲット読者]向けの、[トピック]に関する、[目的]を達成するための、約[文字数]字の、[文体]のブログ記事の導入文を書いてください」のように、できる限り具体的に指示しましょう。

  2. 十分なコンテキストを提供する (Provide Sufficient Context):
    AIがタスクを正しく理解し、適切な判断を下すために必要な背景情報、前提条件、入力データなどを、プロンプト内に含めるようにします。不足している情報があれば、AIは推測に頼らざるを得ず、結果の質が低下します。

  3. 望ましい出力形式を定義する (Define the Desired Output Format):
    箇条書き、表形式、JSONオブジェクト、特定のプログラミング言語のコード、丁寧語、友人向けのカジュアルな言葉遣いなど、期待する出力の形式やスタイルを明確に指示します。

  4. 複雑なタスクは分解する (Break Down Complex Tasks):
    一度に多くのことや非常に複雑な処理をAIに要求すると、品質が低下したり、指示を無視したりすることがあります。そのような場合は、タスクをより小さく、管理しやすい複数のサブタスクに分解し、段階的にAIに指示を与えるアプローチが有効です。(これは、後の回で紹介する「思考の連鎖(Chain of Thought)」などの高度なテクニックにも繋がります。)

  5. 試行錯誤と改善を繰り返す (Iterate and Refine):
    最初から完璧なプロンプトを書けることは稀です。AIの応答を見ながら、プロンプトの表現を変えたり、情報を追加・削除したり、指示の順序を入れ替えたりして、望む結果が得られるまで試行錯誤を繰り返すことが、プロンプトエンジニアリングの醍醐味であり、上達の秘訣です。

基本的なプロンプトテクニックとPythonコード例

それでは、OpenAI API (今回は gpt-4o-mini モデルを使用) を使って、基本的なプロンプトテクニックをPythonで実践してみましょう。

# Python

# (第3回で作成した .env ファイルと APIキー設定が前提です)
import os
from dotenv import load_dotenv
from openai import OpenAI

# .envファイルから環境変数を読み込む
if not load_dotenv():
    print("エラー: .envファイルが見つからないか、読み込めませんでした。")
    exit()

# OpenAIクライアントの初期化
try:
    client = OpenAI()
    print("OpenAIクライアント初期化成功!")
except Exception as e:
    print(f"OpenAIクライアント初期化エラー: {e}")
    exit()

def get_ai_response(prompt_text, system_message=None, model="gpt-4o-mini", max_tokens=150, temperature=0.7):
    """OpenAI APIにリクエストを送信し、AIの応答を取得する関数"""
    messages = []
    if system_message:
        messages.append({"role": "system", "content": system_message})
    messages.append({"role": "user", "content": prompt_text})

    try:
        print(f"\n--- プロンプト送信中 ---\nUser: {prompt_text}")
        if system_message:
            print(f"System: {system_message}")
        
        completion = client.chat.completions.create(
            model=model,
            messages=messages,
            max_tokens=max_tokens,
            temperature=temperature
        )
        ai_response = completion.choices[0].message.content
        print(f"--- AIからの応答 ---\n{ai_response}")
        print(f"(使用トークン数: {completion.usage.total_tokens})") # トークン数表示
        return ai_response
    except Exception as e:
        print(f"APIリクエストエラー: {e}")
        return None

# --- ここから各プロンプトテクニックの例 ---

A. ゼロショット・プロンプティング (Zero-Shot Prompting): 指示のみで実行させる

  • 概念: AIに対して、タスクの実行例(お手本)を一切与えず、タスクの指示(命令)だけを与える方法です。LLMが持つ広範な事前学習知識に依存します。
    Pythonコード例 (感情分析):
# Python

print("\n=== ゼロショット・プロンプティング例:感情分析 ===")
review_text = "この映画は驚くほど素晴らしく、俳優の演技も最高でした!"
prompt_for_sentiment = f"以下の映画レビューの感情を「ポジティブ」「ネガティブ」「ニュートラル」のいずれかで判定してください。\n\nレビュー:\n\"\"\"\n{review_text}\n\"\"\"\n\n感情判定:"
get_ai_response(prompt_for_sentiment, max_tokens=10)
# 期待される応答: ポジティブ

解説

比較的単純なタスクや、LLMが事前学習で十分に知識を持っているであろうタスクに適しています。プロンプトが短く済むため、トークン数(=コスト)を抑えやすいのがメリットです。

B. ワンショット・プロンプティング (One-Shot Prompting): お手本を1つ示す

概念 (ユーザー様提供PDF資料参考): AIに対して、期待する入力と出力のペアの「お手本」を1つだけ提示し、それに倣ってタスクを実行させる方法です。
Pythonコード例 (簡単な言い換え):

# Python

print("\n=== ワンショット・プロンプティング例:言い換え ===")
one_shot_prompt = """
入力された文章を、より丁寧な表現に言い換えてください。

例:
入力: 明日の会議、資料まだ?
出力: 明日の会議の資料ですが、ご準備の状況はいかがでしょうか。

入力: この機能、超使いにくいんだけど。
出力:
"""
get_ai_response(one_shot_prompt, max_tokens=50)
# 期待される応答例: こちらの機能につきまして、少々操作が分かりづらい点があるかと存じます。

解説

ゼロショットではAIが意図を誤解しやすい、少し特殊なフォーマットやスタイルを要求する場合に有効です。お手本を1つ示すことで、AIの「推測の方向性」をより明確にガイドできます。

C. フューショット・プロンプティング (Few-Shot Prompting): お手本を複数示す

  • 概念 (ユーザー様提供PDF資料参考): AIに対して、期待する入力と出力のペアの「お手本」を複数(通常2~5個程度)提示し、そのパターンを学習させてからタスクを実行させる方法です。
    Pythonコード例 (特定のフォーマットでの情報抽出):
# Python

print("\n=== フューショット・プロンプティング例:情報抽出 ===")
few_shot_prompt = """
以下の文章から、社名、製品名、発売日をJSON形式で抽出してください。

文章1: 株式会社テック未来は、2025年5月20日に革新的なAIアシスタント「ContextAI v2.0」を発表しました。
JSON1: {"company": "株式会社テック未来", "product": "ContextAI v2.0", "release_date": "2025-05-20"}

文章2: 本日、スマートガジェット社より、新型ウェアラブルデバイス「FitTrack Pro」が2025年6月1日より販売開始とのアナウンスがありました。
JSON2: {"company": "スマートガジェット社", "product": "FitTrack Pro", "release_date": "2025-06-01"}

文章3: 2025年4月15日、AIソリューションズLLCは、業務効率化SaaS「AutomateFlow Max」の提供を開始したと報じられました。
JSON3:
"""
get_ai_response(few_shot_prompt, max_tokens=100)
# 期待される応答例: {"company": "AIソリューションズLLC", "product": "AutomateFlow Max", "release_date": "2025-04-15"}

解説

より複雑なタスク、特定の出力フォーマットの厳守、あるいはLLMが事前学習データにあまり持っていないような新しいパターンの認識を要求する場合に、非常に高い効果を発揮します。お手本が多いほどAIはパターンを正確に捉えやすくなりますが、プロンプトが長くなりトークン消費量が増えるというトレードオフがあります。

読者向け:プロンプト作成の実践的ヒント

  • 区切り文字(Delimiters)の活用
    プロンプト内で、指示、コンテキスト、入力データ、お手本などを明確に区別するために、```(トリプルバッククオート)、"""(トリプルダブルクォート)、###、XMLタグ(例: <example></example>)のような区切り文字を使いましょう。これにより、AIがプロンプトの構造を理解しやすくなり、指示の解釈精度が向上します。
# Python

# 区切り文字の例
prompt_with_delimiters = """
以下の指示に従ってください。
###指示###
提供された製品レビューから、製品名を抽出してください。

###製品レビュー###
先日購入した「スーパーコードエディタX」は、機能が豊富で非常に使いやすいです。

###製品名###
"""
get_ai_response(prompt_with_delimiters) # => スーパーコードエディタX
  • PlaygroundやJupyter Notebookでの試行錯誤
    OpenAIが提供するPlaygroundや、Jupyter Notebook/Labのような対話的な実行環境は、プロンプトを少しずつ変えながらAIの応答をリアルタイムで確認できるため、プロンプトの試行錯誤と改善に非常に役立ちます。良いプロンプトが見つかったら、それをPythonスクリプトに組み込みましょう。

  • トークン数を常に意識する
    プロンプトが長くなればなるほど、APIコールのコストが増加し、応答速度も低下する可能性があります。APIレスポンスに含まれるusage情報(前回の記事参照)でトークン数を確認し、簡潔かつ効果的なプロンプトを目指しましょう。

おわりに:プロンプトはAIとの「対話の設計図」

今回は、AIの能力を最大限に引き出すための「プロンプトエンジニアリング」の基本原則と、ゼロショット、ワンショット、フューショットといった具体的なテクニックを、Pythonコード例と共に解説しました。

効果的なプロンプトは、AIに対する単なる「お願い」ではなく、AIの思考を導き、望む結果を創り出すための 「対話の設計図」 です。今回学んだ黄金律とテクニックは、その設計図を描くための最初の、しかし最も重要な道具となります。

プロンプトエンジニアリングは、奥が深く、日々新しい発見がある分野です。ぜひ、ご自身で様々なプロンプトを試し、AIとの対話を通じて、その可能性を探求してみてください。そして、得られた知見や面白い発見があれば、こちらのようなコミュニティで共有してみてはいかがでしょうか。

次回予告

プロンプトの基本は掴めました。では、AIに「より広範な背景知識」や「一貫した役割」を与えるにはどうすれば良いのでしょうか?
次回、第6回「MCPの核心技術!AIの理解を深める「コンテキスト情報」のJSON設計パターンとPythonでの実装」では、プロンプトに直接書ききれないような、より構造化された「コンテキスト情報」をJSON形式で設計し、AIに提供する方法(まさに本シリーズにおけるMCPの核心部分)を詳しく解説します。お楽しみに!

5
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
5
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?