HTTP响应实例教程 —— 基本使用、生成Cookie、返回视图、JSON/JSONP、文件下载及重定向


1、Response篇

1.1 基本响应

最基本的HTTP响应只需在路由闭包或控制器动作中返回一个简单字符串即可,但是具体业务逻辑中大部分响应都是在控制器动作中返回Response实例或者视图。Response是继承自 Symfony\Component\HttpFoundation\ResponseIlluminate\Http\Response类的一个实例,我们可以使用该实例上的一系列方法来创建HTTP响应:

  1. use Illuminate\Http\Response;
  2.  
  3. Route::get('testResponse',function(){
  4. $content = 'Hello LaravelAcademy!';
  5. $status = 200;
  6. $value = 'text/html;charset=utf-8';
  7. return (new Response($content,$status))->header('Content-Type',$value);
  8. });

在浏览器中访问,F12查看响应头信息(Chrome浏览器):

基本响应

如果我们尝试修改$status=500,则头信息如下:

基本响应2

为方便使用,我们还可以使用全局帮助函数response来替代生成Response对象实例:

  1. Route::get('testResponse',function(){
  2. $content = 'Hello LaravelAcademy!';
  3. $status = 500;
  4. $value = 'text/html;charset=utf-8';
  5. return response($content,$status)->header('Content-Type',$value);
  6. });

效果一样。以后我们将默认使用这种方式,不再生成Response对象实例。

此外,需要注意的是,Illuminate\Http\Response 类中还使用了ResponseTraitheader方法正是该trait提供的,除了header之外,该trait还提供了withCookiecontentstatus方法。header方法用于设置响应头信息,withCookie方法用于添加cookie,这两个方法都会返回调用它的Response自身对象,所以这两个方法都支持方法链(即多次调用headerwithCookie方法);而contentstatus方法则用于返回当前响应的响应实体内容和响应状态码。

1.2 添加Cookie

正如上面提到的,我们使用withCookie方法为响应添加cookie,由于headerwithCookie支持方法链,所以我们可以这样使用:

  1. Route::get('testResponseCookie',function(){
  2. $content = 'Hello LaravelAcademy!';
  3. $status = 200;
  4. $value = 'text/html;charset=utf-8';
  5. return response($content,$status)->header('Content-Type',$value)
  6. ->withCookie('site','LaravelAcademy.org');
  7. });

在浏览器中访问,F12查看Cookie信息:

生成Cookie

我们还可以使用该cookie的有效期、作用域等信息:

  1. Route::get('testResponseCookie',function(){
  2. $content = 'Hello LaravelAcademy!';
  3. $status = 200;
  4. $value = 'text/html;charset=utf-8';
  5. //设置cookie有效期为30分钟,作用路径为应用根目录,作用域名为laravel.app
  6. return response($content,$status)->header('Content-Type',$value)
  7. ->withCookie('site','LaravelAcademy.org',30,'/','laravel.app');
  8. });

注:withCookie方法实际上是调用了全局帮助函数cookie生成cookie,然后将cookie放到响应头中。

再次在浏览器中访问,F12查看cookie信息如下:

生成cookie

里面多出了有效期及作用域名等相关信息。

此外,我们还关注到该cookie是经过加密的,这一点我们在前面已经提到过,这是为了安全性考虑,如果要取消加密,在app/Http/Middleware/EncryptCookies.php文件中将对应的cookie名添加到EncryptCookies类属性$except中即可:

  1. <?php
  2.  
  3. namespace App\Http\Middleware;
  4.  
  5. use Illuminate\Cookie\Middleware\EncryptCookies as BaseEncrypter;
  6.  
  7. class EncryptCookies extends BaseEncrypter
  8. {
  9. /**
  10. * 不被加密的cookie名
  11. *
  12. * @var array
  13. */
  14. protected $except = [
  15. 'site'
  16. ];
  17. }

再次在浏览器中访问http://laravel.app:8000/testResponseCookie,F12查看Cookie信息如下:

未加密cookie

当然,为了安全起见,我们不建议这么做。

2、ResponseFactory篇

response函数如果不传入参数会返回 Illuminate\Contracts\Routing\ResponseFactory契约的实现——Illuminate\Routing\ResponseFactory,该类中提供了多个方法用来生成更加丰富的响应类型,比如视图响应、JSON响应、文件下载等等。

2.1 视图响应

使用view方法即可返回一个视图作为响应内容:

  1. Route::get('testResponseView',function(){
  2. $value = 'text/html;charset=utf-8';
  3. return response()->view('hello',['message'=>'Hello LaravelAcademy'])
  4. ->header('Content-Type',$value);
  5. });

