web-dev-qa-db-ja.com

PHPのSimpleXMLで、特定の属性を持つ子を削除します

SimpleXMLでアクセスしている異なる属性を持ついくつかの同一の要素があります。

<data>
    <seg id="A1"/>
    <seg id="A5"/>
    <seg id="A12"/>
    <seg id="A29"/>
    <seg id="A30"/>
</data>

「A12」のIDを持つ特定のseg要素を削除する必要がありますが、これを行うにはどうすればよいですか? seg要素とunsettingをループしてみましたが、これは機能しません、要素残ります。

foreach($doc->seg as $seg)
{
    if($seg['id'] == 'A12')
    {
        unset($seg);
    }
}
48
TimTowdi

SimpleXML削除する方法 XMLノードを提供しますが、変更機能は多少制限されています。もう1つの解決策は、 [〜#〜] dom [〜#〜] 拡張機能を使用することです。 dom_import_simplexml() は、SimpleXMLElementDOMElementに変換するのに役立ちます。

いくつかのサンプルコード(PHP 5.2.5)でテスト済み):

$data='<data>
    <seg id="A1"/>
    <seg id="A5"/>
    <seg id="A12"/>
    <seg id="A29"/>
    <seg id="A30"/>
</data>';
$doc=new SimpleXMLElement($data);
foreach($doc->seg as $seg)
{
    if($seg['id'] == 'A12') {
        $dom=dom_import_simplexml($seg);
        $dom->parentNode->removeChild($dom);
    }
}
echo $doc->asXml();

出力

<?xml version="1.0"?>
<data><seg id="A1"/><seg id="A5"/><seg id="A29"/><seg id="A30"/></data>

ちなみに、XPathを使用すると特定のノードを選択するのがはるかに簡単になります( SimpleXMLElement-> xpath ):

$segs=$doc->xpath('//seq[@id="A12"]');
if (count($segs)>=1) {
    $seg=$segs[0];
}
// same deletion procedure as above
52
Stefan Gehrig

既存の回答に対する一般的な信念に反して、各Simplexml要素ノードは、それ自体とunset()だけでドキュメントから削除できます。場合のポイントは、SimpleXMLが実際にどのように機能するかを理解する必要があるということだけです。

最初に、削除する要素を見つけます。

list($element) = $doc->xpath('/*/seg[@id="A12"]');

次に、$elementで表される要素を削除します。そのself-referenceの設定を解除します。

unset($element[0]);

これは、任意の要素の最初の要素がSimplexmlの要素自体(自己参照)であるため機能します。これは魔法の性質と関係があり、数値インデックスはリスト内の要素を表し(例:parent-> children)、単一の子でさえそのようなリストです。

数値ではない文字列インデックスは、属性(配列アクセス)または子要素(プロパティアクセス)を表します。

したがって、次のようなプロパティアクセスでの数値のインデックス:

unset($element->{0});

同様に動作します。

当然、そのxpathの例では、かなり単純です(PHP 5.4):

unset($doc->xpath('/*/seg[@id="A12"]')[0][0]);

完全なサンプルコード( デモ ):

<?php
/**
 * Remove a child with a specific attribute, in SimpleXML for PHP
 * @link http://stackoverflow.com/a/16062633/367456
 */

$data=<<<DATA
<data>
    <seg id="A1"/>
    <seg id="A5"/>
    <seg id="A12"/>
    <seg id="A29"/>
    <seg id="A30"/>
</data>
DATA;


$doc = new SimpleXMLElement($data);

unset($doc->xpath('seg[@id="A12"]')[0]->{0});

$doc->asXml('php://output');

出力:

<?xml version="1.0"?>
<data>
    <seg id="A1"/>
    <seg id="A5"/>

    <seg id="A29"/>
    <seg id="A30"/>
</data>
56
hakre

ノードの設定を解除するだけです:

