Laravel Http 层 请求

访问请求实例

要通过依赖注入获取当前 HTTP 请求实例,需要在控制器的构造函数或方法中对 Illuminate\Http\Request 类进行类型提示,这样当前请求实例会被服务容器自动注入:

<?php

namespace App\Http\Controllers;

use Illuminate\Http\Request;

class UserController extends Controller
{
    /**
     * 存储新用户
     *
     * @param Request $request
     * @return Response
     */
    public function store(Request $request)
    {
        $name=$request->input('name');

        //
    }
}

依赖注入 & 路由参数

如果你的控制器方法还期望获取路由参数,只需要将路由参数置于其它依赖之后即可,例如,如果你的路由定义如下:

Route::put('user/{id}','UserController@update');

你仍然可以对 Illuminate\Http\Request
进行类型提示并通过如下方式定义控制器方法来访问路由参数 id

<?php
  namespace App\Http\Controllers;
  use Illuminate\Http\Request;
class UserController extends Controller{ 
/**
 * 更新指定用户
 *
 * @param Request $request
 * @param int $id
 * @return Response
 */ 
public function update(Request $request,$id) {
 //
 }
}

通过路由闭包访问请求
还可以在路由闭包上类型提示 Illuminate\Http\Request
类,在执行的时候服务容器会自动注入输入的请求到闭包:

use Illuminate\Http\Request;
Route::get('/', function (Request $request) {
 //
});

请求路径 & 方法

Illuminate\Http\Request实例提供了多个方法来检测应用的 HTTP 请求,Laravel 的 Illuminate\Http\Request继承自 Symfony\Component\HttpFoundation\Request类,下面演示了该类提供的一些有用方法:

获取请求路径
path 方法将会返回请求的路径信息,因此,如果进入的请求路径是 http://domain.com/foo/bar,则 path 方法将会返回 foo/bar

$uri=$request->path();

is 方法允许你验证进入的请求是否与给定模式匹配。使用该方法时可以使用 * 通配符:

if($request->is('admin/*')){ 
  //
}

获取请求URL

想要获取完整的 URL,而不仅仅是路径信息,可以使用请求实例提供的 urlfullUrl 方法,url 方法将会返回不带查询字符串的 URL,而 fullUrl 方法返回结果则包含查询字符串:

// 不包含查询字符串
$url = $request->url();
// 包含查询字符串
$url = $request->fullUrl();

