首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >用正确的方式使用xpath,使用foreach函数?

用正确的方式使用xpath,使用foreach函数?
EN

Stack Overflow用户
提问于 2020-06-25 06:58:55
回答 1查看 487关注 0票数 0

我很难从xml (外部url)文件解析到php。我想要完成的是,我的.php脚本加载url并从中获取特定的数据,->将它们解析为变量,然后使用这些变量将它们发布到我的数据库(MySQL)。

我的问题(可能弄错了,因为处理XML-数据是新的,PHP是新的)是使用了"foreach“循环。打开脚本时,它会为我返回以下数据:

代码语言:javascript
复制
1
Array
Array
Array
2
Array
Array
Array
3

当然,在最终版本中(当我完成这项工作时),我没有必要打印/回显这些值,因为这些变量将被张贴到我的数据库中。因此,我现在正在做这件事,了解代码以及问题可能会发生在哪里。

代码语言:javascript
复制
$url = 'http://opendata.fmi.fi/wfs?service=WFS&version=2.0.0&request=getFeature&storedquery_id=fmi::observations::weather::daily::simple&place=tammela&starttime='.$first_image_date.'T00:00:00Z&endtime='.$last_image_date.'T01:00:00Z&parameters=tmin,tday,tmax&';

Original xml url = http://opendata.fmi.fi/wfs?service=WFS&version=2.0.0&request=getFeature&storedquery_id=fmi::observations::weather::daily::simple&place=tammela&starttime=2020-05-24T00:00:00Z&endtime=2020-05-31T01:00:00Z&parameters=tmin,tday,tmax&

下面的变量是我从哪里获取XML数据的。唯一的改变(每当使用php )是$first_image_date$last_image_date.所以我的基本想法是,我的服务器上有图像,现在我有9种不同的图像(它们都是.jpg格式),.php代码从这些图像中提取日期,并与XML-数据匹配。

如果新图像被上传/传输到我的服务器上,这意味着$url变量将获取具有链接到url的更新日期的数据。

代码语言:javascript
复制
<?php

$folder_location = "AarniAnsa/*/*/";
$jpg = glob("" . $folder_location . "*.jpg");

$first_image = reset($jpg);
$last_image = end($jpg);

ob_start();
print end(explode('@',$first_image,-2));
$first_image_date = ob_get_contents();
ob_end_clean();

ob_start();
print end(explode('@',$last_image,-2));
$last_image_date = ob_get_contents();
ob_end_clean();

define('DB_SERVER', '***');
define('DB_USERNAME', '***');
define('DB_PASSWORD', '***');
define('DB_NAME', '***');

$link = mysqli_connect(DB_SERVER, DB_USERNAME, DB_PASSWORD, DB_NAME);

if($link === false) {
  die("ERROR: Could not connect. " . mysqli_connect_error());
}

// Alla oleva linkki antaa pelkästään meille tmin (Alin lämpötila), tday (Keskilämpötila) sekä tmax (Ylin lämpötila) arvot. Muita muuttujia emme tarvitse.
$url = 'http://opendata.fmi.fi/wfs?service=WFS&version=2.0.0&request=getFeature&storedquery_id=fmi::observations::weather::daily::simple&place=tammela&starttime='.$first_image_date.'T00:00:00Z&endtime='.$last_image_date.'T01:00:00Z&parameters=tmin,tday,tmax&';
$xml = simplexml_load_file($url);
$xmlID = 0;

$xml->registerXPathNamespace('wfs', 'http://www.opengis.net/wfs/2.0');
$xml->registerXPathNamespace('BsWfs', 'http://xml.fmi.fi/schema/wfs/2.0');

foreach ($xml->xpath('//wfs:member/BsWfs:BsWfsElement') as $xmlData) {
  $xmlID++;
  echo $xmlID;
  echo "<br>";
  $xmlDate = $xmlData->xpath('BsWfs:Time');
  echo $xmlDate;
  echo "<br>";
  $xmlName = $xmlData->xpath('BsWfs:ParameterName');
  echo $xmlName;
  echo "<br>";
  $xmlValue = $xmlData->xpath('BsWfs:ParameterValue');
  echo $xmlValue;
  echo "<br>";


  $query = "INSERT INTO Kaavio(ID,Päivämäärä,Parametri,Arvo) VALUES ('" . $xmlID . "','" . $xmlDate . "','" . $xmlName . "','" . $xmlValue . "')";

  $result = mysqli_query($link, $query);
}

 ?>

