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



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

我的问题(可能是错误的,因为处理XML数据的新手和PHP的新手(是"foreach"循环。当我打开脚本时,它会为我返回以下数据:

1
Array
Array
Array
2
Array
Array
Array
3

当然,在最终版本中(当我开始工作时(,我没有意义打印/回显这些值,因为这些变量将被发布到我的数据库中。因此,请记住,我现在正在这样做,了解代码以及问题可能出在哪里。

$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 的更新日期的数据。

<?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 :)

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

提前感谢您的帮助! <3

Edit1(对于XML数据,即使有它的链接(:

<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问题(

<?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));
}
?>

以下代码正在制造此错误,我已经检查了数据库设置是否正确 100%,因此问题不可能与此有关,而且我还尝试用普通文本(例如"testi"或"aarni"(替换 php 变量,但这也没有用。

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(修复了我之前遇到的所有问题并在此处添加整个代码,因此将来如果有人遇到类似的问题,他们可以复制/粘贴它(

<?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脚本时,我的数组的外观/显示方式:

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 数据

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

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

$xmlDate = (string)$xmlData->xpath('BsWfs:Time')[0];

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

$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)
]
);
}

输出:

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)
}

最新更新