laravel路由是什么

laravel中,路由是外界访问Laravel应用程序的通路,或者说路由定义了Laravel的应用程序向外界提供服务的具体方式。路由会将用户的请求按照事先规划的方案提交给指定的控制器和方法来进行处理。

laravel路由是什么

本教程操作环境:windows7系统、Laravel6版,DELL G3电脑。

路由是外界访问Laravel应用程序的通路或者说路由定义了Laravel的应用程序向外界提供服务的具体方式:通过指定的URI、HTTP请求方法以及路由参数(可选)才能正确访问到路由定义的处理程序。

无论URI对应的处理程序是一个简单的闭包还是说是控制器方法没有对应的路由外界都访问不到他们

今天我们就来看看Laravel是如何来设计和实现路由的。

我们在路由文件里通常是向下面这样来定义路由的:

Route::get('/user', 'UsersController@index');

通过上面的路由我们可以知道,客户端通过以HTTP GET方式来请求 URI “/user”时,Laravel会把请求最终派发给UsersController类的index方法来进行处理,然后在index方法中返回响应给客户端。

上面注册路由时用到的Route类在Laravel里叫门面(Facade),它提供了一种简单的方式来访问绑定到服务容器里的服务router,Facade的设计理念和实现方式我打算以后单开博文来写,在这里我们只要知道调用的Route这个门面的静态方法都对应服务容器里router这个服务的方法,所以上面那条路由你也可以看成是这样来注册的:

app()->make('router')->get('user', 'UsersController@index');

router这个服务是在实例化应用程序Application时在构造方法里通过注册RoutingServiceProvider时绑定到服务容器里的:

//bootstrap/app.php$app = new IlluminateFoundationApplication(    realpath(__DIR__.'/../'));//Application: 构造方法public function __construct($basePath = null){    if ($basePath) {        $this->setBasePath($basePath);    }    $this->registerBaseBindings();    $this->registerBaseServiceProviders();    $this->registerCoreContainerAliases();}//Application: 注册基础的服务提供器protected function registerBaseServiceProviders(){    $this->register(new EventServiceProvider($this));    $this->register(new LogServiceProvider($this));    $this->register(new RoutingServiceProvider($this));}//IlluminateRoutingRoutingServiceProvider: 绑定router到服务容器protected function registerRouter(){    $this->app->singleton('router', function ($app) {        return new Router($app['events'], $app);    });}

通过上面的代码我们知道了Route调用的静态方法都对应于IlluminateRoutingRouter类里的方法,Router这个类里包含了与路由的注册、寻址、调度相关的方法。

下面我们从路由的注册、加载、寻址这几个阶段来看一下laravel里是如何实现这些的。

路由加载

注册路由前需要先加载路由文件,路由文件的加载是在AppProvidersRouteServiceProvider这个服务器提供者的boot方法里加载的:

class RouteServiceProvider extends ServiceProvider{    public function boot()    {        parent::boot();    }    public function map()    {        $this->mapApiRoutes();        $this->mapWebRoutes();    }    protected function mapWebRoutes()    {        Route::middleware('web')             ->namespace($this->namespace)             ->group(base_path('routes/web.php'));    }    protected function mapApiRoutes()    {        Route::prefix('api')             ->middleware('api')             ->namespace($this->namespace)             ->group(base_path('routes/api.php'));    }}
namespace IlluminateFoundationSupportProviders;class RouteServiceProvider extends ServiceProvider{    public function boot()    {        $this->setRootControllerNamespace();        if ($this->app->routesAreCached()) {            $this->loadCachedRoutes();        } else {            $this->loadRoutes();            $this->app->booted(function () {                $this->app['router']->getRoutes()->refreshNameLookups();                $this->app['router']->getRoutes()->refreshActionLookups();            });        }    }    protected function loadCachedRoutes()    {        $this->app->booted(function () {            require $this->app->getCachedRoutesPath();        });    }    protected function loadRoutes()    {        if (method_exists($this, 'map')) {            $this->app->call([$this, 'map']);        }    }}class Application extends Container implements ApplicationContract, HttpKernelInterface{    public function routesAreCached()    {        return $this['files']->exists($this->getCachedRoutesPath());    }    public function getCachedRoutesPath()    {        return $this->bootstrapPath().'/cache/routes.php';    }}

