[英]Xpath in PHP with OTA standards
我有关于在PHP中使用Xpath的基本知识,但我遇到了一些特定情况的麻烦,我认为问题出在标准中。
这是XML的片段,它基于OTA标准 :
<SendHotelResResult xmlns:a="http://schemas/Models/OTA" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
<a:RoomRates>
<a:RoomRate>
<a:EffectiveDate>2015-11-13T00:00:00</a:EffectiveDate>
<a:ExpireDate>2015-11-15T00:00:00</a:ExpireDate>
<a:RatePlanID>25</a:RatePlanID>
<a:RatesType>
<a:Rates>
<a:Rate>
<a:AgeQualifyingCode i:nil="true"/>
<a:EffectiveDate>2015-11-13T00:00:00</a:EffectiveDate>
<a:Total>
<a:AmountAfterTax>0</a:AmountAfterTax>
<a:AmountBeforeTax>260.00</a:AmountBeforeTax>
<a:CurrencyCode>EUR</a:CurrencyCode>
</a:Total>
</a:Rate>
<a:Rate>
<a:AgeQualifyingCode i:nil="true"/>
<a:EffectiveDate>2015-11-14T00:00:00</a:EffectiveDate>
<a:Total>
<a:AmountAfterTax>0</a:AmountAfterTax>
<a:AmountBeforeTax>260.00</a:AmountBeforeTax>
<a:CurrencyCode>EUR</a:CurrencyCode>
</a:Total>
</a:Rate>
</a:Rates>
</a:RatesType>
<a:RoomID>52</a:RoomID>
<a:Total>
<a:AmountAfterTax>546.00</a:AmountAfterTax>
<a:AmountBeforeTax>520.00</a:AmountBeforeTax>
<a:CurrencyCode>EUR</a:CurrencyCode>
</a:Total>
</a:RoomRate>
</a:RoomRates>
</SendHotelRes>
我想要的 :
<RoomID>
获取特定的<RoomRate>
标记。 <Total>
标记。 我不想要<Rate>
标签内的<Rate>
<Total>
<Rate>
标签。 这就是我使用xpath而不是简单的getElementsByTagName('Total')
。 我不知道OTA标准是否有一些区分Total标签的方法。 我的尝试到现在为止:
$dom = new DOMDocument();
$response = $dom->load($xmlSendHotelRes);
$roomID = '52';
$roomRatesTag = $response->getElementsByTagName('RoomRates')->item(0);
$prefix = $roomRatesTag->prefix;
$namespace = $roomRatesTag->lookupNamespaceURI($prefix);
$xpath = new DOMXpath($dom);
$xpath->registerNamespace($prefix, $namespace);
$roomRateTotal = $xpath->query("//RoomRate[RoomID=$roomID]/Total", $roomRatesTag, true);
我已尝试使用和不使用$roomRatesTag
作为上下文以及其他表达式,例如: ./RoomRate[RoomID=$roomID]/Total
//RoomRate[RoomID=$roomID]/Total
, //RoomRate/[RoomID=$roomID]/Total
//RoomRate[RoomID=$roomID]/Total
, //RoomRate/[RoomID=$roomID]/Total
, //RoomRate[RoomID=$roomID]/Total
和//RoomRate/RoomID[text() = $roomID]/../Total
但其中任何一个都有效。
实际上,甚至$roomRate = $xpath->query("//RoomRate");
返回一个空的DOMNodeList,所以,我不知道我做错了什么,我正在考虑标准中的问题,在不同的地方有2个相同的标签,虽然这没有多大意义。
还有其他一些我需要尝试的表达方式吗?
您正在从文档中获取命名空间。
$prefix = $roomRatesTag->prefix;
$namespace = $roomRatesTag->lookupNamespaceURI($prefix);
但这不是必要的,也不是一个好主意。 您知道该文档使用OTA,因此您知道命名空间是http://schemas/Models/OTA
。
前缀只是实际命名空间值的别名,以下3个XML示例都解析为节点{http://schemas/Models/OTA}RoomRates
<a:RoomRates xmlns:a="http://schemas/Models/OTA"/>
<ota:RoomRates xmlns:ota="http://schemas/Models/OTA"/>
<RoomRates xmlns="http://schemas/Models/OTA"/>
您的Api必须在命名空间内查找节点。
一种可能性是使用*NS
(名称空间感知)方法。
$response->getElementsByTagNameNS('http://schemas/Models/OTA', 'RoomRates')->item(0);
另一种是使用Xpath并为命名空间注册前缀。 这可以是文档中的前缀,也可以是不同的前缀。
$document = new DOMDocument();
$document->load($xmlSendHotelRes);
$xpath = new DOMXpath($document);
$xpath->registerNamespace('ota', 'http://schemas/Models/OTA');
var_dump(
$xpath->evaluate(
'string(//ota:RoomRates/ota:RoomRate[ota:RoomID=$roomID]/ota:Total)')
)
);
对于位置路径, DOMXpath::evaluate()
将返回DOMNodeList
但是使用string()
将第一个找到的节点转换为字符串并返回它。
你需要使用一个前缀(你注册的),我想你想用.//
开始你的路径而不是//
如果你想要相对于上下文节点进行搜索,那么试试".//a:RoomRate[a:RoomID=$roomID]/a:Total"
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.