$str = <<<STR
<a>
  <b>
    <c>
    </c>
  </b>
</a>
STR;

$xml = simplexml_load_string($str);
unset($xml –> a –> b –> c); // this would remove node c
echo $xml –> asXML(); // xml document string without node c

このコードは SimpleXMLでノードを削除/削除する方法 から取られました。

23
datasn.io

ステファンの答えは正しいと思います。 (一致するすべてのノードではなく)1つのノードのみを削除する場合、別の例を次に示します。

//Load XML from file (or it could come from a POST, etc.)
$xml = simplexml_load_file('fileName.xml');

//Use XPath to find target node for removal
$target = $xml->xpath("//seg[@id=$uniqueIdToDelete]");

//If target does not exist (already deleted by someone/thing else), halt
if(!$target)
return; //Returns null

//Import simpleXml reference into Dom & do removal (removal occurs in simpleXML object)
$domRef = dom_import_simplexml($target[0]); //Select position 0 in XPath array
$domRef->parentNode->removeChild($domRef);

//Format XML to save indented tree rather than one line and save
$dom = new DOMDocument('1.0');
$dom->preserveWhiteSpace = false;
$dom->formatOutput = true;
$dom->loadXML($xml->asXML());
$dom->save('fileName.xml');

セクションXMLのロード...(最初)とフォーマットXML ...(最後)は、XMLデータの出所と出力の処理内容に応じて、異なるコードに置き換えることができます。ノードを見つけて削除するのは、その間のセクションです。

さらに、ifステートメントは、移動を試みる前にターゲットノードが存在することを確認するためだけにあります。このケースを処理または無視するさまざまな方法を選択できます。

10
Witman

私のためのこの仕事:

$data = '<data>
<seg id="A1"/>
<seg id="A5"/>
<seg id="A12"/>
<seg id="A29"/>
<seg id="A30"/></data>';

$doc = new SimpleXMLElement($data);

$segarr = $doc->seg;

$count = count($segarr);

$j = 0;

for ($i = 0; $i < $count; $i++) {

    if ($segarr[$j]['id'] == 'A12') {
        unset($segarr[$j]);
        $j = $j - 1;
    }
    $j = $j + 1;
}

echo $doc->asXml();
5
sunnyface45

基本のSimpleXMLElementクラスを拡張する場合、次のメソッドを使用できます。

class MyXML extends SimpleXMLElement {

    public function find($xpath) {
        $tmp = $this->xpath($xpath);
        return isset($tmp[0])? $tmp[0]: null;
    }

    public function remove() {
        $dom = dom_import_simplexml($this);
        return $dom->parentNode->removeChild($dom);
    }

}

// Example: removing the <bar> element with id = 1
$foo = new MyXML('<foo><bar id="1"/><bar id="2"/></foo>');
$foo->find('//bar[@id="1"]')->remove();
print $foo->asXML(); // <foo><bar id="2"/></foo>

将来の参照のために、特にドキュメントの正確な構造がわからない場合は、SimpleXMLを使用してノードを削除するのが面倒な場合があります。そのため、SimpleXMLElementを拡張していくつかの便利なメソッドを追加するクラス SimpleDOM を作成しました。

たとえば、deleteNodes()は、XPath式に一致するすべてのノードを削除します。また、属性「id」が「A5」に等しいすべてのノードを削除する場合、必要なことは次のとおりです。

// don't forget to include SimpleDOM.php
include 'SimpleDOM.php';

// use simpledom_load_string() instead of simplexml_load_string()
$data = simpledom_load_string(
    '<data>
        <seg id="A1"/>
        <seg id="A5"/>
        <seg id="A12"/>
        <seg id="A29"/>
        <seg id="A30"/>
    </data>'
);

// and there the magic happens
$data->deleteNodes('//seg[@id="A5"]');
2
Josh Davis