现在我使用的是"xpath",因为据我所知(到目前为止我学到的)是,如果我使用SimpleXML选项,它就不喜欢带有“前缀”的数据。这个XML数据包含在每个元素上,一些“前缀”标记。这就是为什么我使用"registerXPathNamespace“函数,但还没有解决我的问题。

如果您想知道我为什么使用以下代码,是因为我所有的图像格式都是这样的:@AnsaID-20@2020-05-31@19:55:36@.jpg :)

代码语言:javascript
复制
ob_start();
print end(explode('@',$first_image,-2));
$first_image_date = ob_get_contents();
ob_end_clean();

谢谢你提前提供帮助!

Edit1 (用于XML-data,尽管它有链接):

代码语言:javascript
复制
<wfs:FeatureCollection xmlns:wfs="http://www.opengis.net/wfs/2.0" xmlns:gml="http://www.opengis.net/gml/3.2" xmlns:BsWfs="http://xml.fmi.fi/schema/wfs/2.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" timeStamp="2020-06-24T20:12:30Z" numberReturned="24" numberMatched="24" xsi:schemaLocation="http://www.opengis.net/wfs/2.0 http://schemas.opengis.net/wfs/2.0/wfs.xsd http://xml.fmi.fi/schema/wfs/2.0 http://xml.fmi.fi/schema/wfs/2.0/fmi_wfs_simplefeature.xsd">
<wfs:member>
<BsWfs:BsWfsElement gml:id="BsWfsElement.1.1.1">
<BsWfs:Location>
<gml:Point gml:id="BsWfsElementP.1.1.1" srsDimension="2" srsName="http://www.opengis.net/def/crs/EPSG/0/4258">
<gml:pos>60.81401 23.49761 </gml:pos>
</gml:Point>
</BsWfs:Location>
<BsWfs:Time>2020-05-24T00:00:00Z</BsWfs:Time>
<BsWfs:ParameterName>tmin</BsWfs:ParameterName>
<BsWfs:ParameterValue>8.2</BsWfs:ParameterValue>
</BsWfs:BsWfsElement>
</wfs:member>
<wfs:member>
<BsWfs:BsWfsElement gml:id="BsWfsElement.1.1.2">
<BsWfs:Location>
<gml:Point gml:id="BsWfsElementP.1.1.2" srsDimension="2" srsName="http://www.opengis.net/def/crs/EPSG/0/4258">
<gml:pos>60.81401 23.49761 </gml:pos>
</gml:Point>
</BsWfs:Location>
<BsWfs:Time>2020-05-24T00:00:00Z</BsWfs:Time>
<BsWfs:ParameterName>tday</BsWfs:ParameterName>
<BsWfs:ParameterValue>11.1</BsWfs:ParameterValue>
</BsWfs:BsWfsElement>
</wfs:member>
<wfs:member>
<BsWfs:BsWfsElement gml:id="BsWfsElement.1.1.3">
<BsWfs:Location>
<gml:Point gml:id="BsWfsElementP.1.1.3" srsDimension="2" srsName="http://www.opengis.net/def/crs/EPSG/0/4258">
<gml:pos>60.81401 23.49761 </gml:pos>
</gml:Point>
</BsWfs:Location>
<BsWfs:Time>2020-05-24T00:00:00Z</BsWfs:Time>
<BsWfs:ParameterName>tmax</BsWfs:ParameterName>
<BsWfs:ParameterValue>14.3</BsWfs:ParameterValue>
</BsWfs:BsWfsElement>
</wfs:member>

Edit2 (添加“部分工作”代码,但存在MySQL问题)

代码语言:javascript
复制
<?php

$folder_location = "AarniAnsa/*/*/";
$jpg = glob("" . $folder_location . "*.jpg");

