Laravel 单行为控制器设计的魅力

Laravel 单行为控制器设计的魅力

昨天,Jeffrey Way 发布了一条推文,他问大家更愿意将其控制器命名为单数还是复数。 我回答我两种方案都不选,我使用单动作控制器。随后发生的是,有的人同意,有的不同意,有的甚至做出了最奇怪的事情。

由于十分强烈的反映,我想写一篇文章来解释为什么我爱单行为控制器、还有我为什么觉得它们很美妙。

首先在开始文章之前,我想要说这个东西并不是只有单一的真相。与往常一样,我想指出的是,一切都归结于你的个人喜好。我只能教、建议和指出一些事情,由你来决定是否同意、不同意、接受、学习和 / 或调整。或者都不是。从这篇博客中获得你想要的,随心所欲地做让自己感到舒适的事情吧。

对比 CRUD 和 Domain Modelling

开始前,我们先来想想我们倾向于写 resourceful 的 CRUD 控制器。我相信很多人会坚持使用这种做法,因为这是 Laravel 中的一个标准做法,文档中的大多数示例也是使用这种方法。另外,这或许也是你在各类博客或 app 代码中经常看到的。

但是,如果你停下来思考一下,这是编写它们的最佳方法吗?是软件行业的一般性做法吗?最近几年,我在 “领域驱动设计”(Domain Driven Design) 等领域投入了大量时间,并且思考软件如何应用于你工作的领域(Domian)以及它转化的过程。当您开始考虑模仿您领域中无处不在的语言的术语和措辞时,您会发现您的代码将变得更加清晰明了,更加戳到点子上。(最后这一句话仍值得斟酌、改进)

最后,我相信编写软件的本质是尽可能地应用 domain processes 来让你的代码更加易读和更加可维护。

Resourceful 控制器在这两个方面做得并不好。首先,它们不易读,因为您倾向于根据数据来构建它们,而不是根据领域来构建它们。这样的话,你就会丢失上下文对照。你表现了数据的处理方式,但却没有说明到底发生什么了,也没有说明你使用哪个过程进行处理。

第二,你没有针对可维护性进行优化。由于你是根据数据结构来构建的,因此你也会跟着耦合进去。实际上,您的领域模型在不断发展,数据结构也在不断发展。如果你的数据结构处理着多个过程或领域的多个部分,那你将很难进行调整。

一个实际的例子

因为理论很无聊,上代码更加容易解释,所以我们来看一个实际的例子。

假设您正在构建一个应用,它允许用户去组织事件。您想提供一种创建,更新和删除这些事件的方法。这是一种非常典型的例子,你会用 CRUD 的方式来考虑实现它。那么,让我们看看就这样一个 resourceful 控制器是如何被转换的。

首先我们来看看路由:

Route::get('events', [EventController::class, 'index']);Route::get('events/create', [EventController::class, 'create']);Route::post('events', [EventController::class, 'store']);Route::get('event/{event}', [EventController::class, 'show']);Route::get('events/{event}/edit', [EventController::class, 'edit']);Route::put('events/{event}', [EventController::class, 'update']);Route::destroy('events/{event}', [EventController::class, 'destroy']);

现在对应的控制器:

<?phpnamespace AppHttpControllers;use AppModelsEvent;final class EventController{    public function index()    {        // ...    }    public function create()    {        // ...    }    public function store()    {        // ...    }    public function show(Event $event)    {        // ...    }    public function edit(Event $event)    {        // ...    }    public function update(Event $event)    {        // ...    }    public function destroy(Event $event)    {        // ...    }}

这个 EventController 处理所有的 CRUD 请求,展示事件列表,展示指定的事件,创建一个事件,更新一个现存的事件和删除一个事件。

来看看 index 方法的细节:

public function index(){    $events = Event::paginate(10);    return view('events.index', compact('events'));}

在这个方法中,我们检索出事件们,然后交给视图让它去展示到一个分页列表中。 到目前为止都还好。但是你现在想实现一个方法,用不同的页面去查看过去和即将来的事件。让我们看看如何在 index 方法中实现它:

public function index(Request $request){    if ($request->boolean('past')) {        $events = Event::past()->paginate(10);    } elseif ($request->boolean('upcoming')) {        $events = Event::upcoming()->paginate(10);    } else {        $events = Event::paginate(10);    }    return view('events.index', compact('events'));}

呃啊!看起来好乱啊。尽管我们已经用 Eloquent scopes 来隐藏查询逻辑,但是还是有很丑的链式语句。我们来看看如何用单行为控制器来代替它。

每个单行为控制器只执行一件事情,仅仅一件事情。

首先,我们不使用查询参数去获得不同的事件列表,而是使用专用路由去实现它。

Route::get('events', ShowAllEventsController::class);Route::get('events/past', ShowPastEventsController::class);Route::get('events/upcoming', ShowUpcomingEventsController::class);

这个路由比之前的要长一些,但是这个比之前的要更有表达力。你可以一下子辨识出哪一个控制器处理哪一个特定的逻辑。如果你对比一下 URL,你会看到在可读性上改进了一些:

# Before/events/events?past=true/events?upcoming=true# After/events/events/past/events/upcoming

现在来看其中一个控制器。就看 ShowUpcomingEventsController 这个控制器:

paginate(10);        return view('events.index', compact('events'));    }}