laravel 首先去寻找路由的缓存文件,没有缓存文件再去进行加载路由。缓存文件一般在 bootstrap/cache/routes.php 文件中。
方法loadRoutes会调用map方法来加载路由文件里的路由,map这个函数在AppProvidersRouteServiceProvider类中,这个类继承自IlluminateFoundationSupportProvidersRouteServiceProvider。通过map方法我们能看到laravel将路由分为两个大组:api、web。这两个部分的路由分别写在两个文件中:routes/web.php、routes/api.php。

Laravel5.5里是把路由分别放在了几个文件里,之前的版本是在app/Http/routes.php文件里。放在多个文件里能更方便地管理API路由和与WEB路由

路由注册

我们通常都是用Route这个Facade调用静态方法get, post, head, options, put, patch, delete……等来注册路由,上面我们也说了这些静态方法其实是调用了Router类里的方法:

public function get($uri, $action = null){    return $this->addRoute(['GET', 'HEAD'], $uri, $action);}public function post($uri, $action = null){    return $this->addRoute('POST', $uri, $action);}....

可以看到路由的注册统一都是由router类的addRoute方法来处理的:

//注册路由到RouteCollectionprotected function addRoute($methods, $uri, $action){    return $this->routes->add($this->createRoute($methods, $uri, $action));}//创建路由protected function createRoute($methods, $uri, $action){    if ($this->actionReferencesController($action)) {        //controller@action类型的路由在这里要进行转换        $action = $this->convertToControllerAction($action);    }    $route = $this->newRoute(        $methods, $this->prefix($uri), $action    );    if ($this->hasGroupStack()) {        $this->mergeGroupAttributesIntoRoute($route);    }    $this->addWhereClausesToRoute($route);    return $route;}protected function convertToControllerAction($action){    if (is_string($action)) {        $action = ['uses' => $action];    }    if (! empty($this->groupStack)) {                $action['uses'] = $this->prependGroupNamespace($action['uses']);    }        $action['controller'] = $action['uses'];    return $action;}

注册路由时传递给addRoute的第三个参数action可以闭包、字符串或者数组,数组就是类似[‘uses’ => ‘Controller@action’, ‘middleware’ => ‘…’]这种形式的。如果action是Controller@action类型的路由将被转换为action数组, convertToControllerAction执行完后action的内容为:

[    'uses' => 'AppHttpControllersSomeController@someAction',    'controller' => 'AppHttpControllersSomeController@someAction']

可以看到把命名空间补充到了控制器的名称前组成了完整的控制器类名,action数组构建完成接下里就是创建路由了,创建路由即用指定的HTTP请求方法、URI字符串和action数组来创建IlluminateRoutingRoute类的实例:

protected function newRoute($methods, $uri, $action){    return (new Route($methods, $uri, $action))                ->setRouter($this)                ->setContainer($this->container);}

路由创建完成后将Route添加到RouteCollection中去:

protected function addRoute($methods, $uri, $action){    return $this->routes->add($this->createRoute($methods, $uri, $action));}

router的$routes属性就是一个RouteCollection对象,添加路由到RouteCollection对象时会更新RouteCollection对象的routes、allRoutes、nameList和actionList属性

360安全路由器全屏滚动效果 360安全路由器全屏滚动效果

360安全路由器全屏滚动效果

360安全路由器全屏滚动效果 44 查看详情 360安全路由器全屏滚动效果

