6
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

身の回りの困りごとを楽しく解決! by Works Human IntelligenceAdvent Calendar 2023

Day 4

ChatGPTに3回質問して「文字の類似度」のサジェストするプログラム作成

Last updated at Posted at 2023-12-03

概要

今回は、ChatGPTにレーベンシュタイン距離を使った文字の類似度を測定しサジェストを作成するプログラムを、3回の質問でプログラムを書いて貰います。

ChatGPTへの質問内容をテクニックを交えてを紹介します。

レーベンシュタイン距離とは

レーベンシュタイン距離は、二つの文字列がどの程度異なっているかを示す距離である。
Wikipediaより引用

要するに、計測した数値が低いものほど、文字の類似度が高いということです。

質問1:丸投げする

とりあえず、ChatGPTに丸投げします。

「買い物するアプリを作って」など抽象的なものではなく、具体的な機能の関数やメソッドなど具体的な内容で質問することを意識しています。

質問内容

質問内容
「レーベンシュタイン距離」を使った文字列の類似度を測定するプログラムをjavascriptで書いて!

ChatGPTの回答

結果が返ってきました。for文を多用していますが、比較的シンプルなコードになっているのかなと思います。

chatGPTの回答
function levenshteinDistance(str1, str2) {
  const len1 = str1.length;
  const len2 = str2.length;
  
  const dp = [];
  
  for (let i = 0; i <= len1; i++) {
    dp[i] = [];
    for (let j = 0; j <= len2; j++) {
      if (i === 0) {
        dp[i][j] = j;
      } else if (j === 0) {
        dp[i][j] = i;
      } else {
        dp[i][j] = 0;
      }
    }
  }
  
  for (let i = 1; i <= len1; i++) {
    for (let j = 1; j <= len2; j++) {
      if (str1[i - 1] === str2[j - 1]) {
        dp[i][j] = dp[i - 1][j - 1];
      } else {
        dp[i][j] = 1 + Math.min(dp[i - 1][j], dp[i][j - 1], dp[i - 1][j - 1]);
      }
    }
  }
  
  return dp[len1][len2];
}

// 使用例
const string1 = "";
const string2 = "";
const distance = levenshteinDistance(string1, string2);
console.log(`"${string1}" と "${string2}" のレーベンシュタイン距離は ${distance} です。`);

実行結果

実際に動かしてみました。うまく動いていそうです。

image.png

質問2:自分で書いたコードをリファクタリングしてもらう

先ほど作ってもらったlevenshteinDistance()を利用して、複数のキーワードと類似度を比較して、類似度が高いものをいくつか返してもらうgetSuggest()を作りました。

見ての通り、綺麗なコードではありません。
(変化が分かりやすいようにあえて汚く書きました!)
このコードをリファクタリングしてもらいます。

自分で書いたコード
function getSuggest(searchText, wordList, size) {
    const data = new Object();
    wordList.forEach((s) => {
        data[s] = levenshteinDistance(searchText,s);
    });
    const sortedData = Object.entries(data).sort((a, b) => a[1] - b[1]);
    r = [];
    sortedData.forEach((d)=>{
        k = d[0];
        v = d[1];
        r[k] = v;
    });
    return Object.keys(r).slice(0, size);
}

const searchText = "太田";
const wordList = ['田中', '中山', '山田', '中田', '山中', '田山'];
const size = 3;
console.log(getSuggest(searchText,wordList,size));

質問内容

質問内容
下記コードを簡潔に書いてほしいなー
function getSuggest(searchText, wordList, size) {
    const data = new Object();
    wordList.forEach((s) => {
        data[s] = levenshteinDistance(searchText,s);
    });
    const sortedData = Object.entries(data).sort((a, b) => a[1] - b[1]);
    r = [];
    sortedData.forEach((d)=>{
        k = d[0];
        v = d[1];
        r[k] = v;
    });
    return Object.keys(r).slice(0, size);
}

ChatGPTの回答

前回の質問あって、質問内容は、関数部分だけで読み取ってくれました。さすがです。
とても可読性の高い、シンプルなコードになりました。

(本来は、何度か繰り返したり、ChatGPTの回答結果をさらにリファクタリングしてもらうことでより良い結果を受け取ることができます)

ChatGPT回答
function getSuggest(searchText, wordList, size) {
  return wordList
    .sort((a, b) => levenshteinDistance(searchText, a) - levenshteinDistance(searchText, b))
    .slice(0, size);
}

実行結果

こちらも問題なく動いているようです。

  • ChatGPTリファクタリング前

image.png

  • ChatGPTリファクタリング後

image.png

質問3:データを作ってもらう

データを作るのも開発に必要な作業です。
自分で作るのもが面倒なので、ChatGPTに作ってもらいます。

質問内容