特定の属性値を持つノードを削除/保持するか、属性値の配列に分類するには、次のようにSimpleXMLElementクラスを拡張できます(my GitHub Gist の最新バージョン)。

class SimpleXMLElementExtended extends SimpleXMLElement
{    
    /**
    * Removes or keeps nodes with given attributes
    *
    * @param string $attributeName
    * @param array $attributeValues
    * @param bool $keep TRUE keeps nodes and removes the rest, FALSE removes nodes and keeps the rest 
    * @return integer Number o affected nodes
    *
    * @example: $xml->o->filterAttribute('id', $products_ids); // Keeps only nodes with id attr in $products_ids
    * @see: http://stackoverflow.com/questions/17185959/simplexml-remove-nodes
    */
    public function filterAttribute($attributeName = '', $attributeValues = array(), $keepNodes = TRUE)
    {       
        $nodesToRemove = array();

        foreach($this as $node)
        {
            $attributeValue = (string)$node[$attributeName];

            if ($keepNodes)
            {
                if (!in_array($attributeValue, $attributeValues)) $nodesToRemove[] = $node;
            }
            else
            { 
                if (in_array($attributeValue, $attributeValues)) $nodesToRemove[] = $node;
            }
        }

        $result = count($nodesToRemove);

        foreach ($nodesToRemove as $node) {
            unset($node[0]);
        }

        return $result;
    }
}

$doc XMLを取得したら、<seg id="A12"/>ノード呼び出しを削除できます。

$data='<data>
    <seg id="A1"/>
    <seg id="A5"/>
    <seg id="A12"/>
    <seg id="A29"/>
    <seg id="A30"/>
</data>';

$doc=new SimpleXMLElementExtended($data);
$doc->seg->filterAttribute('id', ['A12'], FALSE);

または、複数の<seg />ノードを削除します。

$doc->seg->filterAttribute('id', ['A1', 'A12', 'A29'], FALSE);

<seg id="A5"/>および<seg id="A30"/>ノードのみを保持し、残りを削除するには:

$doc->seg->filterAttribute('id', ['A5', 'A30'], TRUE);
2

SimpleXmlを使用して子要素を削除する方法があります。コードは要素を探し、何もしません。それ以外の場合は、要素を文字列に追加します。次に、文字列をファイルに書き出します。また、コードは元のファイルを上書きする前にバックアップを保存することに注意してください。

$username = $_GET['delete_account'];
echo "DELETING: ".$username;
$xml = simplexml_load_file("users.xml");

$str = "<?xml version=\"1.0\"?>
<users>";
foreach($xml->children() as $child){
  if($child->getName() == "user") {
      if($username == $child['name']) {
        continue;
    } else {
        $str = $str.$child->asXML();
    }
  }
}
$str = $str."
</users>";
echo $str;

$xml->asXML("users_backup.xml");
$myFile = "users.xml";
$fh = fopen($myFile, 'w') or die("can't open file");
fwrite($fh, $str);
fclose($fh);
1
cud

新しいアイデア:_simple_xml_は配列として機能します。

削除する「配列」のインデックスを検索し、unset()関数を使用してこの配列インデックスを削除します。私の例:

_$pos=$this->xml->getXMLUser();
$i=0; $array_pos=array();
foreach($this->xml->doc->users->usr[$pos]->u_cfg_root->profiles->profile as $profile) {
    if($profile->p_timestamp=='0') { $array_pos[]=$i; }
    $i++;
}
//print_r($array_pos);
for($i=0;$i<count($array_pos);$i++) {
    unset($this->xml->doc->users->usr[$pos]->u_cfg_root->profiles->profile[$array_pos[$i]]);
}
_
1
joan16v

ヘルパー関数に関するアイデアは php.net のDOMに関するコメントの1つからのものであり、unsetの使用に関するアイデアは kavoir.com からのものです。私にとって、このソリューションは最終的に機能しました:

function Myunset($node)
{
 unsetChildren($node);
 $parent = $node->parentNode;
 unset($node);
}

