Есть ли у PHP эквивалент Java-запроса RequestDispatcher.forward?

На Java я могу написать действительно базовую JSP index.jsp вот так:

<% request.getRequestDispatcher("/home.action").forward(request, response); %>

В результате пользователь запрашивает index.jsp (или просто содержащий каталог, предполагая, index.jsp это документ по умолчанию для каталога) увидим home.action без перенаправления из браузера, т.е. [forward] ( http://java.sun.com/javaee/5/docs/api/javax/servlet/RequestDispatcher.html происходит на стороне сервера.

Могу ли я сделать что-то подобное с PHP? Я подозреваю, что возможно настроить Apache для обработки этого случая, но, поскольку у меня может не быть доступа к соответствующей конфигурации Apache, меня заинтересует решение, основанное только на PHP.

7 ответов

Решение

Если вы беспокоитесь о доступности CURL, вы можете использовать file_get_contents() и потоки. Настройка функции, например:

function forward($location, $vars = array()) 
{
    $file ='http://'.$_SERVER['HTTP_HOST']
    .substr($_SERVER['REQUEST_URI'],0,strrpos($_SERVER['REQUEST_URI'], '/')+1)
    .$location;

    if(!empty($vars))
    {
         $file .="?".http_build_query($vars);
    }

    $response = file_get_contents($file);

    echo $response;
}

Это просто устанавливает GET, но вы можете сделать пост с file_get_contents() также.

Попробуй это.

function forward($page, $vars = null){
    ob_clean();
    include($page);
    exit;
}

на включенной странице $vars переменная будет работать как атрибуты запроса Java

Я считаю, что одним из самых близких аналогичных методов было бы использование функции virtual() при запуске php в качестве модуля apache.

virtual() является специфичной для Apache функцией, которая похожа на mod_include. Он выполняет подзапрос Apache.

Если вы используете MVC, например Zend Framework, вы можете изменить действие контроллера или даже переключаться между действиями контроллера. Метод _forward, как описано здесь.

Хитрость в Request.Forward заключается в том, что он дает вам чистый новый запрос к желаемому действию. Следовательно, у вас нет остатков от текущего запроса, и, например, нет проблем со сценариями, которые основаны на java-эквалайзере $_SERVER['REQUEST_URI'].

Вы можете просто зайти в класс CURL и написать простую функцию для этого:

<?php 
/**
 * CURLHandler handles simple HTTP GETs and POSTs via Curl 
 * 
 * @author SchizoDuckie
 * @version 1.0
 * @access public
 */
class CURLHandler
{

    /**
     * CURLHandler::Get()
     * 
     * Executes a standard GET request via Curl.
     * Static function, so that you can use: CurlHandler::Get('http://www.google.com');
     * 
     * @param string $url url to get
     * @return string HTML output
     */
    public static function Get($url)
    {
       return self::doRequest('GET', $url);
    }

    /**
     * CURLHandler::Post()
     * 
     * Executes a standard POST request via Curl.
     * Static function, so you can use CurlHandler::Post('http://www.google.com', array('q'=>'belfabriek'));
     * If you want to send a File via post (to e.g. PHP's $_FILES), prefix the value of an item with an @ ! 
     * @param string $url url to post data to
     * @param Array $vars Array with key=>value pairs to post.
     * @return string HTML output
     */
    public static function Post($url, $vars, $auth = false) 
    {
       return self::doRequest('POST', $url, $vars, $auth);
    }

    /**
     * CURLHandler::doRequest()
     * This is what actually does the request
     * <pre>
     * - Create Curl handle with curl_init
     * - Set options like CURLOPT_URL, CURLOPT_RETURNTRANSFER and CURLOPT_HEADER
     * - Set eventual optional options (like CURLOPT_POST and CURLOPT_POSTFIELDS)
     * - Call curl_exec on the interface
     * - Close the connection
     * - Return the result or throw an exception.
     * </pre>
     * @param mixed $method Request Method (Get/ Post)
     * @param mixed $url URI to get or post to
     * @param mixed $vars Array of variables (only mandatory in POST requests)
     * @return string HTML output
     */
    public static function doRequest($method, $url, $vars=array(), $auth = false)
    {
        $curlInterface = curl_init();

        curl_setopt_array ($curlInterface, array( 
            CURLOPT_URL => $url,
            CURLOPT_CONNECTTIMEOUT => 2,
            CURLOPT_RETURNTRANSFER => 1,
            CURLOPT_FOLLOWLOCATION =>1,
            CURLOPT_HEADER => 0));

        if (strtoupper($method) == 'POST')
        {
            curl_setopt_array($curlInterface, array(
                CURLOPT_POST => 1,
                CURLOPT_POSTFIELDS => http_build_query($vars))
            );  
        }
        if($auth !== false)
        {
              curl_setopt($curlInterface, CURLOPT_USERPWD, $auth['username'] . ":" . $auth['password']);
        }
        $result = curl_exec ($curlInterface);
        curl_close ($curlInterface);

        if($result === NULL)
        {
            throw new Exception('Curl Request Error: '.curl_errno($curlInterface) . " - " . curl_error($curlInterface));
        }
        else
        {
            return($result);
        }
    }

}

Просто дамп это в class.CURLHandler.php, и вы можете сделать это:

Конечно, использование $_REQUEST не совсем безопасно (вы должны проверить $_SERVER['REQUEST_METHOD']), но вы поймете.

<?php
include('class.CURLHandler.php');
die CURLHandler::doRequest($_SERVER['REQUEST_METHOD'], 'http://server/myaction', $_REQUEST);
?>

Конечно, CURL не везде установлен, но для этого у нас есть встроенные эмуляторы завитков PHP.

Кроме того, это дает вам еще большую гибкость, чем Request.Forward, поскольку вы также можете перехватывать и обрабатывать вывод.

Концепции Redirect и Forward, как в Java, могут быть достижимы и в PHP.

Redirect:: header("Location: redirect.php"); - (URL в адресной строке изменяется)

Вперед:: include forward.php ; - (URL неизменен в адресной строке)

Его можно управлять с помощью этой и других программных логик

Вы можете использовать как:

header ("Location: /path/");
exit;

Выход необходим только в том случае, если какой-либо вывод HTML был отправлен ранее, header() не будет работать, поэтому вы должны отправить новый заголовок перед любым выводом в браузер.

Другие вопросы по тегам