我一直在互联网上寻找,但没有找到答案,所以我在这里。
我有一个返回记录的 PL/SQL 函数:
create or replace package pck_test is
TYPE coord_geo is record (coord_x float, coord_y float);
function ArrondiGeo(coord_x float, coord_y float) return coord_geo;
end pck_test;
/
create or replace package body pck_test is
FUNCTION ArrondiGeo(coord_x FLOAT,coord_y FLOAT) RETURN coord_geo
IS
temp_x FLOAT(24);
temp_y FLOAT(24);
rec_coord coord_geo;
BEGIN
temp_x := ROUND(coord_x,4)/5;
temp_y := ROUND(coord_y,4)/5;
temp_x := ROUND((temp_x*5),3);
temp_y := ROUND((temp_y*5),3);
rec_coord.coord_x := temp_x;
rec_coord.coord_y := temp_y;
RETURN rec_coord;
END;
END pck_test;
/
我想在PHP函数中使用它,但我真的不知道如何...
我试过这个,但它不起作用:
public function get_tronc($x,$y)
{
$q = oci_parse($this->_db, 'begin :r := pck_test.ArrondiGeo(:x,:y); end;');
oci_bind_by_name($q, ':x', $x);
oci_bind_by_name($q, ':y', $y);
oci_bind_by_name($q, ':r', $r);
oci_execute($q);
return $r;
}
错误是:
Warning: oci_execute(): ORA-06550: line 1, column 13: PLS-00382: expression is of wrong type ORA-06550: line 1, column 7: PL/SQL: Statement ignored in /users/info/il3/jboeglin/Bureau/BDD/site/models/userManager.php on line 77
所以这是一个自我解释的错误,但我仍然不知道如何使用它。
谢谢。
对于命名数据类型,可能需要绑定指定SQLT_NTY
类型的参数:
$r = oci_new_collection($this->db, 'COORD_GEO');
oci_bind_by_name($q, ':r', $r, -1, SQLT_NTY);
...
oci_execute($q);
// do whatever you need with your data
$data = $elem = $collection->getElem(1);
// then discard it
$r->free();
有关详细信息,请参阅oci_bind_by_name手册。
对于普通的PL/SQL记录,您可能不走运:根据Oracle的文档,您不能使用OCI来获取记录。截至甲骨文 11g:
以下两种类型是 PL/SQL 的内部类型,OCI 不能作为值返回:
- 布尔值,
SQLT_BOL
- 记录,
SQLT_REC
如果这是正确的(我不是普通的PHP用户(,你可能不得不在PL/SQ级别包装你的函数:
- 要么,在具有所需数量的
OUT
参数的过程中; - 或者,根据您的需要,在表格函数中 。
为了完整起见,请注意,Oracle 12c + OCI8 2.0.7 删除了以前不允许返回布尔值的限制。
谢谢你的回答,我试过SQLT_NTY但它不起作用,可能是因为 OCI 无法像你说的那样返回记录,...我将尝试查看表函数。
编辑:我终于用了一个程序
create or replace package pck_test is
PROCEDURE ArrondiGeo(coord_x IN FLOAT,coord_y IN FLOAT,temp_x OUT FLOAT,temp_y OUT FLOAT);
end pck_test;
/
create or replace package body pck_test is
PROCEDURE ArrondiGeo(coord_x IN FLOAT,coord_y IN FLOAT,temp_x OUT FLOAT,temp_y OUT FLOAT)
IS
BEGIN
temp_x := ROUND(coord_x,4)/5;
temp_y := ROUND(coord_y,4)/5;
temp_x := ROUND((temp_x*5),3);
temp_y := ROUND((temp_y*5),3);
END;
END pck_test;
/
而这个PHP函数:
public function get_tronc($x,$y)
{
$q = oci_parse($this->_db, 'begin pck_test.ArrondiGeo(:x,:y,:xm,:ym); end;');
oci_bind_by_name($q, ':x', $x);
oci_bind_by_name($q, ':y', $y);
oci_bind_by_name($q, ':xm', $xm,40);
oci_bind_by_name($q, ':ym', $ym,40);
oci_execute($q);
$r=array($xm,$ym);
return $r;
}