class RouteCollection implements Countable, IteratorAggregate{    public function add(Route $route)    {        $this->addToCollections($route);        $this->addLookups($route);        return $route;    }        protected function addToCollections($route)    {        $domainAndUri = $route->getDomain().$route->uri();        foreach ($route->methods() as $method) {            $this->routes[$method][$domainAndUri] = $route;        }        $this->allRoutes[$method.$domainAndUri] = $route;    }        protected function addLookups($route)    {        $action = $route->getAction();        if (isset($action['as'])) {            //如果时命名路由,将route对象映射到以路由名为key的数组值中方便查找            $this->nameList[$action['as']] = $route;        }        if (isset($action['controller'])) {            $this->addToActionList($action, $route);        }    }}

RouteCollection的四个属性

routes中存放了HTTP请求方法与路由对象的映射:

[    'GET' => [        $routeUri1 => $routeObj1        ...    ]    ...]

allRoutes属性里存放的内容时将routes属性里的二位数组编程一位数组后的内容:

[    'GET' . $routeUri1 => $routeObj1    'GET' . $routeUri2 => $routeObj2    ...]

nameList是路由名称与路由对象的一个映射表

[    $routeName1 => $routeObj1    ...]

actionList是路由控制器方法字符串与路由对象的映射表

[    'AppHttpControllersControllerOne@ActionOne' => $routeObj1]

这样就算注册好路由了。

路由寻址

中间件的文章里我们说过HTTP请求在经过Pipeline通道上的中间件的前置操作后到达目的地:

//IlluminateFoundationHttpKernelclass Kernel implements KernelContract{    protected function sendRequestThroughRouter($request)    {        $this->app->instance('request', $request);        Facade::clearResolvedInstance('request');        $this->bootstrap();        return (new Pipeline($this->app))                    ->send($request)                    ->through($this->app->shouldSkipMiddleware() ? [] : $this->middleware)                    ->then($this->dispatchToRouter());    }        protected function dispatchToRouter()    {        return function ($request) {            $this->app->instance('request', $request);            return $this->router->dispatch($request);        };    }    }

上面代码可以看到Pipeline的destination就是dispatchToRouter函数返回的闭包:

$destination = function ($request) {    $this->app->instance('request', $request);    return $this->router->dispatch($request);};

在闭包里调用了router的dispatch方法,路由寻址就发生在dispatch的第一个阶段findRoute里:

class Router implements RegistrarContract, BindingRegistrar{        public function dispatch(Request $request)    {        $this->currentRequest = $request;        return $this->dispatchToRoute($request);    }        public function dispatchToRoute(Request $request)    {        return $this->runRoute($request, $this->findRoute($request));    }        protected function findRoute($request)    {        $this->current = $route = $this->routes->match($request);        $this->container->instance(Route::class, $route);        return $route;    }    }

寻找路由的任务由 RouteCollection 负责,这个函数负责匹配路由,并且把 request 的 url 参数绑定到路由中:

class RouteCollection implements Countable, IteratorAggregate{    public function match(Request $request)    {        $routes = $this->get($request->getMethod());        $route = $this->matchAgainstRoutes($routes, $request);        if (! is_null($route)) {            //找到匹配的路由后,将URI里的路径参数绑定赋值给路由(如果有的话)            return $route->bind($request);        }        $others = $this->checkForAlternateVerbs($request);        if (count($others) > 0) {            return $this->getRouteForMethods($request, $others);        }        throw new NotFoundHttpException;    }    protected function matchAgainstRoutes(array $routes, $request, $includingMethod = true)    {        return Arr::first($routes, function ($value) use ($request, $includingMethod) {            return $value->matches($request, $includingMethod);        });    }}class Route{    public function matches(Request $request, $includingMethod = true)    {        $this->compileRoute();        foreach ($this->getValidators() as $validator) {            if (! $includingMethod && $validator instanceof MethodValidator) {                continue;            }            if (! $validator->matches($this, $request)) {                return false;            }        }        return true;    }}

$routes = $this->get($request->getMethod());会先加载注册路由阶段在RouteCollection里生成的routes属性里的值,routes中存放了HTTP请求方法与路由对象的映射。

然后依次调用这堆路由里路由对象的matches方法, matches方法, matches方法里会对HTTP请求对象进行一些验证,验证对应的Validator是:UriValidator、MethodValidator、SchemeValidator、HostValidator。
在验证之前在$this->compileRoute()里会将路由的规则转换成正则表达式。

UriValidator主要是看请求对象的URI是否与路由的正则规则匹配能匹配上:

class UriValidator implements ValidatorInterface{    public function matches(Route $route, Request $request)    {        $path = $request->path() == '/' ? '/' : '/'.$request->path();        return preg_match($route->getCompiled()->getRegex(), rawurldecode($path));    }}

MethodValidator验证请求方法, SchemeValidator验证协议是否正确(http|https), HostValidator验证域名, 如果路由中不设置host属性,那么这个验证不会进行。

一旦某个路由通过了全部的认证就将会被返回,接下来就要将请求对象URI里的路径参数绑定赋值给路由参数:

路由参数绑定

class Route{    public function bind(Request $request)    {        $this->compileRoute();        $this->parameters = (new RouteParameterBinder($this))                        ->parameters($request);        return $this;    }}class RouteParameterBinder{    public function parameters($request)    {        $parameters = $this->bindPathParameters($request);        if (! is_null($this->route->compiled->getHostRegex())) {            $parameters = $this->bindHostParameters(                $request, $parameters            );        }        return $this->replaceDefaults($parameters);    }        protected function bindPathParameters($request)    {            preg_match($this->route->compiled->getRegex(), '/'.$request->decodedPath(), $matches);            return $this->matchToKeys(array_slice($matches, 1));    }        protected function matchToKeys(array $matches)    {        if (empty($parameterNames = $this->route->parameterNames())) {            return [];        }        $parameters = array_intersect_key($matches, array_flip($parameterNames));        return array_filter($parameters, function ($value) {            return is_string($value) && strlen($value) > 0;        });    }}

赋值路由参数完成后路由寻址的过程就结束了,结下来就该运行通过匹配路由中对应的控制器方法返回响应对象了。

class Router implements RegistrarContract, BindingRegistrar{        public function dispatch(Request $request)    {        $this->currentRequest = $request;        return $this->dispatchToRoute($request);    }        public function dispatchToRoute(Request $request)    {        return $this->runRoute($request, $this->findRoute($request));    }        protected function runRoute(Request $request, Route $route)    {        $request->setRouteResolver(function () use ($route) {            return $route;        });        $this->events->dispatch(new EventsRouteMatched($route, $request));        return $this->prepareResponse($request,            $this->runRouteWithinStack($route, $request)        );    }        protected function runRouteWithinStack(Route $route, Request $request)    {        $shouldSkipMiddleware = $this->container->bound('middleware.disable') &&                            $this->container->make('middleware.disable') === true;    //收集路由和控制器里应用的中间件        $middleware = $shouldSkipMiddleware ? [] : $this->gatherRouteMiddleware($route);        return (new Pipeline($this->container))                    ->send($request)                    ->through($middleware)                    ->then(function ($request) use ($route) {                        return $this->prepareResponse(                            $request, $route->run()                        );                    });        }    }namespace IlluminateRouting;class Route{    public function run()    {        $this->container = $this->container ?: new Container;        try {            if ($this->isControllerAction()) {                return $this->runController();            }            return $this->runCallable();        } catch (HttpResponseException $e) {            return $e->getResponse();        }    }}

这里我们主要介绍路由相关的内容,runRoute的过程通过上面的源码可以看到其实也很复杂, 会收集路由和控制器里的中间件,将请求通过中间件过滤才会最终到达目的地路由,执行目的路由地run()方法,里面会判断路由对应的是一个控制器方法还是闭包然后进行相应地调用,最后把执行结果包装成Response对象返回给客户端。这个过程还会涉及到我们以前介绍过的中间件过滤、服务解析、依赖注入方面的信息。

相关推荐:最新的五个Laravel视频教程

以上就是laravel路由是什么的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/627153.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
win8.1显卡驱动装不上怎么解决
上一篇 2025年11月11日 09:26:56
我的世界手机网页版怎么进-我的世界手机网页版入口地址链接一览
下一篇 2025年11月11日 09:27:07

相关推荐

