mirror of
https://gitee.com/hyperf/hyperf.git
synced 2024-12-05 05:07:58 +08:00
302 lines
9.5 KiB
Markdown
302 lines
9.5 KiB
Markdown
# 請求物件
|
||
|
||
`請求物件(Request)` 是完全基於 [PSR-7](https://www.php-fig.org/psr/psr-7/) 標準實現的,由 [hyperf/http-message](https://github.com/hyperf/http-message) 元件提供實現支援。
|
||
|
||
> 注意 [PSR-7](https://www.php-fig.org/psr/psr-7/) 標準為 `請求(Request)` 進行了 `immutable 機制` 的設計,所有以 `with` 開頭的方法的返回值都是一個新物件,不會修改原物件的值
|
||
|
||
## 安裝
|
||
|
||
該元件完全獨立,適用於任何一個框架專案。
|
||
|
||
```bash
|
||
composer require hyperf/http-message
|
||
```
|
||
|
||
> 如用於其它框架專案則僅支援 PSR-7 提供的 API,具體可直接查閱 PSR-7 的相關規範,該文件所描述的使用方式僅限於使用 Hyperf 時的用法。
|
||
|
||
## 獲得請求物件
|
||
|
||
可以通過容器注入 `Hyperf\HttpServer\Contract\RequestInterface` 獲得 對應的 `Hyperf\HttpServer\Request`,實際注入的物件為一個代理物件,代理的物件為每個請求的 `PSR-7 請求物件(Request)`,也就意味著僅可在 `onRequest` 生命週期內可獲得此物件,下面是一個獲取示例:
|
||
|
||
```php
|
||
declare(strict_types=1);
|
||
|
||
namespace App\Controller;
|
||
|
||
use Hyperf\HttpServer\Contract\RequestInterface;
|
||
use Hyperf\HttpServer\Annotation\AutoController;
|
||
|
||
/**
|
||
* @AutoController()
|
||
*/
|
||
class IndexController
|
||
{
|
||
public function info(RequestInterface $request)
|
||
{
|
||
// ...
|
||
}
|
||
}
|
||
```
|
||
|
||
### 依賴注入與引數
|
||
|
||
如果希望通過控制器方法引數獲取路由引數,可以在依賴項之後列出對應的引數,框架會自動將對應的引數注入到方法引數內,比如您的路由是這樣定義的:
|
||
|
||
```php
|
||
// 註解方式
|
||
/**
|
||
* @GetMapping(path="/user/{id:\d+}")
|
||
*/
|
||
|
||
// 配置方式
|
||
use Hyperf\HttpServer\Router\Router;
|
||
|
||
Router::addRoute(['GET', 'HEAD'], '/user/{id:\d+}', [\App\Controller\IndexController::class, 'user']);
|
||
```
|
||
|
||
則可以通過在方法引數上宣告 `$id` 引數獲得 `Query` 引數 `id`,如下所示:
|
||
|
||
```php
|
||
declare(strict_types=1);
|
||
|
||
namespace App\Controller;
|
||
|
||
use Hyperf\HttpServer\Contract\RequestInterface;
|
||
use Hyperf\HttpServer\Annotation\AutoController;
|
||
|
||
/**
|
||
* @AutoController()
|
||
*/
|
||
class IndexController
|
||
{
|
||
public function info(RequestInterface $request, int $id)
|
||
{
|
||
// ...
|
||
}
|
||
}
|
||
```
|
||
|
||
除了可以通過依賴注入獲取路由引數,還可以通過 `route` 方法獲取,如下所示:
|
||
|
||
```php
|
||
declare(strict_types=1);
|
||
|
||
namespace App\Controller;
|
||
|
||
use Hyperf\HttpServer\Contract\RequestInterface;
|
||
use Hyperf\HttpServer\Annotation\AutoController;
|
||
|
||
/**
|
||
* @AutoController()
|
||
*/
|
||
class IndexController
|
||
{
|
||
public function info(RequestInterface $request)
|
||
{
|
||
// 存在則返回,不存在則返回預設值 null
|
||
$id = $request->route('id');
|
||
// 存在則返回,不存在則返回預設值 0
|
||
$id = $request->route('id', 0);
|
||
// ...
|
||
}
|
||
}
|
||
```
|
||
|
||
### 請求路徑 & 方法
|
||
|
||
`Hyperf\HttpServer\Contract\RequestInterface` 除了使用 [PSR-7](https://www.php-fig.org/psr/psr-7/) 標準定義的 `APIs` 之外,還提供了多種方法來檢查請求,下面我們提供一些方法的示例:
|
||
|
||
#### 獲取請求路徑
|
||
|
||
`path()` 方法返回請求的路徑資訊。也就是說,如果傳入的請求的目標地址是 `http://domain.com/foo/bar?baz=1`,那麼 `path()` 將會返回 `foo/bar`:
|
||
|
||
```php
|
||
$uri = $request->path();
|
||
```
|
||
|
||
`is(...$patterns)` 方法可以驗證傳入的請求路徑和指定規則是否匹配。使用這個方法的時,你也可以傳遞一個 `*` 字元作為萬用字元:
|
||
|
||
```php
|
||
if ($request->is('user/*')) {
|
||
// ...
|
||
}
|
||
```
|
||
|
||
#### 獲取請求的 URL
|
||
|
||
你可以使用 `url()` 或 `fullUrl()` 方法去獲取傳入請求的完整 `URL`。`url()` 方法返回不帶有 `Query 引數` 的 `URL`,而 `fullUrl()` 方法的返回值包含 `Query 引數` :
|
||
|
||
```php
|
||
// 沒有查詢引數
|
||
$url = $request->url();
|
||
|
||
// 帶上查詢引數
|
||
$url = $request->fullUrl();
|
||
```
|
||
|
||
#### 獲取請求方法
|
||
|
||
`getMethod()` 方法將返回 `HTTP` 的請求方法。你也可以使用 `isMethod(string $method)` 方法去驗證 `HTTP` 的請求方法與指定規則是否匹配:
|
||
|
||
```php
|
||
$method = $request->getMethod();
|
||
|
||
if ($request->isMethod('post')) {
|
||
// ...
|
||
}
|
||
```
|
||
|
||
### PSR-7 請求及方法
|
||
|
||
[hyperf/http-message](https://github.com/hyperf/http-message) 元件本身是一個實現了 [PSR-7](https://www.php-fig.org/psr/psr-7/) 標準的元件,相關方法都可以通過注入的 `請求物件(Request)` 來呼叫。
|
||
如果注入時宣告為 [PSR-7](https://www.php-fig.org/psr/psr-7/) 標準的 `Psr\Http\Message\ServerRequestInterface` 介面,則框架會自動轉換為等同於 `Hyperf\HttpServer\Contract\RequestInterface` 的 `Hyperf\HttpServer\Request` 物件。
|
||
|
||
> 建議使用 `Hyperf\HttpServer\Contract\RequestInterface` 來注入,這樣可獲得 IDE 對專屬方法的自動完成提醒支援。
|
||
|
||
## 輸入預處理 & 規範化
|
||
|
||
## 獲取輸入
|
||
|
||
### 獲取所有輸入
|
||
|
||
您可以使用 `all()` 方法以 `陣列` 形式獲取到所有輸入資料:
|
||
|
||
```php
|
||
$all = $request->all();
|
||
```
|
||
|
||
### 獲取指定輸入值
|
||
|
||
通過 `input(string $key, $default = null)` 和 `inputs(array $keys, $default = null): array` 獲取 `一個` 或 `多個` 任意形式的輸入值:
|
||
|
||
```php
|
||
// 存在則返回,不存在則返回 null
|
||
$name = $request->input('name');
|
||
// 存在則返回,不存在則返回預設值 Hyperf
|
||
$name = $request->input('name', 'Hyperf');
|
||
```
|
||
|
||
如果傳輸表單資料中包含「陣列」形式的資料,那麼可以使用「點」語法來獲取陣列:
|
||
|
||
```php
|
||
$name = $request->input('products.0.name');
|
||
|
||
$names = $request->input('products.*.name');
|
||
```
|
||
### 從查詢字串獲取輸入
|
||
|
||
使用 `input`, `inputs` 方法可以從整個請求中獲取輸入資料(包括 `Query 引數`),而 `query(?string $key = null, $default = null)` 方法可以只從查詢字串中獲取輸入資料:
|
||
|
||
```php
|
||
// 存在則返回,不存在則返回 null
|
||
$name = $request->query('name');
|
||
// 存在則返回,不存在則返回預設值 Hyperf
|
||
$name = $request->query('name', 'Hyperf');
|
||
// 不傳遞引數則以關聯陣列的形式返回所有 Query 引數
|
||
$name = $request->query();
|
||
```
|
||
|
||
### 獲取 `JSON` 輸入資訊
|
||
|
||
如果請求的 `Body` 資料格式是 `JSON`,則只要 `請求物件(Request)` 的 `Content-Type` `Header 值` 正確設定為 `application/json`,就可以通過 `input(string $key, $default = null)` 方法訪問 `JSON` 資料,你甚至可以使用 「點」語法來讀取 `JSON` 陣列:
|
||
|
||
```php
|
||
// 存在則返回,不存在則返回 null
|
||
$name = $request->input('user.name');
|
||
// 存在則返回,不存在則返回預設值 Hyperf
|
||
$name = $request->input('user.name', 'Hyperf');
|
||
// 以陣列形式返回所有 Json 資料
|
||
$name = $request->all();
|
||
```
|
||
|
||
### 確定是否存在輸入值
|
||
|
||
要判斷請求是否存在某個值,可以使用 `has($keys)` 方法。如果請求中存在該值則返回 `true`,不存在則返回 `false`,`$keys` 可以傳遞一個字串,或傳遞一個數組包含多個字串,只有全部存在才會返回 `true`:
|
||
|
||
```php
|
||
// 僅判斷單個值
|
||
if ($request->has('name')) {
|
||
// ...
|
||
}
|
||
// 同時判斷多個值
|
||
if ($request->has(['name', 'email'])) {
|
||
// ...
|
||
}
|
||
```
|
||
|
||
## Cookies
|
||
|
||
### 從請求中獲取 Cookies
|
||
|
||
使用 `getCookieParams()` 方法從請求中獲取所有的 `Cookies`,結果會返回一個關聯陣列。
|
||
|
||
```php
|
||
$cookies = $request->getCookieParams();
|
||
```
|
||
|
||
如果希望獲取某一個 `Cookie` 值,可通過 `cookie(string $key, $default = null)` 方法來獲取對應的值:
|
||
|
||
```php
|
||
// 存在則返回,不存在則返回 null
|
||
$name = $request->cookie('name');
|
||
// 存在則返回,不存在則返回預設值 Hyperf
|
||
$name = $request->cookie('name', 'Hyperf');
|
||
```
|
||
|
||
## 檔案
|
||
|
||
### 獲取上傳檔案
|
||
|
||
你可以使用 `file(string $key, $default): ?Hyperf\HttpMessage\Upload\UploadedFile` 方法從請求中獲取上傳的檔案物件。如果上傳的檔案存在則該方法返回一個 `Hyperf\HttpMessage\Upload\UploadedFile` 類的例項,該類繼承了 `PHP` 的 `SplFileInfo` 類的同時也提供了各種與檔案互動的方法:
|
||
|
||
```php
|
||
// 存在則返回一個 Hyperf\HttpMessage\Upload\UploadedFile 物件,不存在則返回 null
|
||
$file = $request->file('photo');
|
||
```
|
||
|
||
### 檢查檔案是否存在
|
||
|
||
您可以使用 `hasFile(string $key): bool` 方法確認請求中是否存在檔案:
|
||
|
||
```php
|
||
if ($request->hasFile('photo')) {
|
||
// ...
|
||
}
|
||
```
|
||
|
||
### 驗證成功上傳
|
||
|
||
除了檢查上傳的檔案是否存在外,您也可以通過 `isValid(): bool` 方法驗證上傳的檔案是否有效:
|
||
|
||
```php
|
||
if ($request->file('photo')->isValid()) {
|
||
// ...
|
||
}
|
||
```
|
||
|
||
### 檔案路徑 & 副檔名
|
||
|
||
`UploadedFile` 類還包含訪問檔案的完整路徑及其副檔名方法。`getExtension()` 方法會根據檔案內容判斷檔案的副檔名。該副檔名可能會和客戶端提供的副檔名不同:
|
||
|
||
```php
|
||
// 該路徑為上傳檔案的臨時路徑
|
||
$path = $request->file('photo')->getPath();
|
||
|
||
// 由於 Swoole 上傳檔案的 tmp_name 並沒有保持檔案原名,所以這個方法已重寫為獲取原檔名的字尾名
|
||
$extension = $request->file('photo')->getExtension();
|
||
```
|
||
|
||
### 儲存上傳檔案
|
||
|
||
上傳的檔案在未手動儲存之前,都是存在一個臨時位置上的,如果您沒有對該檔案進行儲存處理,則在請求結束後會從臨時位置上移除,所以我們可能需要對檔案進行持久化儲存處理,通過 `moveTo(string $targetPath): void` 將臨時檔案移動到 `$targetPath` 位置持久化儲存,程式碼示例如下:
|
||
|
||
```php
|
||
$file = $request->file('photo');
|
||
$file->moveTo('/foo/bar.jpg');
|
||
|
||
// 通過 isMoved(): bool 方法判斷方法是否已移動
|
||
if ($file->isMoved()) {
|
||
// ...
|
||
}
|
||
```
|