zoukankan      html  css  js  c++  java
  • 五、基础组件 —— HTTP 请求

    接受请求

       要通过依赖注入获取当前 HTTP 请求实例,你应该在控制器上引入 IlluminateHttpRequest 类, 传入的请求实例将会由 服务容器 自动注入:

    <?php
    
    namespace AppHttpControllers;
    
    use IlluminateHttpRequest;
    
    class UserController extends Controller
    {
        /**
         * 存储一个新用户。
         *
         * @param  Request  $request
         * @return Response
         */
        public function store(Request $request)
        {
            $name = $request->input('name');
    
            //
        }
    }

       依赖注入 & 路由参数

         如果你的控制器需要从路由参数中获取数据,你应该在其他依赖项之后列入参数。举个例子,你的路由是这样定义的:

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

         你可以通过下面的方法来定义控制器,使用 IlluminateHttpRequest 类来获取你的路由参数 id

    <?php
    
    namespace AppHttpControllers;
    
    use IlluminateHttpRequest;
    
    class UserController extends Controller
    {
        /**
         * 更新指定用户
         *
         * @param  Request  $request
         * @param  string  $id
         * @return Response
         */
        public function update(Request $request, $id)
        {
            //
        }
    }

       通过闭包路由获取参数

         你也可以在路由闭包中使用 IlluminateHttpRequest 类, 服务容器会自动的将请求参数注入到路由闭包中:

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

       请求路径 & 方法

         IlluminateHttpRequest 实例提供了一系列方法来验证 HTTP 请求参数,并继承了 SymfonyComponentHttpFoundationRequest 类 。

             下面是该类的一些重要方法:

               获取请求路径

           path 方法返回请求的路径信息。所以,如果请求的路径是 http://domain.com/foo/bar, path 方法将会返回 foo/bar:

          $uri = $request->path();

           is 可以验证传入的请求路径是否与给定的模式匹配。 在这个方法中,你也可以使用 * 字符作为通配符:

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

              获取请求 URL

                 可以使用 url 或 fullUrl 方法获取完整的请求 URL。 url 方法返回不含有查询串的 URL, fullUrl 获取包含查询串的 URL

          // 不附带查询串...
          $url = $request->url();
    
         // 附带查询串...
         $url = $request->fullUrl();

              获取请求 Method

            method 方法返回请求的 HTTP 动作。还可以使用 isMethod 方法校验 HTTP 动作是否与给定的客串上匹配:

          $method = $request->method();
    
          if ($request->isMethod('post')) {
             //
           }

       PSR-7 请求 

             PSR-7 standard 定义了 HTTP 消息接口,包括 请求 和 响应。如果你想用 PSR-7 请求代替 Laravel 请求,需要先安装几个库。 Laravel 使用 Symfony HTTP Message Bridge 组件将典型的 Laravel 请求 和 响应 转换为 PSR-7 兼容实现:

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

             一旦安装了这些库,就可以通过在路由闭包或控制器方法中的请求接口类型提示来获取 PSR-7 请求:

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

             注:如果从路由或控制器返回 PSR-7 响应实例,框架会自动将其转换回 Laravel 响应实例并显示。

    输入的裁剪和标准化

       默认情况下,Laravel 在应用的全局中间件堆栈中包含了 TrimStrings 和 ConvertEmptyStringsToNull 中间件。这些中间件被放在 AppHttpKernel 类的堆栈列表中。它们自动裁剪请求中的所有输入字符串域,同时将空字符串域转换为 null。这样一来,你就不必担心路由和控制器中的标准化规约问题。

       如果想要禁用这个行为,只需要通过从 AppHttpKernel 类的 $middleware 属性中移除它(相当于从应用的中间件堆栈中移除)。

    获取输入

       获取所有的输入数据

         可以使用 all 方法获取所有输入数据数组:

     URL /user/store?name=zhangsan&sto=326187
    $input
    = $request->all();
     array:2 [
       "name" => "zhangsan"
       "sto" => "326187"
     ]

       获取单个输入值

         使用一些简便方法,就可以通 实例过 IlluminateHttpRequest 实例获取用户的全部输入,不需要担心用户请求用的是哪种 HTTP 动作。无论哪种 HTTP 动作, 用户的请求都能被 input 方法获取:

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

         可以将默认值作为传递给 input 方法的第二个参数。这个值将在请求没有包含该参数时被返回:

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

         当与包含数组输入的表单协作时,使用 「点」 运算符访问数组元素:

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

         不带参数调用 input 方法,能够获取全部输入值(关联数组形式):

     $input = $request->input();

       从查询串中获取输入

       input 方法从整个请求载体中获取值(包括查询串), query 方法则仅从查询串中获取值:

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

         如果查询串值不存在,query 方法的第二个参数将被作为该参数的默认值被返回:

     $name = $request->query('name', 'Helen');

         不带参数调用 query 方法,能够获取查询串的所有值(关联数组形式):

     $query = $request->query();

       通过动态属性获取输入

         可以通过 IlluminateHttpRequest 实例的动态属性访问用户输入。例如,如果应用表单包含 name 域,可以像下面这样访问该域的值:

     $name = $request->name;

         在使用动态属性时,Laravel 首先会在请求载体中查找参数的值。如果该值不存在,Lavarel 将在路由参数中搜索。

       获取 JSON 输入

         当向应用传递 JSON 请求时,可以通过 input 方法访问 JSON 数据,只要将请求的 Content-Type 头设置为 application/json。 同样可以使用 「点」语法访问 JSON 数组:

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

       获取部分输入数据

         如果需要获取输入数据的子集,可以使用 only 或 except 方法。它们接受单个 array 或者动态参数列表:

     $input = $request->only(['username', 'password']);     #only 只获取username,password
    
     $input = $request->only('username', 'password');
    
     $input = $request->except(['credit_card']);            #except 获取除了credit_card 之外其他的参数
    
     $input = $request->except('credit_card');

         注:only 方法返回请求中的全部键值对;但是它不返回请求中不存在的键值对。

       判断输入值是否存在

       has 方法用于判定请求中是否存在指定的值。如果请求中存在该值, has 方法返回 true :

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

         如果给出一个数组, has 方法将判断在请求中,指定的值是否全部存在:

     if ($request->has(['name', 'email'])) {
        //
     }

         如果想要判断一个值在请求中是否存在,并且不为空,需要使用 filled 方法:

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

    旧数据

         Laravel 允许你在两次请求之间保持数据。这个特性在有效性校验出错后重新填充表单时非常有用。不过,如果你使用 Lavarel 自带 验证特性,不需要自己手动调用这些方法因为一些 Laravel 内置的验证功能会自动调用它们。

       将输入数据传送到 Session

       IlluminateHttpRequest 类的 flash 方法将把当前的输入传送到 session ,在用户向应用发出这一次请求时它们仍然可用:

     $request->flash();

         可以使用 flashOnly 或 flashExcept 方法将请求数据的子集传送给 session。这些方法常用于将密码之类的敏感数据排除在 session 保持之外:

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

       传送数据并跳转

         当你经常需要将输入传送至 session 并紧接着跳转至之前的页面,可以通过在跳转函数后链接调用 withInput 方法轻易地实现:

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

       获取旧数据

         要获取前一次请求传送的数据,可以使用 Request 实例的 old 方法。 old 方法将从 session 拉取之前传送的值:

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

         Laravel 还提供了全局的 old 助手。如果要在 Blade模板中 显示旧数据, old 助手更易用。如果给定域的旧值不存在,它将返回 null :

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

    Cookies

       从请求中获取 Cookies

         Lavarel 框架生成的全部 cookies 都是加密的,并且已经用授权码签名,这意味着如果它们被客户端改变就会失效。使用 IlluminateHttpRequest 实例的 cookie 方法可以从请求中获取 cookie 值:

     $value = $request->cookie('name');

         也可以使用 Cookie facade 访问 cookie 值:

     $value = Cookie::get('name');

       将 Cookies 附加到响应中

         可以使用 cookie 方法向输出的 IlluminateHttpResponse 实例附加 cookie。需要传递 名称、值、cookie 的过期时间(以分钟为单位)给该方法:

     return response('Hello World')->cookie(
        'name', 'value', $minutes
     );

       cookie 还可以接受另外几个不太常用的参数。通常这些参数和 PHP 内置的  setcookie 方法的参数有着相同的作用和意义:

     return response('Hello World')->cookie(
        'name', 'value', $minutes, $path, $domain, $secure, $httpOnly
     );

         同样,你可以使用 Cookie facade 来 「排列」 用于从应用中附加到输出响应的 cookiesqueue 方法接受一个 Cookie 实例或者用于创建 Cookie 所需的参数列表。这些 cookies 将在输出响应被发送至浏览器前被附加:

     Cookie::queue(Cookie::make('name', 'value', $minutes));
    
     Cookie::queue('name', 'value', $minutes);

       生成 Cookie 实例

         如果想要生成一个随后可以提供给响应实例的 SymfonyComponentHttpFoundationCookie 实例,可以使用全局的 cookie 助手。这个 cookie 在没有附加到响应实例前不会传回客户端:

     $cookie = cookie('name', 'value', $minutes);
    
     return response('Hello World')->cookie($cookie);

    文件

       获取上传的文件

         可以使用 IlluminateHttpRequest 实例的 file 方法或者动态属性访问上传文件。 file 方法返回 IlluminateHttpUploadedFile 类的实例,这个类扩展自 PHP 的 SplFileInfo 类并提供用于文件交互的多个方法:

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

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

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

       验证成功上传

         除了验证文件是否存在,还可以使用 isValid 方法校验上传的文件有没有问题:

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

       文件路径 & 扩展名

       UploadedFile 类还包含访问文件的全路径和扩展名的方法。 

     $path = $request->photo->path();
     $extension = $request->photo->extension();                   #extension 方法基于文件的内容猜测匹配的文件扩展名。这个扩展名有可能和客户端提供的扩展名不同:
    $filename=$request->photo->getClientOriginalName(); #获取上传文件的文件名(带后缀,如abc.png)
    $fileextension=$request->photo->getClientOriginalExtension();#获取上传文件的后缀(如abc.png,获取到的为png)
    $filesize=$request->photo->getClientSize(); #获取上传文件的大小
    $filaname=$request->photo->getFilename(); #获取缓存在tmp目录下的文件名(带后缀,如php8933.tmp)
    $realpath=$request->photo->getRealPath(); #获取上传的文件缓存在tmp文件夹下的绝对路径
    $path=$request->photo->move(path,newname); #将缓存在tmp目录下的文件移到某个位置,返回的是这个文件移动过后的路径。
    #move()方法有两个参数,第一个参数是文件移到哪个文件夹下的路径,第二个参数是将上传的文件重新命名的文件名
     $request->photo->isValid()                                   #检测上传的文件是否合法,返回值为true或false

       保存上传的文件

         要保存上传的文件,需要使用你所配置的某个文件系统,对应配置位于 config/filesystems.php

         

         Laravel 默认使用 local 配置存放上传文件,即本地文件系统,默认根目录是 storage/apppublic 也是本地文件系统,只不过存放在这里的文件可以被公开访问,其对应的根目录是 storage/app/public,要让 Web 用户访问到该目录下存放文件的前提是在应用入口 public 目录下建一个软链 storage 链接到 storage/app/public

       UploadedFile 类有一个 store 方法,该方法会将上传文件移动到相应的磁盘路径上,该路径可以是本地文件系统的某个位置,也可以是云存储(如Amazon S3)上的路径。

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

       store 方法还接收一个可选的参数 —— 用于存储文件的磁盘名称作为第二个参数(对应文件系统配置 disks 的键名,默认值是 local),该方法会返回相对于根目录的文件路径:

     $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');

         下面来简单演示下文件上传功能,在 routes/api.php 中定义如下文件上传路由: 

    Route::post('file/upload', function(IlluminateHttpRequest $request) {
        if ($request->hasFile('photo') && $request->file('photo')->isValid()) {
            $photo = $request->file('photo');
            $extension = $photo->extension();
            //$store_result = $photo->store('photo');
            $store_result = $photo->storeAs('photo', 'test.jpg');
            $output = [
                'extension' => $extension,
                'store_result' => $store_result
            ];
            print_r($output);exit();
        }
        exit('未获取到上传文件或上传过程出错');
    });

    配置可信代理

       如果你的应用程序运行在失效的 TLS / SSL 证书的负载均衡器后,你可能会注意到你的应用程序有时不能生成 HTTPS 链接。通常这是因为你的应用程序正在从端口 80 上的负载均衡器转发流量,却不知道是否应该生成安全链接。

       解决这个问题需要在 Laravel 应用程序中包含 AppHttpMiddlewareTrustProxies 中间件,这使得你可以快速自定义应用程序信任的负载均衡器或代理。你的可信代理应该作为这个中间件的 $proxies 属性的数组列出。除了配置受信任的代理之外,还可以配置应该信任的代理 $header:

    <?php
    
    namespace AppHttpMiddleware;
    
    use IlluminateHttpRequest;
    use FideloperProxyTrustProxies as Middleware;
    
    class TrustProxies extends Middleware
    {
        /**
         * 应用程序的可信代理列表
         *
         * @var array
         */
        protected $proxies = [
            '192.168.1.1',
            '192.168.1.2',
        ];
    
        /**
         * 应该用来检测代理的头信息
         *
         * @var string
         */
        protected $headers = Request::HEADER_X_FORWARDED_ALL;
    }

         注:如果你使用 AWS 弹性负载平衡,你的 $header 值应该是 Request::HEADER_X_FORWARDED_AWS_ELB。常量的更多信息,可用于 $headers 属性,看看 Symfony 的 文档信任代理.

       信任所有代理

         如果你使用 Amazon AWS 或其他的「云」负载均衡器提供程序,你可能不知道负载均衡器的实际 IP 地址。在这种情况下,你可以使用 * 来信任所有代理:

     /**
      * 应用程序的可信代理列表
      *
      * @var array
      */
     protected $proxies = '*';
    
    
    
    
    
    
  • 相关阅读:
    一个简单的反反爬~
    查缺补漏 -- python 之 and or的优先级
    从今天开始看《Redis深度历险》--HyperLogLog
    从今天开始看《Redis深度历险》--位图
    从今天开始看《Redis深度历险》--延时队列
    从今天开始看《Redis深度历险》--分布式锁
    redis之set【官方文档搬运+翻译】
    从今天开始看《Redis深度历险》--基础
    collections模块学习之namedtuple
    元组赋值谜题
  • 原文地址:https://www.cnblogs.com/mzhaox/p/11262309.html
Copyright © 2011-2022 走看看