Могу ли я вернуть значения в PHP из анонимного блока PL/SQL?

Я использую PHP и OCI8 для выполнения анонимных блоков кода Oracle PL/SQL. Есть ли способ связать переменную и получить ее вывод после завершения блока, как я могу, когда аналогичным образом вызываю хранимые процедуры?

$SQL = "declare
something varchar2 := 'I want this returned';
begin
  --How can I return the value of 'something' into a bound PHP variable?
end;";

person aw crud    schedule 01.06.2010    source источник


Ответы (2)


Вы определяете параметр out, используя ключевое слово OUT между именем и объявлением типа данных. IE:

CREATE OR REPLACE PROCEDURE blah (OUT_PARAM_EXAMPLE OUT VARCHAR2) IS ...

Если не указано, по умолчанию используется IN. Если вы хотите использовать параметр как для входа, так и для выхода, используйте:

CREATE OR REPLACE PROCEDURE blah (INOUT_PARAM_EXAMPLE IN OUT VARCHAR2) IS ...

В следующем примере создается процедура с параметрами IN и OUT. Затем процедура выполняется, и результаты распечатываются.

<?php
   // Connect to database...
   $c = oci_connect("hr", "hr_password", "localhost/XE");
   if (!$c) {
      echo "Unable to connect: " . var_dump( oci_error() );
      die();
   }

   // Create database procedure...
   $s = oci_parse($c, "create procedure proc1(p1 IN number, p2 OUT number) as " .
                     "begin" .
                     "  p2 := p1 + 10;" .
                     "end;");
   oci_execute($s, OCI_DEFAULT);

   // Call database procedure...
   $in_var = 10;
   $s = oci_parse($c, "begin proc1(:bind1, :bind2); end;");
   oci_bind_by_name($s, ":bind1", $in_var);
   oci_bind_by_name($s, ":bind2", $out_var, 32); // 32 is the return length
   oci_execute($s, OCI_DEFAULT);
   echo "Procedure returned value: " . $out_var;

   // Logoff from Oracle...
   oci_free_statement($s);
   oci_close($c);
 ?>

Ссылка:

person OMG Ponies    schedule 02.06.2010
comment
Хранимая процедура создается постоянно или только для этого сеанса? Я использовал этот метод для получения параметров OUT из процедур, которые я скомпилировал в базе данных (похоже, что это делает то же самое, за исключением того, что сначала создается процедура). Я хотел бы избежать создания каких-либо постоянных объектов в базе данных при выполнении этого кода. - person aw crud; 02.06.2010
comment
@Renderlin: Да, пример с orafaq.com создаст хранимую процедуру. Вам нужно поместить хранимую процедуру в блок DECLARE анонимного блока PLSQL и сделать proc1(... в части BEGIN/END, чтобы не создавать процедуру. - person OMG Ponies; 02.06.2010
comment
@OMGPonies Есть ли параметры, поддерживаемые в 10g, я читал, что они убрали поддержку! - person Arian Faurtosh; 20.06.2013

Вот мое решение:

function execute_procedure($procedure_name, array $params = array(), &$return_value = ''){
    $sql = "
    DECLARE
        ERROR_CODE      VARCHAR2(2000);
        ERROR_MSG       VARCHAR2(2000);
        RETURN_VALUE    VARCHAR2(2000);
    BEGIN ";

    $c = $this->get_connection();

    $prms = array();
    foreach($params AS $key => $value) $prms[] = ":$key";
    $prms = implode(", ", $prms);

    $sql .= ":RETURN_VALUE := ".$procedure_name."($prms);";
    $sql .= " END;";


    $s = oci_parse($c, $sql);

    foreach($params AS $key => $value)
    {
        $type = SQLT_CHR;
        if(is_array($value))
        {
            if(!isset($value['value'])) continue;
            if(!empty($value['type'])) $type = $value['type'];
            $value = $value['value'];
        }
        oci_bind_by_name($s, ":$key", $value, -1, $type);
    }

    oci_bind_by_name($s, ":RETURN_VALUE", $return_value, 2000);

    try{
        oci_execute($s);
        if(!empty($ERROR_MSG))
        {
            $data['success'] = FALSE;
            $this->errors = "Ошибка: $ERROR_CODE $ERROR_MSG";
        }
        return TRUE;
    }
    catch(ErrorException $e)
    {
        $this->errors = $e->getMessage();
        return FALSE;
    }
}

пример:

    execute_procedure('My_procedure', array('code' => 5454215), $return_value);
    echo $return_value;
person Darkhan ZD    schedule 17.01.2016
comment
Сброс блока кода я вряд ли считаю ценным ответом. Пожалуйста, объясните, почему ваш код решает проблему ОП. - person hotzst; 17.01.2016