与之对应的,我们需要在resources/views下新建一个视图文件hello.blade.php,其内容如下:

  1. {{$message}}

接下来我们在浏览器中访问http://laravel.app:8000/testResponseView,页面输出:

  1. Hello LaravelAcademy

还可以再简化,如果我们不需要自定义HTTP响应头,还可以直接使用全局帮助函数view

  1. Route::get('testResponseView',function(){
  2. $value = 'text/html;charset=utf-8';
  3. return view('hello',['message'=>'Hello LaravelAcademy']));
  4. });

效果和上面一样。

2.2 返回JSON/JSONP

还可以使用json方法返回json格式数据:

  1. Route::get('testResponseJson',function(){
  2. return response()->json(['name'=>'LaravelAcademy','passwd'=>'LaravelAcademy.org']);
  3. });

在浏览器中访问,F12查看响应头信息:

生成json响应

根据输出信息可见,json方法会自动设置Content-Typeapplication/json,并调用PHP内置函数json_encode讲数组转化为json格式字符串。

如果返回的是JSONP响应,也很简单,只需要在json后面再调用setCallback即可:

  1. Route::get('testResponseJson',function(){
  2. return response()->json(['name'=>'LaravelAcademy','passwd'=>'LaravelAcademy.org'])
  3. ->setCallback(request()->input('callback'));
  4. });

2.3 文件下载

使用download方法可生成用于下载给定路径文件的响应,这里我们下载上一节上传的文件:

  1. Route::get('testResponseDownload',function(){
  2. return response()->download(
  3. realpath(base_path('public/images')).'/laravel-5-1.jpg',
  4. 'Laravel学院.jpg'
  5. );
  6. });

在浏览器中访问http://laravel.app:8000/testResponseDownload,页面将会下载laravel-5-1.jpg文件并保存为Laravel学院.jpg

3、 RedirectResponse篇

重定向响应是 Illuminate\Http\RedirectResponse类的实例,我们通常使用全局帮助函数redirect来生成 RedirectResponse实例。和response类似,redirect函数如果接收参数则调用的是Illuminate\Routing\Redirector类的to方法,如果无参调用则返回的是Redirector对象实例。

3.1 基本重定向

  1. Route::get('dashboard', function () {
  2. return redirect('home/dashboard');
  3. });

如果要重定向到上一个位置,则使用back方法:

  1. Route::post('user/profile', function () {
  2. // 验证请求...
  3. return back()->withInput();
  4. });

3.2 重定向到命名路由

使用route方法重定向到命名路由:

  1. Route::get('/hello/laravelacademy',['as'=>'academy',function(){
  2. return 'Hello LaravelAcademy';
  3. }]);
  4.  
  5. Route::get('testResponseRedirect',function(){
  6. return redirect()->route('academy');
  7. });

在浏览器中访问http://laravel.app:8000/testResponseRedirect,页面会跳转到http://laravel.app:8000/hello/laravelacademy并输出:

  1. Hello LaravelAcademy

如果命名路由中有参数,那么我们可以在route中传入参数:

  1. Route::get('/hello/laravelacademy/{id}',['as'=>'academy',function($id){
  2. return 'Hello LaravelAcademy '.$id;
  3. }]);
  4.  
  5. Route::get('testResponseRedirect',function(){
  6. return redirect()->route('academy',100);
  7. });

在浏览器中访问http://laravel.app:8000/testResponseRedirect,页面会跳转到http://laravel.app:8000/hello/laravelacademy/100并输出:

  1. Hello LaravelAcademy 100

3.3 重定向到控制器动作

使用action方法重定向到控制器动作:

  1. Route::resource('post','PostController');
  2.  
  3. Route::get('testResponseRedirect',function(){
  4. return redirect()->action('PostController@index');
  5. });

在浏览器中访问http://laravel.app:8000/testResponseRedirect,则页面会跳转到http://laravel.app:8000/post并输出对应内容。

当然也可以传递参数到action方法:

  1. Route::get('testResponseRedirect',function(){
  2. return redirect()->action('PostController@show',[1]);
  3. });

3.4 带一次性Session数据的重定向

使用with方法可以携带一次性session数据到重定向请求页面(一次性session数据即使用后立即销毁的session数据项):

  1. Route::post('user/profile', function () {
  2. // 更新用户属性...
  3. return redirect('dashboard')->with('status', 'Profile updated!');
  4. });

这种特性通常在提交表单验证失败返回错误信息时很有用。