丑陋的 if 语句没了, and has made way for the same readable three liner we had from our first CRUD controller example. But instead of having all of the other CRUD operations we now have a dedicated controller for a dedicated action.

简单,易读,便于维护。

你可能会问自己,这样做值么,毕竟之前的 if 语句也没那么坏吧?但是我想向你展示的是你正在为未来的改进做优化,并改进维护性。下次你想要对这三个页面做任何指定改变的时候,你会知道在哪里改,并且不需要艰难地更新一个 if 语句。

当然,上面的例子很简单,我们来看一个更复杂一点的。我们试试重构 create 和 store 方法:

public function create(){    return view('events.create');}public function store(Request $request){    $data = $request->validate([        'name' => 'required',        'start' => 'required',        'end' => 'required|after:start',    ])    $event = Event::create($data);    return redirect()->route('event.show', $event);}

我们要做的就是把这两个方法移到专用的控制器,这样更好地解释了这些方法做了啥。这些方法更好地服务于你,比起把它们放在一个叫做 ScheduleNewEventController 的控制器中。我们接着更新这个控制器的路由:

Route::get('events/schedule', [ScheduleNewEventController::class, 'showForm']);Route::post('events/schedule', [ScheduleNewEventController::class, 'schedule']);

我不会向你展示一个确切的控制器,因为它们有和上面的例子一样,有两个方法,只不过把 showForm 和 schedule 重新命名为更能表达它们干了啥的名字。即使这个不是单行为控制器,但是方法论是一样的:把你应用中的专用行为(方法)和它对应的控制器拆分到一起。

好了,现在你已经看了单行为控制器的例子了。你可能会想,这会导致越来越多的文件。但事实上,这个根本就不是问题。文件多又没啥。有更多、更小、更容易维护的文件比有更大、更难分析的要好。你可以打开一个单行为控制器的文件,然后快速扫描代码,马上就能知道这是干嘛的。

我经常把他们分组到不同的目录,这些目录负责领域的各个部分。这让你从文件结构的角度看控制器时,更加容易。

拆分控制器也让你跟容易找到特定的一个控制器。想象一下,你要寻找那个可以安排事件的控制器时。现在你只需要按照文件名搜索编辑器,而不是一个通用的 EventController。

其他情况

我也被问到是否要对所有控制器执行此操作。不总是。在命名控制器时,我倾向于严谨且简洁,但我也会像你一样适应各种情况。

当然,有时候你还是想用 resourceful 控制器。比如在你构建 RESTful API 时。这样做是很有意义,因为你经常直接与数据本身交互,而没有经常与领域或任何进程进行交互。CMS(内容管理系统)或 Laravel Nova 等应用程序就是最好的例子。

但是在需要的时候,您最好问问自己的方案是否更接近领域和处理过程。在需要根据领域执行操作的时候,比如 GraphQL 之类的或 API 之类的 RPC ,这样做可能更适合。

结论

我希望这有一点见地,你现在能更理解我为什么如此喜欢单行为控制器了吧。我相信,结合小的 classes,再使用无处不在的语言、显式地命名,会带来更可维护的代码,甚至是控制器,不仅仅是领域对象。但是正如我开头所说,选择能帮助你的部分,好好分辨哪些适用于你,哪些不行。

推荐教程:《PHP教程》《Laravel教程》

以上就是Laravel 单行为控制器设计的魅力的详细内容,更多请关注php中文网其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月18日 05:28:03
下一篇 2025年11月18日 05:55:34