$first_image = reset($jpg);
$last_image = end($jpg);

ob_start();
print end(explode('@',$first_image,-2));
$first_image_date = ob_get_contents();
ob_end_clean();

ob_start();
print end(explode('@',$last_image,-2));
$last_image_date = ob_get_contents();
ob_end_clean();

define('DB_SERVER', '***');
define('DB_USERNAME', '***');
define('DB_PASSWORD', '***');
define('DB_NAME', '***');

$link = mysqli_connect(DB_SERVER, DB_USERNAME, DB_PASSWORD, DB_NAME);

if($link === false) {
  die("ERROR: Could not connect. " . mysqli_connect_error());
}

// Alla oleva linkki antaa pelkästään meille tmin (Alin lämpötila), tday (Keskilämpötila) sekä tmax (Ylin lämpötila) arvot. Muita muuttujia emme tarvitse.
$url = 'http://opendata.fmi.fi/wfs?service=WFS&version=2.0.0&request=getFeature&storedquery_id=fmi::observations::weather::daily::simple&place=tammela&starttime='.$first_image_date.'T00:00:00Z&endtime='.$last_image_date.'T01:00:00Z&parameters=tmin,tday,tmax&';
$xml = simplexml_load_file($url);
$xmlID = 0;

$xml->registerXPathNamespace('wfs', 'http://www.opengis.net/wfs/2.0');
$xml->registerXPathNamespace('BsWfs', 'http://xml.fmi.fi/schema/wfs/2.0');

foreach ($xml->xpath('//wfs:member/BsWfs:BsWfsElement') as $xmlData) {
  $xmlID++;
  echo $xmlID;
  echo "<br>";
  $xmlDate = (string)$xmlData->xpath('BsWfs:Time')[0];
  echo $xmlDate;
  echo "<br>";
  $xmlName = (string)$xmlData->xpath('BsWfs:ParameterName')[0];
  echo $xmlName;
  echo "<br>";
  $xmlValue = (string)$xmlData->xpath('BsWfs:ParameterValue')[0];
  echo $xmlValue;
  echo "<br>";

  $sql = "INSERT INTO Kaavio(Päivämäärä, Parametri, Arvo) VALUES('$xmlDate', '$xmlName', '$xmlValue')";
  mysqli_query($link, $sql) or die(mysqli_error($link));

}

 ?>