  • composer require-dev和require有什么不同_Composer Require与Require-Dev区别解析

    require用于声明项目运行必需的依赖,如框架、数据库组件和第三方SDK,这些包会随项目部署到生产环境;2. require-dev用于声明仅在开发和测试阶段需要的工具,如PHPUnit、PHPStan、Faker等,不会默认部署到生产环境;3. 安装时composer install根据环境决定…

    2026年5月10日
    1000
  • 开源免费PHP工具 PHP开发效率提升利器

    推荐开源免费PHP开发工具以提升效率:VS Code、Sublime Text轻量高效,PhpStorm专业强大;调试用Xdebug、Kint、Ray;依赖管理选Composer;代码质量工具包括PHPStan、Psalm、PHP_CodeSniffer;数据库管理可用%ignore_a_1%MyA…

    2026年5月10日
    000
  • 深入理解 Express.js 中 next() 参数的作用与中间件机制

    本文深入探讨 express.js 中间件函数中的 `next()` 参数。它负责将控制权传递给请求-响应周期中的下一个中间件或路由处理程序。文章将详细解释 `next()` 的工作原理、中间件的注册与执行顺序,以及不正确使用 `next()` 可能导致请求挂起的风险,并通过代码示例和实际应用场景,…

    2026年5月10日
    000
  • Discord.py 交互按钮超时与持久化解决方案

