XML creation and manipulation can be quite complex and so, PHP has provided several methods of handling XML. Each method has a varying degree of complexity, but perhaps the simplest of them all, is the appropriately named extension, SimpleXML
This tutorial requires a fundamental understanding of XML structures. If you lack this, it is highly recommended brushing up on XML and how it can be used.
This is an example of any XML file that may be found for user data in an address book, or contact list or other data source. Note that in the content node, there are several sub nodes containing the phone, url, and email for the user. SimpleXML will allow traversing this XML structure in one simple pass as will be demonstrated here.
Loading an XML file with simpleXML is achieved with a single function call to the simplexml_load_file() function. Once an XML file is loaded into SimpleXML, the data can be accessed.
if( ! $xml = simplexml_load_file('address.xml') )
{
echo 'unable to load XML file';
}
else
{
echo 'XML file loaded successfully';
} ?>
Similarly, if the XML data is in a string, rather than a file, the simplexml_load_string() function can be used in the same fashion. In this example, the XML is shortened a little for the sake of sanity, but the functionality remains the same
$xml_string = '
2 Good St
Campbelltown
Australia
1234 1234
http://example.com
pamela@example.com
';
if( ! $xml = simplexml_load_string( $xml_string ) )
{
echo 'Unable to load XML string';
}
else
{
echo 'XML String loaded successfully';
} ?>
Regardless of which function is used to load the XML data, the resulting xml object is the same.
; In the above script, the XML file was loaded using the simplexml_load_file() function. With the file loaded successfully, the data in each of the nodes can be accessed, and displayed, or assigned to variables, or manipulated any way program requires.
Each node is transformed into a corresponding variable name, which contains the data within that node. In the case of nested variables such as those nodes in the contact section of the XML, an easy path is provided to access these values, as if in a tree structure.
if( ! $xml = simplexml_load_file('address.xml') )
{
echo 'unable to load XML file';
}
else
{
foreach( $xml as $user )
{
echo 'Firstname: '.$user->firstname.'
';
echo 'Surname: '.$user->surname.'
';
echo 'Address: '.$user->address.'
';
echo 'City: '.$user->city.'
';
echo 'Country: '.$user->country.'
';
echo 'Email: '.$user->contact->phone.'
';
echo 'Email: '.$user->contact->url.'
';
echo 'Email: '.$user->contact->email.'
';
}
} ?>
As seen earlier, loading XML from a file or a string is simply a matter of a single call to the simple_load_string() or the simplexml_load_file() functions. SimpleXML also provides a method to import XML which has been loaded via the DOM extension.
$xml_string = '
2 Good St
Campbelltown
Australia
1234 1234
http://example.com
pamela@example.com
';
/*** a new DOM object ***/
$dom = new DOMDocument;
/*** load the XML string ***/
$dom->loadXML( $xml_string );
$sxe = simplexml_import_dom($dom);
echo $sxe->user[0]->surname;
?>
The script above loads the XML string via the DomDocument::loadXML method. This object can now be imported directly into SimpleXML with the simplexml_import_dom() function. The result set is an array of objects, each containing a user node. Each node within the array may accessed with the array key, in this example, the array key is zero.
It is at this point, a limitation of SimpleXML is reached. If the XML above is compared to the initial XML in this tutorial, it is plain to see the contact details are missing. The limitation of is that SimpleXML cannot add nodes. To achieve this, the DOM extension needs to be used, remember, this is SimpleXML, not FullyFeaturedXML as you get when using the DOM extension. Whilst SimpleXML provides a fast and efficient method for reading XML, its abilities to create detailed XML trees is somewhat limited.
Having said that it cannot be done, here is how to do it. Rather than adding a whole node, the tree can be created by adding children to each element. Each element represents a SimpleXML object and can be used to add a child. By adding each child to a parent element, the node is gradually created.
try
{ /*** a new dom object ***/
$dom = new domDocument;
/*** make the output tidy ***/
$dom->formatOutput = true;
/*** create the root element ***/ $root = $dom->appendChild($dom->createElement( "users" ));
/*** create the simple xml element ***/
$sxe = simplexml_import_dom( $dom );
/*** add a user node ***/ $user = $sxe->addchild("user");
/*** add a firstname element ***/ $user->addChild("firstname", "John");
/*** add a surname element ***/ $user->addChild("surname", "Brady");
/*** add address element ***/ $user->addChild("address", "1 Bunch St");
/*** add the city element ***/
$user->addChild("city", "Downtown");
/*** add the country ***/
$user->addChild("country", "America");
$contact = $user->addChild("contact");
$contact->addChild("phone", "4444 4444");
$contact->addChild("url", "http://phpro.org");
$contact->addChild("email", "brady@bunch.example.com");
echo $sxe->asXML();
}
catch( Exception $e )
{
echo $e->getMessage();
}
?>
In the XML created so far, the phone element has simply been a number. But, it does not tell us what type of phone it is. Is it a land line? Is it a cell/mobile or fax or satellite phone?
An attribute named type can be added as an attribute for the phone element. In SimpleXML, attributes are added with the addAttribute() method. In this following example, an attribute named type is added with a value of mobile.
try
{ /*** a new dom object ***/
$dom = new domDocument;
/*** make the output tidy ***/
$dom->formatOutput = true;
/*** create the root element ***/ $root = $dom->appendChild($dom->createElement( "users" ));
/*** create the simple xml element ***/
$sxe = simplexml_import_dom( $dom );
/*** add a user node ***/ $user = $sxe->addchild("user");
/*** add a firstname element ***/ $user->addChild("firstname", "John");
/*** add a surname element ***/ $user->addChild("surname", "Brady");
/*** add address element ***/ $user->addChild("address", "1 Bunch St");
/*** add the city element ***/
$user->addChild("city", "Downtown");
/*** add the country ***/
$user->addChild("country", "America");
/*** add the contact element ***/ $contact = $user->addChild("contact");
/*** add children to the contact element ***/
$phone = $contact->addChild("phone", "4444 4444");
/*** add an attribute to the phone element ***/
$phone->addAttribute("type", "mobile");
/*** more children for the contact element ***/
$contact->addChild("url", "http://phpro.org");
$contact->addChild("email", "brady@bunch.example.com");
/*** show the xml ***/ echo $sxe->asXML();
}
catch( Exception $e )
{
echo $e->getMessage();
}
?>
Attributes can be accessed in much the same way as seen with accessing elements in the previous section. Once again, an appropriately named function is provided by SimpleXML. The function attributes() makes this step easy. In the address.xml file, only the phone element has an attribute, so we can direct our query directly at that element to retrieve an attributes it has.
/*** create a SimpleXML object ***/ if( ! $xml = simplexml_load_file("address.xml") )
{
echo "Unable to load XML file";
}
else
{ $i = 0;
/*** loop over the elements ***/
foreach($xml as $node)
{
echo $xml->user[$i]->contact->phone->attributes().'
';
$i++;
}
} ?>
The above query will fetch all the attributes of the phone element and show their values.
mobile
landline
mobile
fax
XPath provides a standardized method to query XML regardless of the language being used to program with. To utilize xpath in SimpleXML, a single function is all that is required, amazingly named xpath(). Lets begin with a look at how xpath can be used to get the firstnames.
/*** create a SimpleXML object ***/ if( ! $xml = simplexml_load_file("address.xml") )
{
echo "Unable to load XML file";
}
else
{
/*** show the firstname element from all nodes ***/
print_r($xml->xpath("/users/user/firstname"));
} ?>
The resulting array shows an array of simpleXML objects, each containing a single element, the firstname.Searching a path with XPath breaks down the path into its component elements and extracts the values. A similar search to the previous could therefore be expressed like this.
/*** create a SimpleXML object ***/ if( ! $xml = simplexml_load_file("address.xml") )
{
echo "Unable to load XML file";
}
else
{
/*** show the firstname element from all nodes ***/
print_r($xml->xpath("//firstname"));
} ?>
The resulting array if objects is identical to the previous method of access the firstname elements. This functionality can be further extended by searching for a node by an elements value. For example, if all the information about Sheila was required, the tag name and value can be supplied, and SimpleXML will traverse the XML tree to retrieve the node data it belongs to.
/*** create a SimpleXML object ***/
if( ! $xml = simplexml_load_file("address.xml") )
{
echo "Unable to load XML file";
}
else
{
/*** show the firstname element from all nodes ***/
print_r($xml->xpath("//*[firstname='Sheila']"));
} ?>
The resulting array contains all the information about Sheila.With this information, it is now just a short step to retrieve the value of just the email element, based on the firstname elements value.
error_reporting(E_ALL);
/*** create a SimpleXML object ***/ if( ! $xml = simplexml_load_file("address.xml") )
{
echo "Unable to load XML file";
}
else
{
/*** show the firstname element from all nodes ***/
$info = $xml->xpath("//*[firstname='Sheila']");
/*** fetch the email address ***/
echo $info[0]->contact->email;
}
?>
This results simply in the email address being returned.
pamela@example.com
All this creating, manipulating and reading of XML is all well and good. The ability to create an array of objects from an xpath query is quite clever, however, is not truly in a machine readable format. Their needs to be a method to create an XML representation of the object. To this end, the asXML() method is provided for just this purpose. In this example, the XPath query for Sheila is saved as XML, rather than the array of objects and values.
/*** create a SimpleXML object ***/ if( ! $xml = simplexml_load_file("address.xml") )
{
echo "Unable to load XML file";
}
else
{
/*** show the firstname element from all nodes ***/
$info = $xml->xpath("//*[firstname='Sheila']");
/*** initialize the string ***/ $xml_string = '';
/*** loop over the results ***/ while(list( , $node) = each($info))
{
$xml_string .= $node->asXML(); //
/*** output the xml ***/ echo $xml_string;
} ?>
Copyright © 2011 - All Rights Reserved - Softron.in
Template by Softron Technology