获取请求方法
method
方法将会返回 HTTP 请求方式。你还可以使用 isMethod
方法来验证 HTTP 请求方式是否匹配给定字符串:
$method=$request->method();if($request->isMethod('post')){ //}

PSR-7 请求

PSR-7 标准指定了 HTTP 消息接口,包括请求和响应。如果你想要获取 PSR-7 请求实例,首先需要安装一些库,Laravel 使用 Symfony HTTP Message Bridge 组件将典型的 Laravel 请求和响应转化为 兼容PSR-7的实现:

composer require symfony/psr-http-message-bridgecomposer 
require zendframework/zend-diactoros

安装完这些库之后,你只需要在路由或控制器中通过对请求类型进行类型提示就可以获取PSR-7 请求:

use Psr\Http\Message\ServerRequestInterface;

Route::get('/', function (ServerRequestInterface $request) {
 //
});

注:如果从路由或控制器返回的是 PSR-7 响应实例,则其将会自动转化为 Laravel 响应实例并显示出来。

输入修整 & 正常化

默认情况下,Laravel 在 App\Http\Kernel 的全局中间件堆栈中引入了 TrimStringsConvertEmptyStringsToNull 中间件。这些中间件会自动对请求中的字符串字段进行处理,前者将字符串两端的空格清除,后者将空字符串转化为 null。这样,在路由和控制器中我们就不必对字符串字段做额外的处理。

如果你想要禁止该行为,可以从App\Http\Kernel 的中间件堆栈属性 $middleware 中移除这两个中间件。

获取请求输入

获取所有输入值
你可以使用 all 方法以数组格式获取所有输入值:

$input = $request->all();

获取单个输入值
使用一些简单的方法,就可以从 Illuminate\Http\Request 实例中访问用户输入。你不需要关心请求所使用的 HTTP 请求方法,因为对所有请求方式的输入都是通过input 方法获取用户输入:

$name = $request->input('name');

你还可以传递一个默认值作为第二个参数给 input 方法,如果请求输入值在当前请求未出现时该值将会被返回:

$name = $request->input('name', 'Sally');

处理表单数组输入时,可以使用”.”来访问数组输入:

$input = $request->input('products.0.name');
$names = $request->input('products.*.name');

通过动态属性获取输入

此外,你还可以通过使用 Illuminate\Http\Request 实例上的动态属性来访问用户输入。例如,如果你的应用表单包含 name字段,那么可以像这样访问提交的值:

$name = $request->name;

使用动态属性的时候,Laravel 首先会在请求中查找参数的值,如果不存在,还会到路由参数中查找。

获取JSON输入值
发送JSON请求到应用的时候,只要 Content-Type 请求头被设置为 application/json,都可以通过input 方法获取 JSON 数据,还可以通过“.”号解析数组:

$name = $request->input('user.name');

获取输入的部分数据

如果你需要取出输入数据的子集,可以使用 onlyexcept 方法,这两个方法都接收一个数组或动态列表作为唯一参数:

$input = $request->only(['username', 'password']);
$input = $request->only('username', 'password');
$input = $request->except(['credit_card']);
$input = $request->except('credit_card');

only 方法返回你所请求的所有键值对,即使输入请求中不包含你所请求的键,当对应键不存在时,对应返回值为 null ,如果你想要获取输入请求中确实存在的部分数据,可以使用 intersect 方法:

$input = $request->intersect(['username', 'password']);

判断输入值是否存在

判断值是否在请求中存在,可以使用 has 方法,如果值出现过了且不为空,has 方法返回 true

if ($request->has('name')) {
 //
}

上一次请求输入

Laravel 允许你在两次请求之间保存输入数据,这个特性在检测校验数据失败后需要重新填充表单数据时很有用,但如果你使用的是 Laravel 自带的验证服务,则不需要手动使用这些方法,因为一些 Laravel 自带的校验设置会自动调用它们。

将输入存储到一次性 Session
Illuminate\Http\Request 实例的 flash 方法会将当前输入存放到一次性 Session(所谓的一次性指的是从 Session 中取出数据后,对应数据会从 Session 中销毁)中,这样在下一次请求时数据依然有效:

$request->flash();

你还可以使用 flashOnlyflashExcept 方法将输入数据子集存放到 Session 中,这些方法在 Session 之外保存敏感信息时很有用:

$request->flashOnly('username', 'email');
$request->flashExcept('password');

将输入存储到一次性 Session 然后重定向

如果你经常需要一次性存储输入并重定向到前一页,可以使用 withInput 方法来将输入数据添加到 redirect后面:

return redirect('form')->withInput();
return redirect('form')->withInput($request->except('password'));

取出上次请求数据
要从 Session 中取出上次请求的输入数据,可以使用 Request 实例的 old方法。old 方法可以很方便地从Session 中取出一次性数据:

$username = $request->old('username');

Laravel 还提供了一个全局的辅助函数 old ,如果你是在 Blade 模板中显示上次输入数据,使用辅助函数 old 更方便,如果给定参数没有对应输入,返回null

<input type="text" name="username" value="{{ old('username') }}">

文件上传

获取上传的文件

可以使用 Illuminate\Http\Request 实例提供的 file 方法或者动态属性来访问上传文件, file 方法返回Illuminate\Http\UploadedFile 类的一个实例,该类继承自 PHP 标准库中提供与文件交互方法的 SplFileInfo类:

$file = $request->file('photo');$file = $request->photo;

你可以使用 hasFile 方法判断文件在请求中是否存在:

if ($request->hasFile('photo')) {
 //
}

验证文件是否上传成功
使用 isValid 方法判断文件在上传过程中是否出错:

if ($request->file('photo')->isValid()){
 //
}

文件路径 & 扩展名
UploadedFile 类还提供了访问上传文件绝对路径和扩展名的方法。 extension 方法可以基于文件内容判断文件扩展名,该扩展名可能会和客户端提供的扩展名不一致:

$path = $request->photo->path();
$extension = $request->photo->extension();

其他文件方法
UploadedFile 实例上还有很多其他可用方法,查看该类的API文档了解更多信息。

保存上传的文件

要保存上传的文件,通常需要使用你所配置的其中一个文件系统UploadedFile 类有一个 store 方法,该方法会将上传文件移动到相应的磁盘路径上,该路径可以是本地文件系统的某个位置,也可以是云存储(如Amazon S3)上的路径。

store 方法接收一个文件保存的相对路径(相对于文件系统配置的根目录 ),该路径不应该包含文件名,因为系统会自动生成一个唯一ID作为文件名。

store 方法还接收一个可选的参数——用于存储文件的磁盘名称作为第二个参数,该方法会方返回相对于根目录的文件路径:

$path = $request->photo->store('images');
$path = $request->photo->store('images', 's3');

如果你不想自动生成文件名,可以使用 storeAs方法,该方法接收保存路径、文件名和磁盘名作为参数:

$path = $request->photo->storeAs('images', 'filename.jpg');
$path = $request->photo->storeAs('images', 'filename.jpg', 's3');
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,064评论 5 466
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,606评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,011评论 0 328
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,550评论 1 269
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,465评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 47,919评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,428评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,075评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,208评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,185评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,191评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,914评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,482评论 3 302
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,585评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,825评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,194评论 2 344
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,703评论 2 339

推荐阅读更多精彩内容

  • 先说几句废话,调和气氛。事情的起由来自客户需求频繁变更,伟大的师傅决定横刀立马的改革使用新的框架(created ...
    wsdadan阅读 3,029评论 0 12
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,551评论 18 139
  • 1、简介 Laravel 提供了多种方法来验证应用输入数据。默认情况下,Laravel 的控制器基类使用Valid...
    伊Summer阅读 1,517评论 0 4
  • HTTP请求 访问请求 为了通过依赖注入能够方便的获取http请求实例,你应该在控制器的构造函数或者控制函数中写入...
    Dearmadman阅读 2,412评论 0 5
  • 今天我来分享 Laravel 中 HTTP 层关于请求、响应与表单验证的知识。 0x00 HTTP 请求 获取请求...
    胖福哥阅读 1,352评论 1 20