    本教程旨在解决Discord.py中交互按钮在一段时间后出现“This Interaction Failed”错误的问题。我们将深入探讨视图(View)的超时机制,并提供通过正确设置timeout参数以及利用bot.add_view()方法实现按钮持久化的具体方案,确保您的机器人交互功能稳定可靠,即…

    2026年5月10日
    000
  • 深入理解MQTT多级通配符#的用法限制与Paho-MQTT订阅实践

    本文旨在解析mqtt多级通配符`#`在订阅主题时的严格使用规则,尤其是在paho-mqtt库中遇到的`valueerror: ‘invalid subscription filter.’`问题。我们将详细阐述mqtt规范中关于`#`必须作为主题过滤器最后一个字符的规定,并通过…

    2026年5月10日
    000
  • CodeIgniter在IIS环境下实现URL重写与index.php移除指南

    本教程详细指导如何在IIS服务器上部署的CodeIgniter应用中,移除URL中不必要的index.php。核心解决方案涉及修改CodeIgniter的config.php文件,将$config[‘index_page’]设置为空,并辅以正确的IIS web.config重…

    2026年5月10日
    100
  • 深入理解 Laravel Session::put:避免常见陷阱与实现表单限流

    本文旨在深入探讨 laravel 框架中 `session::put` 方法的正确用法及其常见误区。针对用户在实现表单提交限流时遇到的问题,详细阐述了 `session::put` 必须提供键值对的原理,并提供了如何在控制器中利用会话机制有效防止重复提交的实战代码示例。通过本文,读者将掌握 lara…

    2026年5月10日
    000
  • Voyager 中关联关系的翻译问题解决方案

    本文档旨在解决在使用 TCGVoyager 管理后台时,关联模型无法正确翻译的问题。主要针对 Laravel 项目中,使用 Voyager 1.4 版本以及 Laravel 8.0 版本,并且已经配置多语言支持的情况下,如何确保关联关系中的可翻译字段能够根据当前应用语言环境进行正确翻译。通过修改 B…

    2026年5月10日
    000
  • C# 怎么使用 Serilog 或 NLog 记录日志_C# 日志记录框架使用指南

    Serilog和NLog是.NET中常用日志框架,Serilog支持结构化日志,配置简洁,适合集成Seq、Elasticsearch;NLog配置灵活,支持复杂规则,适用于企业级应用。两者均通过NuGet安装,配合配置文件或代码初始化,并通过ILogger接口写入日志,可根据项目需求选择其一。 在 …

    2026年5月10日
    000
  • React Redux 中 useSelector 的自动订阅与取消订阅机制

    React Redux 中 useSelector 的自动订阅与取消订阅机制React Redux 中 useSelector 的自动订阅与取消订阅机制React Redux 中 useSelector 的自动订阅与取消订阅机制React Redux 中 useSelector 的自动订阅与取消订阅机制

    本文深入探讨 react redux 中 `useselector` hook 的核心机制。它详细解释了 `useselector` 如何在组件挂载时自动订阅 redux store 的状态更新,并在组件卸载时智能地取消订阅。这确保了应用程序的性能和内存效率,避免了对已卸载组件进行不必要的更新,从而…

    2026年5月10日 用户投稿
    100
  • C#怎么进行UDP通信 C# UdpClient实现UDP协议编程

    使用UdpClient类可简化C#中的UDP通信。1. 发送数据:创建UdpClient实例,调用Send()方法指定目标IP和端口,如向127.0.0.1:8888发送”Hello UDP!”;2. 接收数据:绑定端口(如8888),使用Receive()阻塞等待数据,通过…

