在本文中,我们将讨论Magento 2 Routing中的一个重要部分。该路线将为此我们可以在URL中使用查找模块和执行控制动作的模块定义名称。
目录
- Magento 2请求流程
- 在前端/管理员上创建自定义路由
- 前端路线
- routes.xml
- 管理路线
- 使用route重写控制器
- 前端路线
Magento 2请求流程
在Magento 2中,请求URL将如下所示:
http://example.com/index.php/front_name/controller/action
在该URL中,您将看到front_name
用于查找模块的URL 。路由器通过routes.xml中的define定义每个模块的名称,我们将在下面看到更多细节。
当您在Magento 2中提出请求时,它将按照此流程查找controller/action
: index.php → HTTP app → FrontController → Routing → Controller processing → etc
该FrontController
会在HTTP中类中调用路由信息,会发现该请求controller/action
匹配。
文件: vendor/magento/framework/App/FrontController.php
public function dispatch(RequestInterface $request)
{
\Magento\Framework\Profiler::start('routers_match');
$routingCycleCounter = 0;
$result = null;
while (!$request->isDispatched() && $routingCycleCounter++ < 100) {
/** @var \Magento\Framework\App\RouterInterface $router */
foreach ($this->_routerList as $router) {
try {
$actionInstance = $router->match($request);
if ($actionInstance) {
$request->setDispatched(true);
$this->response->setNoCacheHeaders();
if ($actionInstance instanceof \Magento\Framework\App\Action\AbstractAction) {
$result = $actionInstance->dispatch($request);
} else {
$result = $actionInstance->execute();
}
break;
}
} catch (\Magento\Framework\Exception\NotFoundException $e) {
$request->initForward();
$request->setActionName('noroute');
$request->setDispatched(false);
break;
}
}
}
\Magento\Framework\Profiler::stop('routers_match');
if ($routingCycleCounter > 100) {
throw new \LogicException('Front controller reached 100 router match iterations');
}
return $result;
}
正如您在此dispatch()
方法中所看到的,路由器列表将循环以查找与此请求匹配的路由器列表。如果它找到该请求的控制器操作,则将调用并执行该操作。
在前端/管理员上创建自定义路由
在这一部分中,我们将使用一个简单的模块Mageplaza_HelloWorld。请按照上一篇文章了解如何在Magento 2中创建模块。
我们将找到如何创建前端路由,管理路由以及如何使用路由重写控制器。
前端路线
Routes.xml
要注册前端路由,我们必须创建一个routes.xml文件:
文件: app/code/Mageplaza/HelloWorld/etc/frontend/routes.xml
<?xml version="1.0" ?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:App/etc/routes.xsd">
<!--Use router 'standard' for frontend route-->
<router id="standard">
<!--Define a custom route with id and frontName-->
<route frontName="helloworld" id="helloworld">
<!--The module which this route match to-->
<module name="Mageplaza_HelloWorld"/>
</route>
</router>
</config>
请查看代码,您将看到注册路由非常简单。您必须使用标准路由器作为前端。这条路线将有一个孩子为它定义模块和2个属性:
- id属性是一个唯一的字符串,用于标识此路由。您将使用此字符串声明此模块操作的布局句柄
- frontName属性也是一个唯一的字符串,它将显示在url请求中。例如,如果您声明这样的路线:
<route frontName="helloworld" id="helloworld">
该模块的URL应该是:
http://example.com/index.php/helloworld/controller/action
此操作的布局句柄是:helloworld_controller_action.xml
因此,使用此示例路径,您必须在此文件夹中创建操作类:{namespace}/{module}/Controller/{Controller}/{Action}.php
管理路线
此路由与前端路由相同,但您必须在adminhtml文件夹中将其声明为路由器ID为admin
。
文件: app/code/Mageplaza/HelloWorld/etc/adminhtml/routes.xml
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:App/etc/routes.xsd">
<!--Use router 'admin' for admin route -->
<router id="admin">
<!--Define a custom route with id and frontName -->
<route id="mageplaza_helloworld" frontName="mageplaza_helloworld">
<!--The module which this route match to-->
<module name="Mageplaza_HelloWorld"/>
</route>
</router>
</config>
管理页面的URL与前端页面的结构相同,但是在识别这是管理员路由器admin_area
之前将添加名称route_frontName
。例如,admin cms页面的url :
http://example.com/index.php/admin/mageplaza_helloworld/controller/action
管理页面的控制器操作将添加到文件夹内Controller/Adminhtml
。例如,对于以上网址:
{namespace}/{module}/Controller/Adminhtml/{Controller}/{Action}.php
使用route重写控制器
在这条路径中,我们将看到如何用路由器重写控制器。如上面的路径,您可以看到每个路由都有一个id属性来识别。那么如果我们定义具有相同id属性的2路由会发生什么?
答案是控制器操作将在两个模块中找到。Magento系统在配置模块排序顺序之前/之后提供属性,该顺序定义了首先找到的模块控制器。这是控制器重写的逻辑。
例如,如果我们想重写控制器customer/account/login
,我们将在route.xml中定义更多路由,如下所示:
文件: app/code/Mageplaza/HelloWorld/etc/frontend/routes.xml
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:App/etc/routes.xsd">
<!--Use router 'standard' for frontend route-->
<router id="standard">
<!--Define a custom route with id and frontName-->
<route frontName="helloworld" id="helloworld">
<!--The module which this route match to-->
<module name="Mageplaza_HelloWorld"/>
</route>
<route id="customer">
<module name="Mageplaza_HelloWorld" before="Magento_Customer" />
</route>
</router>
</config>
和控制器文件: app/code/Mageplaza/HelloWorld/Controller/Account/Login.php
所以frontController 首先在我们的模块中找到Login动作,如果找到它,它将运行并且不会运行Login动作Magento_Customer
。我们成功地重写了一个控制器。
您也可以使用它来使第二个模块与另一个模块具有相同的路由器。例如,通过上述声明,您可以使用路由'customer'作为控制器操作。如果您有控制器'博客'和操作'Index.php',您可以使用此网址:
http://example.com/customer/blog/index