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

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

新規登録して質問してみよう
ただいま回答率
85.48%
PHP

PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

Q&A

解決済

2回答

330閲覧

[php]ob_startで取得した内容をDOMで置き換えすると一つ飛ばしになる

helloman

総合スコア39

PHP

PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

0グッド

0クリップ

投稿2018/08/30 02:17

phpのDOMでhtmlタグの置き換えで質問中の内容で、新たに出た問題なのですが、

html

1<?php ob_start('buffer_page');?> 2<html> 3<body> 4<div class="target-class">hogehoge</div> 5<div class="target-class">hogehoge</div> 6<div class="target-class">hogehoge</div> 7<div class="target-class">hogehoge</div> 8<div class="target-class">hogehoge</div> 9<div class="target-class">hogehoge</div> 10<div class="target-class">hogehoge</div> 11<div class="target-class">hogehoge</div> 12<div class="target-class">hogehoge</div> 13<div class="target-class">hogehoge</div> 14<div class="target-class">hogehoge</div> 15<div class="target-class">hogehoge</div> 16<div class="target-class">hogehoge</div> 17<div class="target-class">hogehoge</div> 18<div class="target-class">hogehoge</div> 19<div class="target-class">hogehoge</div> 20<div class="target-class">hogehoge</div> 21<div class="target-class">hogehoge</div> 22<div class="target-class">hogehoge</div> 23<div class="target-class">hogehoge</div> 24</body> 25</html> 26<?php ob_end_flush();?>

php

1function buffer_page($content) 2{ 3 $buffer = new bufferPage(); 4 $content = $buffer->formatHtml($content); 5 return $content; 6} 7 8class bufferPage 9{ 10 public function formatHtml($content) 11 { 12 $dom = new DOMDocument(); 13 $dom->loadHTML($content); 14 $divs = $dom->getElementsByTagName('div'); 15 foreach ($divs as $div): 16 if ($div->hasAttribute('class') && preg_match('/target-class/i', $div->getAttribute('class'))) { 17 18 $new_p = $dom->createElement('p'); 19 // 属性を全部コピー 20 foreach ($div->attributes as $attr) { 21 $new_p->setAttribute($attr->nodeName, $attr->nodeValue); 22 } 23 $new_p_content = $div->nodeValue; 24 $new_p->nodeValue = "<span class=\"target-class-inner\">{$new_p_content}</span>"; 25 $div->parentNode->replaceChild($new_p, $div); 26 27 } 28 endforeach; 29 return mb_convert_encoding($dom->saveHTML(), 'UTF-8', 'HTML-ENTITIES'); 30 } 31}

でhtmlを置き換えているのですが、実行すると

html

1<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 2<div class="target-class">hogehoge</div> 3<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 4<div class="target-class">hogehoge</div> 5<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 6<div class="target-class">hogehoge</div> 7<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 8<div class="target-class">hogehoge</div> 9<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 10<div class="target-class">hogehoge</div> 11<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 12<div class="target-class">hogehoge</div> 13<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 14<div class="target-class">hogehoge</div> 15<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 16<div class="target-class">hogehoge</div> 17<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 18<div class="target-class">hogehoge</div> 19<p class="target-class"><span class="target-class-inner">hogehoge</span></p> 20<div class="target-class">hogehoge</div>

と、一つ飛ばしに実行されます。
どこが悪いのお手上げになってしまいました。

よろしくお願い致します。

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

ベストアンサー

JavaScriptのDOMでもそうですが、getElementsByTagNameが返してくるリストは動的なもので、DOMの状態を反映していきます。

そのため、foreachで参照しながらもとのDOMを書き換えていくと、「0番目を参照して<p>に置き換え」→「1番目を参照したつもりが、書き換えでずれていてもとの1番目(書き換え後の0番目)は放置される」、というような挙動になってしまいます。

いったん単なる配列にコピーしてから置き換えを行う、あるいは通常のfor文で後ろから処理していく(一番最後のが減っても、残りのものの添字には影響しない)など、工夫が必要です。

投稿2018/08/30 02:28

maisumakun

総合スコア145184

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

helloman

2018/08/30 03:05

勉強になります。forで後ろから処理する方法でうまくいきました。 ありがとうございます。
guest

0

PHP

1for($i=$divs->length-1;$i>=0;$i--){ 2 $div=$divs->item($i); 3 ・・・ 4}

投稿2018/08/30 02:40

yambejp

総合スコア114839

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

helloman

2018/08/30 03:06

ありがとうございます。 こちらの方法でうまくできました。 勉強になります。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問