    2026年5月10日
    100
  • Nginx 子目录应用URI重写与参数传递教程

    本教程详细阐述了如何在Nginx中为PHP应用实现子目录URI重写,特别是如何从请求URI中剥离子目录路径并将其余部分作为参数传递给主入口文件。通过try_files和rewrite指令的组合,本教程提供了一种高效且准确的解决方案,以替代Apache .htaccess的RewriteRule功能,…

    2026年5月10日
    000
  • 优化 Laravel Eloquent 查询:高效构建用户排行榜数据

    本教程详细讲解如何优化 Laravel Eloquent 查询以高效生成基于关联记录计数的排行榜。通过识别并消除冗余的 whereHas 子句,并巧妙利用 withCount 的条件闭包,我们能显著提升查询性能,大幅缩短数据获取时间,从而改善用户体验并降低数据库负载。 在 laravel 应用开发中…

    2026年5月10日
    000
  • 多表单单按钮提交与Flask后端处理教程

    本教程将详细介绍如何使用JavaScript和Flask处理通过一个按钮提交多个HTML表单的场景。我们将探讨直接提交的局限性,并提供基于XMLHttpRequest的异步提交解决方案,确保所有表单数据都能被Flask后端正确接收和处理,避免仅接收到最后一个表单数据的问题。 1. 多表单提交的挑战 …

    2026年5月10日
    000
  • 在Python Flask中实现在线图片URL到Blurhash编码

    本教程详细介绍了如何在python flask应用中,将在线图片url转换为blurhash键。针对官方文档主要侧重本地文件处理的局限,文章通过整合`requests`库下载图片内容和`blurhash-python`库进行编码,提供了完整的解决方案,并包含代码示例、依赖安装、错误处理及在flask…

    2026年5月10日
    000
  • Golang如何实现基础的用户权限控制

    答案:Go语言中通过用户角色模型、中间件权限检查和上下文传递实现基础权限控制,结合JWT或Session认证,利用路由中间件限制访问,适合中小型项目快速搭建。 在Go语言中实现基础的用户权限控制,核心思路是结合用户身份认证与权限判断逻辑。通常通过中间件、角色系统和路由控制来完成。下面介绍一种简单但实…

    2026年5月10日
    000
  • 告别重复:使用Laravel Precognition统一前后端API验证

    本文旨在解决在Laravel后端与前端API交互中,如何高效复用后端验证规则的挑战。传统方案常限于表单元素,难以覆盖所有API请求。通过引入Laravel Precognition,开发者能够实现后端验证逻辑在前端的无缝应用,避免规则重复编写,从而提升开发效率与代码一致性,确保所有API请求的数据完…

    2026年5月10日
    200
  • Laravel Session::put 正确用法详解与常见误区规避

    本文详细探讨了 laravel 中 `session::put` 方法的正确用法,特别指出在仅提供键名而未指定值时可能导致会话数据未被正确设置的问题。通过示例代码,阐述了如何为会话数据赋予明确的值,并演示了如何正确地检查和获取会话数据,以确保会话管理功能按预期工作,有效避免常见的会话操作错误。 La…

    2026年5月10日
    000
  • Go应用中基于gorilla/mux的模块化路由管理策略

    本文探讨了在go应用中使用`gorilla/mux`实现模块化路由的有效策略。针对大型应用中路由配置日益复杂的问题,我们提出了一种去中心化的解决方案:通过在各个模块的`init()`函数中注册其专属路由到全局路由表,`main`函数统一加载,从而实现路由的清晰分离与高效管理,提升代码可维护性。 在构…

    2026年5月10日
    000
  • PHP中批量为嵌套数组元素添加公共属性的教程

    本教程将详细介绍在php中如何高效地为包含多个关联数组的集合中的每个子数组添加一个或多个新的公共键值对。我们将探讨使用循环和数组合并函数实现这一目标的方法,并提供清晰的代码示例,帮助开发者处理此类数据结构转换。 在PHP开发中,我们经常会遇到处理复杂数据结构的需求,其中一种常见场景是拥有一个由多个关…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信