質問内容
初代ポケモンの配列作れる?まさか、作れないわけ無いよねぇ(^^

ChatGPTの回答

完璧なデータは、作れませんでした。
ゴローニャゴローンダになっていますが、テストするデータとしては十分です。
(ほかにも間違いがあるかもしれないので探してみてください)

ChatGPT回答
const firstGenPokemon = [
  "フシギダネ", "フシギソウ", "フシギバナ", "ヒトカゲ", "リザード", "リザードン", 
  "ゼニガメ", "カメール", "カメックス", "キャタピー", "トランセル", "バタフリー", 
  "ビードル", "コクーン", "スピアー", "ポッポ", "ピジョン", "ピジョット", 
  "コラッタ", "ラッタ", "オニスズメ", "オニドリル", "アーボ", "アーボック", 
  "ピカチュウ", "ライチュウ", "サンド", "サンドパン", "ニドラン♀", "ニドリーナ", 
  "ニドクイン", "ニドラン♂", "ニドリーノ", "ニドキング", "ピッピ", "ピクシー", 
  "ロコン", "キュウコン", "プリン", "プクリン", "ズバット", "ゴルバット", 
  "ナゾノクサ", "クサイハナ", "ラフレシア", "パラス", "パラセクト", "コンパン", 
  "モルフォン", "ディグダ", "ダグトリオ", "ニャース", "ペルシアン", "コダック", 
  "ゴルダック", "マンキー", "オコリザル", "ガーディ", "ウインディ", "ニョロモ", 
  "ニョロゾ", "ニョロボン", "ケーシィ", "ユンゲラー", "フーディン", "ワンリキー", 
  "ゴーリキー", "カイリキー", "マダツボミ", "ウツドン", "ウツボット", "メノクラゲ", 
  "ドククラゲ", "イシツブテ", "ゴローン", "ゴローンダ", "ポニータ", "ギャロップ", 
  "ヤドン", "ヤドラン", "コイル", "レアコイル", "カモネギ", "ドードー", 
  "ドードリオ", "パウワウ", "ジュゴン", "ベトベター", "ベトベトン", "シェルダー", 
  "パルシェン", "ゴース", "ゴースト", "ゲンガー", "イワーク", "スリープ", 
  "スリーパー", "クラブ", "キングラー", "ビリリダマ", "マルマイン", "タマタマ", 
  "ナッシー", "カラカラ", "ガラガラ", "サワムラー", "エビワラー", "ベロリンガ", 
  "ドガース", "マタドガス", "サイホーン", "サイドン", "ラッキー", "モンジャラ", 
  "ガルーラ", "タッツー", "シードラ", "トサキント", "アズマオウ", "ヒトデマン", 
  "スターミー", "バリヤード", "ストライク", "ルージュラ", "エレブー", "ブーバー", 
  "カイロス", "ケンタロス", "コイキング", "ギャラドス", "ラプラス", "メタモン", 
  "イーブイ", "シャワーズ", "サンダース", "ブースター", "ポリゴン", "オムナイト", 
  "オムスター", "カブト", "カブトプス", "プテラ", "カビゴン", "フリーザー", 
  "サンダー", "ファイヤー", "ミニリュウ", "ハクリュー", "カイリュー", "ミュウツー", 
  "ミュウ"
];

完成

それでは、いままでのコードと組み合わせて実行してみましょう

左のデータがより類似度が高い文字列です。
間違った名前で実行してみましたが、想定されたポケモンが出てきます。

image.png

ChatGPTを利用したプログラミングの利点と欠点

利点

  • コーディングが速い
    今回作った「レーベシュタイン距離を使った文字の類似度を計測」するプログラムは、自分で作ろうとすると、結構時間がかかると思いますが、ChatGPTを利用することで数秒で作ってくれます。

  • 新しい知識が増える
    コーディング方法やメソッドの利用方法など、自分が想像しているものと違う手法で実装させるので、新しい知識が増える

欠点

  • ブラックボックス化してしまう
    書いてもらったプログラムが理解ができない場合、ブラックボックス化してしまう可能性があり、今後のメンテナンスなどに影響が出ます。実際に組み込む場合は責任を持って、組み込みましょう。

  • 間違っている可能性がある
    基本的なことですが、書いてもらったプログラムが間違っていることがあります(結構多いです)。ChatGPTを利用するのであれば、それを見極められるレベルの技術者になってほしいです。

おわりに

今回は、ChatGPTに質問しながら、「レーベンシュタイン距離を使い、類似度の高い文字を返却するプログラム」が作成できました。

ChatGPTの使い方としては、特にリファクタリングがおすすめです。

次回は、今回作ったプログラムを実際にアプリに組み込みする内容で記事を書きたいと思います。
ここまで読んで頂きありがとうございました。


2023/12/07追記 Reactで検索のサジェスト作りました!

6
3
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
6
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?