下面的代码造成了这个错误,并且我已经检查了数据库设置是否正确,所以问题不会出现,而且我还尝试用普通文本(如'testi‘或'aarni')替换php变量,但这也不起作用。

代码语言:javascript
复制
1
2020-05-24T00:00:00Z
tmin
8.2
You have an error in your SQL syntax; check the manual that corresponds to your MariaDB server version for the right syntax to use near '�ivämäärä, Parametri, Arvo) VALUES('2020-05-24T00:00:00Z', 'tmin...' at line 1

Edit3 (修复了我以前遇到的所有问题,并在这里添加了整个代码,所以将来如果有人有类似的问题,他们只需复制/粘贴它)

代码语言:javascript
复制
<?php

$folder_location = "AarniAnsa/*/*/";
$jpg = glob("" . $folder_location . "*.jpg");

$first_image = reset($jpg);
$last_image = end($jpg);

ob_start();
print end(explode('@',$first_image,-2));
$first_image_date = ob_get_contents();
ob_end_clean();

ob_start();
print end(explode('@',$last_image,-2));
$last_image_date = ob_get_contents();
ob_end_clean();

define('DB_SERVER', '***');
define('DB_USERNAME', '***');
define('DB_PASSWORD', '***');
define('DB_NAME', '***');

$link = mysqli_connect(DB_SERVER, DB_USERNAME, DB_PASSWORD, DB_NAME);

if($link === false) {
  die("ERROR: Could not connect. " . mysqli_connect_error());
}

// Alla oleva linkki antaa pelkästään meille tmin (Alin lämpötila), tday (Keskilämpötila) sekä tmax (Ylin lämpötila) arvot. Muita muuttujia emme tarvitse.
$url = 'http://opendata.fmi.fi/wfs?service=WFS&version=2.0.0&request=getFeature&storedquery_id=fmi::observations::weather::daily::simple&place=tammela&starttime='.$first_image_date.'T00:00:00Z&endtime='.$last_image_date.'T01:00:00Z&parameters=tmin,tday,tmax&';
$xml = simplexml_load_file($url);

$xml->registerXPathNamespace('wfs', 'http://www.opengis.net/wfs/2.0');
$xml->registerXPathNamespace('BsWfs', 'http://xml.fmi.fi/schema/wfs/2.0');

foreach ($xml->xpath('//wfs:member/BsWfs:BsWfsElement') as $xmlData) {
  $xmlDate = (string)$xmlData->xpath('BsWfs:Time')[0];
  $xmlDate = substr($xmlDate,0,10);
  echo $xmlDate;
  echo "<br>";
  $xmlName = (string)$xmlData->xpath('BsWfs:ParameterName')[0];
  echo $xmlName;
  echo "<br>";
  $xmlValue = (string)$xmlData->xpath('BsWfs:ParameterValue')[0];
  echo $xmlValue;
  echo "<br>";

  $sql = "INSERT INTO `Kaavio` (`date`, `parameter`, `value`) VALUES('$xmlDate', '$xmlName', '$xmlValue')";
  $result = mysqli_query($link, $sql);
}

 ?>

现在,当我打开.php脚本时,数组的外观/显示方式如下:

代码语言:javascript
复制
2020-05-24
tmin
8.2
2020-05-24
tday
11.1
2020-05-24
tmax
14.3
2020-05-25
tmin
8.0
2020-05-25
tday
13.0
2020-05-25
tmax
18.2
2020-05-26
tmin
4.6
2020-05-26
tday
14.9

这就是我的phpMyAdmin的样子(使用MariaDB):

解析到我的数据库的XML数据:)

EN

回答 1

Stack Overflow用户

发布于 2020-06-26 09:36:21

您需要在调用xpath()方法的任何元素上注册名称空间。在这种情况下,您应该在循环中的http://xml.fmi.fi/schema/wfs/2.0变量上注册$xmlData

此外,SimpleXMLElement::xpath()返回一个SimpleXMLElement实例数组。要获取单个值引用第一个元素并强制转换它:

代码语言:javascript
复制
$xmlDate = (string)$xmlData->xpath('BsWfs:Time')[0];

只有DOMxpath::evaluate()才能直接返回标量值。类型强制转换是表达式的一部分。下面是一个示例:

代码语言:javascript
复制
$document = new DOMDocument();
$document->loadXML(getXMLString());
$xpath = new DOMXpath($document);
$xpath->registerNamespace('wfs', 'http://www.opengis.net/wfs/2.0');
$xpath->registerNamespace('BsWfs', 'http://xml.fmi.fi/schema/wfs/2.0');

foreach ($xpath->evaluate('//wfs:member/BsWfs:BsWfsElement') as $index => $element) {
    var_dump(
        [
             'id' => $index + 1,
             'date' => $xpath->evaluate('string(BsWfs:Time)', $element),
             'name' => $xpath->evaluate('string(BsWfs:ParameterName)', $element),
             'value' => $xpath->evaluate('number(BsWfs:ParameterValue)', $element)
        ]
    );
}

输出:

代码语言:javascript
复制
array(4) {
  ["id"]=>
  int(1)
  ["date"]=>
  string(20) "2020-05-24T00:00:00Z"
  ["name"]=>
  string(4) "tmin"
  ["value"]=>
  float(8.2)
}
array(4) {
  ["id"]=>
  int(2)
  ["date"]=>
  string(20) "2020-05-24T00:00:00Z"
  ["name"]=>
  string(4) "tday"
  ["value"]=>
  float(11.1)
}
array(4) {
  ["id"]=>
  int(3)
  ["date"]=>
  string(20) "2020-05-24T00:00:00Z"
  ["name"]=>
  string(4) "tmax"
  ["value"]=>
  float(14.3)
}
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/62569576

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档