function unsetChildren($node)
{
 while (isset($node->firstChild))
 {
 unsetChildren($node->firstChild);
 unset($node->firstChild);
 }
}

それを使用して:$ xmlはSimpleXmlElementです

Myunset($xml->channel->item[$i]);

結果は$ xmlに保存されるため、変数に代入する心配はありません。

0

RSSフィードのアイテムなど、同様の(一意ではない)子要素のリストをカットする場合は、次のコードを使用できます。

for ( $i = 9999; $i > 10; $i--) {
    unset($xml->xpath('/rss/channel/item['. $i .']')[0]->{0});
}

RSSの末尾を10要素にカットします。私はで削除しようとしました

for ( $i = 10; $i < 9999; $i ++ ) {
    unset($xml->xpath('/rss/channel/item[' . $i . ']')[0]->{0});
}

しかし、それは何らかの形でランダムに機能し、一部の要素のみをカットします。

0
Columbus

私もこの問題に苦労しており、答えはここで提供されているものよりも簡単です。 xpathを使用して検索し、次の方法で設定解除できます。

unset($XML->xpath("NODESNAME[@id='test']")[0]->{0});

このコードは、id属性が「test」である「NODESNAME」という名前のノードを探し、最初の出現を削除します。

$ XML-> saveXML(...);を使用してxmlを保存することを忘れないでください。

0
Ben Yitzhaki

SimpleXMLには要素を削除する詳細な方法はありませんが、PHPのunset()を使用してSimpleXMLから要素をcan削除できます。これを行うための鍵は、目的の要素をターゲットに管理することです。ターゲティングを行うための少なくとも1つの方法は、要素の順序を使用することです。最初に(たとえばループを使用して)削除する要素の順序番号を見つけてから、要素を削除します。

_$target = false;
$i = 0;
foreach ($xml->seg as $s) {
  if ($s['id']=='A12') { $target = $i; break; }
  $i++;
}
if ($target !== false) {
  unset($xml->seg[$target]);
}
_

ターゲット項目の順序番号を配列に保存することにより、これで複数の要素を削除することもできます。アイテムを削除すると、その後に来るアイテムの順序番号が自然に減少するため、逆の順序(array_reverse($targets))で削除することを忘れないでください。

確かに、それはちょっとしたやり直しですが、うまくいくようです。

0
Ilari Kajaste

FluidXML を使用すると、XPathを使用して削除する要素を選択できます。

$doc = fluidify($doc);

$doc->remove('//*[@id="A12"]');

https://github.com/servo-php/fluidxml


XPath //*[@id="A12"]の意味:

  • ドキュメントの任意のポイント(//
  • すべてのノード(*
  • 属性idA12[@id="A12"])と等しい場合。
0
Daniele Orlando

Gerryと同じ致命的なエラーが発生し、DOMに詳しくないので、次のように実行することにしました。

$item = $xml->xpath("//seg[@id='A12']");
$page = $xml->xpath("/data");
$id = "A12";

if (  count($item)  &&  count($page) ) {
    $item = $item[0];
    $page = $page[0];

     // find the numerical index within ->children().
    $ch = $page->children();
    $ch_as_array = (array) $ch;

    if (  count($ch_as_array)  &&  isset($ch_as_array['seg'])  ) {
        $ch_as_array = $ch_as_array['seg'];
        $index_in_array = array_search($item, $ch_as_array);
        if (  ($index_in_array !== false)
          &&  ($index_in_array !== null)
          &&  isset($ch[$index_in_array])
          &&  ($ch[$index_in_array]['id'] == $id)  ) {

             // delete it!
            unset($ch[$index_in_array]);

            echo "<pre>"; var_dump($xml); echo "</pre>";
        }
    }  // end of ( if xml object successfully converted to array )
}  // end of ( valid item  AND  section )
0
WoodrowShigeru