質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.09%

WordPress「update_post_meta」を爆速にカスタム

受付中

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 723

score 16

実現したいこと

update_post_metaを一括でwp_postmetaINSERTしたいです。

質問の背景

カスタムフィールドが1件の記事につき数十個あり、更新のパフォーマンスを懸念しています。

そこでupdate_post_meta

INSERT INTO wp_postmeta (post_id, meta_key, meta_value)
VALUES (1, 'key1', 'one');
INSERT INTO wp_postmeta (post_id, meta_key, meta_value)
VALUES (2, 'key2', 'two');
INSERT INTO wp_postmeta (post_id, meta_key, meta_value)
VALUES (10, 'key10', 'ten');


のような処理を以下のように一括でできる関数を作ろうと考えました。

INSERT INTO wp_postmeta
  (post_id, meta_key, meta_value)
VALUES 
  (1, 'key1', 'one'),
  (2, 'key2', 'two'),
  (10, 'key10', 'ten');

該当のソースコード

一括で登録する関数はできたように思います。以下my_update_post_metasです。

// 登録する値たち
$post_id = 1;
$update = ['color'=>'red','class'=>'low'];
$result = my_update_post_metas( $post_id, $update );
var_dump( $result );

// 一括でpost_metaを登録
function my_update_post_metas( $post_id, $update ){ 
    global $wpdb;

    $formats = '';
    $values  = [];
    foreach ( $update as $meta_key => $meta_value ) {

        $formats .= '(';
        $formats .= '%d, ';
        $formats .= '%s, ';
        $formats .= my_get_sql_format( $meta_value );
        $formats .= '), ';

        $values[] = $post_id;
        $values[] = $meta_key;
        $values[] = $meta_value;            
    }
    $formats = rtrim($formats,', ');

    $sql = $wpdb->query( $wpdb->prepare(
        "
        INSERT INTO wp_postmeta
        (post_id, meta_key, meta_value)
        VALUES $formats    
        ",
        $values
    ));    
    return $sql;
}

// %dや%sなどを取得
function my_get_sql_format( $v ){
    $format;
    $type = gettype( $v );
    if ( $type == 'string' ) {
        $format = '%s';
    } elseif ( $type == 'integer' ) {
        $format = '%d';
    }    
    return $format;
}

発生している問題

問題となっているのは、「値があれば更新、なければ登録」という処理です。
普通のupdate_post_metaには備わっている処理なので、今回のmy_update_post_metasにも実装したいのです。

試したこと

ON DUPLICATE KEY UPDATEを以下# ここのように追加しました。しかしこれでは実現できませんでした。

function my_update_post_metas( $post_id, $update ){ 

    /* 上と同じコードがこの部分に入るので省略 */

    // 以下にON DUPLICATE KEY UPDATEを追加
    $sql = $wpdb->query( $wpdb->prepare(
        "
        INSERT INTO wp_postmeta
        (post_id, meta_key, meta_value)
        VALUES $formats
        ON DUPLICATE KEY UPDATE meta_value = VALUES (meta_value); # ここ
        ",
        $values
    ));    
    return $sql;


なぜできないのかわからないので、ひとまず簡易なものでON DUPLICATE KEY UPDATEをを試してもみました。以下testです。するとこちらはできました。

つまり3があれば変更されましたと更新され、なければ登録されました。

test();
function test(){
    global $wpdb;
    $sql = $wpdb->query(
        "
        INSERT INTO wp_posts
        (ID, post_author, post_content)
        VALUES (3,8,'変更されました')
        ON DUPLICATE KEY UPDATE post_content = VALUES (post_content);    
        "
    );        
}


結局my_update_post_metasを簡易にしただけのtestができたのですが、それではどうやってmy_update_post_metasを直せばいいのかわからないという状況です。

ご意見、ご回答をどうぞ宜しくお願い致します。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

+1

ON DUPLICATE KEY UPDATE はその名前の通り、テーブルに対してキーが同じである必要があります。(主キーまたはユニークインデックス)

wp_postmeta の主キーはmeta_idであり、post_idmeta_keyでユニークになるのかもしれませんが、ユニークインデックスは存在しません。そのためmeta_idを指定しないON DUPLICATE KEY UPDATEはエラーになるかと思います。

SQLでやるとするなら

  • DELETEしてINSERTする
  • 一度SELECTして、INSERTとUPDATEを分ける

あたりでしょうか。

私なら大量にデータがありそうであれば、post_id / meta_keyは1つにして、meta_valueの中身を配列にします。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.09%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る