相关推荐

  • PHP 中堆栈溢出对递归函数的影响:性能下降和崩溃

    堆栈溢出对 php 递归函数的影响:性能下降:增加的堆栈帧数量会减慢执行速度。崩溃:当堆栈填满时会导致程序崩溃,中断关键操作。实战案例展示了对于大型输入,递归 fibonacci 函数执行时间的显著差异。尾递归优化可以通过将递归调用转换为迭代循环来减轻堆栈溢出风险。 PHP 中堆栈溢出对递归函数的影…

    2025年12月9日
    000
  • PHP 函数中堆栈溢出的原因和解决方法

    php 函数中堆栈溢出是由过多的函数嵌套调用引起的。解决方法包括:减少递归深度、避免无限循环、重构嵌套函数、增加 php 内存限制、使用尾递归优化。通过遵循这些准则,可以防止函数中的堆栈溢出。 PHP 函数中堆栈溢出的原因和解决方法 简介 堆栈溢出是一种常见的编程错误,当函数调用过多时就会发生。在 …

    2025年12月9日
    000
  • PHP 程序设计中匿名函数的最佳实践

    php 匿名函数最佳实践:保持函数简洁;正确使用范围;使用类型提示;限制匿名函数的深度;正确处理异常。 PHP 程序设计中匿名函数最佳实践 简介 匿名函数,也称为闭包,在 PHP 中是强大而灵活的工具。它们允许您定义即时执行的代码块,无需将它们分配给变量。在本篇文章中,我们将探讨匿名函数的最佳实践,…

    2025年12月9日
    000
  • PHP 函数参数绑定如何减少代码重复性?

    参数绑定是一种 php 技术,可通过以下步骤减少重复代码:准备带有占位符的 sql 查询。创建预处理语句对象。使用 bind_param() 方法绑定参数。执行查询。关闭预处理语句对象。通过参数绑定,我们只需更新实际参数即可执行多个查询,从而简化代码并提高可维护性。 PHP 函数参数绑定:减少代码重…

    2025年12月9日
    000
  • 解决 PHP 递归函数堆栈溢出的方法

    解决 php 递归函数堆栈溢出问题的四种方法:优化代码,最小化递归调用的次数;增加 php 限制,提高最大堆栈深度;使用尾部递归,递归调用不增加堆栈深度;使用遍历方法,模拟递归行为。 解决 PHP 递归函数堆栈溢出的方法 简介 递归函数是一种通过调用自身来解决问题的函数。当递归调用次数过多时,可能会…

    2025年12月9日
    000
  • 使用 PHP 匿名函数中的命名空间?

    php 匿名函数可以在命名空间中定义,继承命名空间中的类、常量和函数。为了访问外部变量,匿名函数可以使用 use 子句捕获它们,即使是在命名空间之外声明的变量。 使用 PHP 匿名函数中的命名空间 PHP 匿名函数提供了一种定义不需要命名函数的便捷方式。但是,匿名函数不能直接访问其包含范围之外的变量…

    2025年12月9日
    000
  • PHP 堆栈溢出的未来发展趋势

    php堆栈溢出是一种运行时错误,未来php将通过优化以下方面来解决:优化jit编译器,提高检测和防止堆栈溢出的能力;引入动态堆栈分配,动态调整堆栈大小;改善错误处理机制,更易于检测和处理堆栈溢出。 PHP 堆栈溢出的未来发展趋势 堆栈溢出是一种当函数调用深度达到操作系统或虚拟机 (VM) 允许的最大…

    2025年12月9日
    000
  • 如何理解和应用 PHP 应用程序中的命名空间

    命名空间在 php 中用于将类和函数分组,以避免名称冲突并组织代码。声明命名空间时使用 namespace 关键字,如 namespace myprojectcontrollers;。要使用命名空间成员,请使用反斜杠转义符,如 $usercontroller = new myprojectcontr…

    2025年12月9日
    000
  • 异步 PHP 函数如何避免堆栈溢出

    避免异步 php 函数堆栈溢出的技巧包括:使用生成器分块执行函数,避免创建新堆栈帧。使用协程在不同堆栈帧之间切换,避免创建新堆栈帧。使用 exec() 函数限制 php 的最大堆栈深度。 异步 PHP 函数避免堆栈溢出的技巧 在 PHP 中使用异步函数时,可能会碰到堆栈溢出问题,这会导致脚本崩溃。本…

    2025年12月9日
    000
  • 在 Laravel 11 中的支付处理编译时实现上下文绑定

    在我们之前的文章(如何在 laravel 11 中添加和实现支付处理接口:硬编码绑定)中,我们通过硬编码 paymentprocessorinterface 和特定实现(如 stripepaymentprocessor)之间的绑定来探索设置支付处理器的第一步。 虽然这种方法对于小型应用程序来说简单有…

    2025年12月9日
    000
  • 命名参数在 PHP 生态系统中其他组件(如框架)中的应用

    命名参数在 PHP 生态系统中其他组件的应用 在 PHP 中,命名参数允许你按名称将参数传递给函数或方法,而不是按位置。这提供了更好的可读性和可维护性,尤其是在处理具有许多参数的函数时。 除了核心 PHP,命名参数还在框架和库中得到了广泛应用,例如: Laravel Framework 立即学习“P…

    2025年12月9日
    000
  • PHP 中递归函数的堆栈限界:如何确定和调整

    php 中递归函数的堆栈限界默认值为 100,可通过 echo ini_get(‘max_recursion_depth’) 确定。调整堆栈限界需要修改 php.ini 文件中的 max_recursion_depth 设置。对于较大的递归函数,如阶乘计算,通过增加堆栈限界可…

    2025年12月9日
    000
  • PHP 参数绑定与 NoSQL 数据库的兼容性

    nosql数据库中参数绑定兼容性因数据库类型而异:mongodb:使用函数参数传递实现类似于参数绑定的功能。redis:不支持参数绑定。cassandra:支持使用prepared statement实现参数绑定。hbase:不支持参数绑定。dynamodb:支持使用expressionattrib…

    2025年12月9日
    000
  • PHP 函数的函数指针如何提高代码可重用性?

    摘要:php 函数指针允许您调用其他函数,提升代码重用性和灵活性,语法为 $func = ‘function_name’; 优势包括:代码重用性:创建可重复使用相同函数逻辑的不同函数。灵活性:动态选择要调用的函数。可扩展性:轻松添加新函数。解耦:提高代码的可维护性。 利用 P…

    2025年12月9日
    000
  • 如何通过调整 PHP 配置来预防堆栈溢出?

    通过调整 php 配置中的 memory_limit 设置来增加堆栈大小,可以有效防止堆栈溢出。建议的堆栈大小为 256m 至 512m,但需根据应用程序的实际情况灵活调整。此外,避免过多的递归调用、限制函数调用深度和使用内存剖析器识别内存泄漏等措施也有利于预防堆栈溢出。 如何通过调整 PHP 配置…

    2025年12月9日
    000
  • 可变参数在 PHP 数组中扮演什么角色?

    php 中的可变参数允许向函数传递任意数量的参数,提供灵活性。语法:三个点(…)表示可变参数,必须是参数列表最后一个。实战:可变参数可用于创建接受未知数量输入的函数,例如求和函数。可变参数的作用包括:灵活性、简洁性和代码重用性。 可变参数:PHP 数组的强大工具 在 PHP 中,可变参数…

    2025年12月9日
    000
  • 使用命名参数在 PHP 函数中实现函数重载

    答案: php 中的命名参数可实现函数重载。详细描述:命名参数允许使用不同的参数名称调用同一函数,执行不同操作。在函数定义中,为可选参数指定默认值,并通过参数名称匹配进行调用。命名参数提供了类似函数重载的机制,使一个函数可以处理不同类型和数量的参数,实现更灵活的代码。 使用命名参数在 PHP 函数中…

    2025年12月9日
    000
  • PHP 函数递归调用的堆栈溢出处理方法

    php 函数递归调用堆栈溢出处理方法有 3 种:1. 提高 php 堆栈大小;2. 使用尾递归优化;3. 使用迭代。尾递归优化是指将最后一次递归调用置于函数末尾,以减少堆栈空间使用。 PHP 函数递归调用的堆栈溢出处理方法 PHP 中的递归调用是指一个函数调用自身。虽然递归可以是一种强大的编程技术,…

    2025年12月9日
    000
  • PHP 函数中堆栈溢出的常见触发因素有哪些

    php 函数中的堆栈溢出通常是由于以下常见触发因素造成的:递归函数未正确终止嵌套过多函数调用使用无限循环使用易于触发堆栈溢出的函数避免堆栈溢出需要明确设置递归函数的终止条件,控制函数嵌套层次,慎用无限循环,并谨慎使用可能生成大量数据的函数。 PHP 函数中堆栈溢出:常见触发因素及实战案例 堆栈溢出是…

    2025年12月9日
    000
  • PHP 函数的函数指针如何与用户自定义函数配合使用?

    PHP 函数的函数指针如何与用户自定义函数配合使用 在 PHP 中,函数指针是一种指向函数的变量。它可以分配给变量,然后像函数一样调用。这可以让你创建动态代码库,根据运行时条件调用不同的函数。 函数指针的语法 函数指针的语法如下: 立即学习“PHP免费学习笔记(深入)”; $function_poi…

    2025年12月9日
    000

发表回复

登录后才能评论
关注微信