一文详解Laravel中的事件溯源

一文详解Laravel中的事件溯源

事件溯源是一个在过去几年 PHP 社区越来越流行的术语,但对很多开发人员来说仍然是个谜。这些问题总是如何以及为什么,当然这是可以理解的。本教程的目的是通过一个实际的方式不仅仅是帮你理解什么是事件溯源,同样也会让你知道什么时候你可能会想使用它。

在传统应用程序中,我们的应用程序状态直接表示在我们所连接的数据库中。我们不能够完全理解它是怎么到那里的。我们只知道他是这样的。我们可以通过一些方法来进一步理解一点,使用审计模型更改的工具,这样我们就可以看到更改了什么,是谁做的。 这也是朝着正确方向迈出的一步。然而,我们仍然不明白这个关键的问题。【相关推荐:laravel视频教程】

为什么?为什么这个模型发生了变化?这改变的目的是什么?

这就是事件溯源发挥作用的地方,它保留了应用程序状态发生了什么以及为什么发生变化的历史视图。事件溯源允许你根据过去做出决定,从而你能够生成报告。但是在基本层面,他能让你知道为什么这个应用的状态改变了,这是通过事件完成的。

我将会构建一个基础的 Laravel 项目来引导你理解它是如何工作的。我们会将这个应用构建的简单些,以便你能理解时间溯源的逻辑而不是对于应用程序逻辑困惑。我们正在建立一个可以庆祝团队成员的应用程序。这就对了。简单而易于理解。我们与用户有团队,并且我们希望能够在团队中公开庆祝一些东西。

我们将新建一个 Laravel 项目,但我将使用 Jetstream,因为我想启用身份验证和团队结构和功能。 一旦你创建了这个项目, 请在你的IDE中打开它。(这里的正确答案当然是 PHPStorm ), 现在我们已经准备好在 Laravel 深入事件溯源。

我们希望为应用程序创建一个附加模型,这是唯一的一个。这是一个 Celebration 模型,你可以使用以下的Artisan命令创建它:

php artisan make:model Celebration -m

修改你的迁移文件 up 方法,它看起来应该像这样:

public function up(): void{    Schema::create('celebrations', static function (Blueprint $table): void {        $table->id();        $table->string('reason');        $table->text('message')->nullable();        $table            ->foreignId('user_id')            ->index()            ->constrained()            ->cascadeOnDelete();        $table            ->foreignId('sender_id')            ->index()            ->constrained('users')            ->cascadeOnDelete();        $table            ->foreignId('team_id')            ->index()            ->constrained()            ->cascadeOnDelete();        $table->timestamps();    });}

我们有一个庆祝的原因 reason,一个简单的句子,然后是我们可能希望与庆祝活动发送的可选消 message。除此之外,我们有三个关系,正在庆祝的用户,发送庆祝的用户,以及他们所在的团队。使用 Jetstream,一个用户可以属于多个团队,并且可能存在两个用户在同一个团队中的情况
,我们要确保我们在正确的团队中公开庆祝他们。

一旦我们有了这个设定,让我们看看模型本身:

declare(strict_types=1);namespace AppModels;use IlluminateDatabaseEloquentFactoriesHasFactory;use IlluminateDatabaseEloquentModel;use IlluminateDatabaseEloquentRelationsBelongsTo;use IlluminateDatabaseEloquentRelationsHasMany;final class Celebration extends Model{    use HasFactory;    protected $fillable = [        'reason',        'message',        'user_id',        'sender_id',        'team_id',    ];    public function user(): BelongsTo    {        return $this->belongsTo(            related: User::class,            foreignKey: 'user_id',        );    }    public function sender(): BelongsTo    {        return $this->belongsTo(            related: User::class,            foreignKey: 'sender_id',        );    }    public function team(): BelongsTo    {        return $this->belongsTo(            related: Team::class,            foreignKey: 'team_id',        );    }}

我们可以将这些关系映射到其他相关的模型上。尽管,默认情况下,我会将关联关系的另一端添加到每一个模型中,使它们的关联关系更加清楚,无论它是否严格需要这些关联关系。这是我养成的习惯,为了帮助他人理解模型本身。

现在我们有了从模型视角创建的我们的应用基础。我想我们需要一些安装一些对我们有帮助的软件包(依赖)。对于我的应用,我使用 Laravel Livewire 来控制 UI 。但是我并不会在本教程详细介绍这个包,因为我想确保我能专注于讲事件溯源这个方面。

与我创建的大多数项目一样,无论大小,我都为应用程序采用了模块化布局——一个领域驱动模型设计 ( Domain Driven Design ) 方法。这只是我做的事情,不要觉得你自己必须遵循这个,因为它是非常主观的。

我的下一步是设置我的域,对于这个演示,我只有一个域:文化。在文化中,我为我可能需要的一切创建了名称空间。但我会经历它,这样你就明白了过程

第一步是安装一个软件包,使我能够在Laravel中使用事件源。为此,我使用了一个 Spatie package包,它为我做了大量的后台工作

composer require spatie/laravel-event-sourcing

安装后,请确保按照包的安装说明进行操作,因为配置和迁移需要发布。正确安装后,运行迁移,使数据库处于正确状态。

php artisan migrate

现在我们可以开始思考我们要如何实现事件溯源。你可以通过两种方式实现这一点:投影仪来投影你的状态或聚合。

Projector 是一个位于你的应用程序中并处理你调度的事件的类。然后,这些将更改你的应用程序的状态。这不仅仅是简单地更新你的数据库。它位于中间,捕获一个事件,存储它,然后进行所需的更改 —— 然后 “投射” 应用程序的新状态

另一种方法,我的首选方法,聚合 – 这些是像投影仪一样为你处理应用程序状态的类。我们不是在我们的应用程序中自己触发事件,而是将其留给聚合为我们做。把它想象成一个中继,你要求中继做某事,它会为你处理。

在我们创建第一个聚合之前,需要在后台做一些工作。我非常喜欢为每个聚合创建一个事件存储,以便查询更快,并且该存储不会很快填满。这在包文档中进行了解释,但我将亲自引导你完成它,因为它在文档中并不是最清楚的。

第一步是创建模型和迁移,因为你将来需要一种方法来查询它以进行报告等。运行以下 artisan 命令来创建这些:

php artisan make:model CelebrationStoredEvent -m

以下代码是你在 up 方法中进行迁移所需的代码:

public function up(): void{    Schema::create('celebration_stored_events', static function (Blueprint $table): void {        $table->id();        $table->uuid('aggregate_uuid')->nullable()->unique();        $table        ->unsignedBigInteger('aggregate_version')        ->nullable()        ->unique();        $table->integer('event_version')->default(1);        $table->string('event_class');        $table->json('event_properties');        $table->json('meta_data');        $table->timestamp('created_at');        $table->index('event_class');        $table->index('aggregate_uuid');    });}

如你所见,我们为我们的活动收集了大量数据。现在模型要简单得多。它应该如下所示:

declare(strict_types=1);namespace AppModels;use SpatieEventSourcingStoredEventsModelsEloquentStoredEvent;final class CelebrationStoredEvent extends EloquentStoredEvent{    public $table = 'celebration_stored_events';}

当我们扩展 EloquentStoredEvent 模型时,我们需要做的就是改变它正在查看的表。模型的其余功能已经在父级上到位。

要使用这些模型,你必须创建一个存储库来查询事件。这是一个非常简单的存储库 —— 然而,这是一个重要的步骤。我将我的添加到我的域代码中,位于 src/Domains/Culture/Repositories/ 下,但您可以随意添加对您最有意义的位置:

declare(strict_types=1);namespace DomainsCultureRepositories;use AppModelsCelebrationStoredEvent;use SpatieEventSourcingStoredEventsRepositoriesEloquentStoredEventRepository;final class CelebrationStoredEventsRepository extends EloquentStoredEventRepository{    public function __construct(        protected string $storedEventModel = CelebrationStoredEvent::class,    ) {        parent::__construct();    }}

既然我们有了存储事件和查询它们的方法,我们可以继续我们的聚合本身。同样,我将我的存储在我的域中,但可以随意将你的存储在你的应用程序上下文中。

declare(strict_types=1);namespace DomainsCultureAggregates;use DomainsCultureRepositoriesCelebrationStoredEventsRepository;use SpatieEventSourcingAggregateRootsAggregateRoot;use SpatieEventSourcingStoredEventsRepositoriesStoredEventRepository;final class CelebrationAggregateRoot extends AggregateRoot{    protected function getStoredEventRepository(): StoredEventRepository    {        return app()->make(            abstract: CelebrationStoredEventsRepository::class,        );    }}

到目前为止,除了为我们连接到正确的事件存储之外,此聚合不会执行任何操作。要让它开始跟踪事件,我们首先需要创建它们。但在此之前,我们需要停下来想一想。我们希望在活动中存储哪些数据?我们想要存储我们需要的每一个属性吗?或者我们是否希望存储一个数组,就像它来自一个表单一样?我两种方法都不用,因为为什么要保持简单呢?我在所有事件中使用数据传输对象,以确保始终维护上下文并始终提供类型安全。

巧文书 巧文书

巧文书是一款AI写标书、AI写方案的产品。通过自研的先进AI大模型,精准解析招标文件,智能生成投标内容。

巧文书 61 查看详情 巧文书

我构建了一个软件包,让我做这件事更容易。可以通过以下 Composer 命令安装它:

composer require juststeveking/laravel-data-object-tools

和以前一样, 我默认将我的数据对象保存在我的领域, 但你可以添加到对你最有意义的地方。 我创建了一个名为 Celebration 的数据对象,可以传递给事件和聚合器:

declare(strict_types=1);namespace DomainsCultureDataObjects;use JustSteveKingDataObjectsContractsDataObjectContract;final class Celebration implements DataObjectContract{    public function __construct(        private readonly string $reason,        private readonly string $message,        private readonly int $user,        private readonly int $sender,        private readonly int $team,    ) {}    public function userID(): int    {        return $this->user;    }    public function senderID(): int    {        return $this->sender;    }    public function teamUD(): int    {        return $this->team;    }    public function toArray(): array    {        return [            'reason' => $this->reason,            'message' => $this->message,            'user_id' => $this->user,            'sender_id' => $this->sender,            'team_id' => $this->team,        ];    }}

当我升级到 PHP 8.2 时,这会容易得多,因为我可以创建只读类 – 是的,我的包已经支持它们。

现在我们有了我们的数据对象。我们可以回到我们想要存储的事件。我已经调用了我的CelebrationWasCreated,因为事件名称应该总是过去时。让我们看看这个事件:

declare(strict_types=1);namespace DomainsCultureEvents;use DomainsCultureDataObjectsCelebration;use SpatieEventSourcingStoredEventsShouldBeStored;final class CelebrationWasCreated extends ShouldBeStored{    public function __construct(        public readonly Celebration $celebration,    ) {}}

因为我们使用的是数据对象,所以我们的类保持干净。所以,现在我们有了一个事件——以及一个可以发送的数据对象,我们需要考虑如何触发它。这让我们回到了聚合本身,所以让我们在聚合上创建一个可以用于此目的的方法:

declare(strict_types=1);namespace DomainsCultureAggregates;use DomainsCultureDataObjectsCelebration;use DomainsCultureEventsCelebrationWasCreated;use DomainsCultureRepositoriesCelebrationStoredEventsRepository;use SpatieEventSourcingAggregateRootsAggregateRoot;use SpatieEventSourcingStoredEventsRepositoriesStoredEventRepository;final class CelebrationAggregateRoot extends AggregateRoot{    protected function getStoredEventRepository(): StoredEventRepository    {        return app()->make(            abstract: CelebrationStoredEventsRepository::class,        );    }    public function createCelebration(Celebration $celebration): CelebrationAggregateRoot    {        $this->recordThat(            domainEvent: new CelebrationWasCreated(                celebration: $celebration,            ),        );        return $this;    }}

在这一点上,我们有一种方法来要求一个类记录事件。但是,这一事件还不会持续下去 —— 那是以后的事。此外,我们不会以任何方式改变应用程序的状态。那么,我们该如何做这项活动采购工作呢?这一部分是关于 Livewire 中的实现的,我现在将向你介绍它。

我喜欢通过调度一个事件来管理这个过程,因为它更高效。如果你考虑如何与应用程序交互,你可以从 Web 访问它,通过 API 端点发送请求,或者发生 CLI 命令可能运行的事件 —— 可能是一个 Cron 作业。在所有这些方法中,通常,你需要即时响应,或者至少您不想等待。我将在我的 Livewire 组件上向你展示我为此使用的方法:

public function celebrate(): void{    $this->validate();    dispatch(new TeamMemberCelebration(        celebration: Hydrator::fill(            class: Celebration::class,            properties: [                'reason' => $this->reason,                'message' => $this->content,                'user' => $this->identifier,                'sender' => auth()->id(),                'team' => auth()->user()->current_team_id,            ]        ),    ));    $this->closeModal();}

在这一点上,我们有一种方法来要求一个类记录事件。但是,这一事件还不会持续下去 —— 那是以后的事。此外,我们不会以任何方式改变应用程序的状态。那么,我们该如何做这项活动采购工作呢?这一部分是关于 Livewire 中的实现的,我现在将向你介绍它。

我喜欢通过调度一个事件来管理这个过程,因为它更高效。如果你考虑如何与应用程序交互,你可以从 Web 访问它,通过 API 端点发送请求,或者发生 CLI 命令可能运行的事件 —— 可能是一个 Cron 作业。在所有这些方法中,通常,你需要即时响应,或者至少你不想等待。我将在我的 Livewire 组件上向你展示我为此使用的方法:

public function celebrate(): void{    $this->validate();    dispatch(new TeamMemberCelebration(        celebration: Hydrator::fill(            class: Celebration::class,            properties: [                'reason' => $this->reason,                'message' => $this->content,                'user' => $this->identifier,                'sender' => auth()->id(),                'team' => auth()->user()->current_team_id,            ]        ),    ));    $this->closeModal();}

当我验证来自组件的用户输入,可以分派处理一个新的作业,然后结束这个流程。我使用我的包将一个新的数据对象传递给作业。它有一个 Facade,可以让我用一系列属性来为类添加——到目前为止它工作得很好。那么这是怎么实现的呢?让我们来看看。

declare(strict_types=1);namespace AppJobsTeam;use DomainsCultureAggregatesCelebrationAggregateRoot;use DomainsCultureDataObjectsCelebration;use IlluminateBusQueueable;use IlluminateContractsQueueShouldQueue;use IlluminateFoundationBusDispatchable;use IlluminateQueueInteractsWithQueue;use IlluminateQueueSerializesModels;use IlluminateSupportStr;final class TeamMemberCelebration implements ShouldQueue{    use Queueable;    use Dispatchable;    use SerializesModels;    use InteractsWithQueue;    public function __construct(        public readonly Celebration $celebration,    ) {}    public function handle(): void    {        CelebrationAggregateRoot::retrieve(            uuid: Str::uuid()->toString(),        )->createCelebration(            celebration: $this->celebration,        )->persist();    }}

我们的工作将数据对象接受到它的构造函数中,然后在处理它时存储它。处理作业时,它使用 CelebrationAggregateRoot 按 UUID 检索聚合,然后调用我们之前创建的 createCelebration 方法。在它调用了这个方法之后 – 它在聚合本身上调用了 persist。这就是将为我们存储事件的内容。但是,同样,我们还没有改变我们的应用程序状态。我们所做的只是存储一个不相关的事件而不是创建我们想要创建的庆祝活动?那么我们缺少什么?

我们的事件也需要处理。在另一种方法中,我们使用投影仪来处理我们的事件,但我们必须手动调用它们。这是一个类似的过程,但是我们的聚合正在触发事件,我们仍然需要一个投影仪来处理事件并改变我们的应用程序状态。

让我们创建我们的投影仪,我称之为处理程序 —— 因为它们处理事件。但我会让你决定如何命名你的。

declare(strict_types=1);namespace DomainsCultureHandlers;use DomainsCultureEventsCelebrationWasCreated;use SpatieEventSourcingEventHandlersProjectorsProjector;use InfrastructureCultureActionsCreateNewCelebrationContract;final class CelebrationHandler extends Projector{    public function __construct(        public readonly CreateNewCelebrationContract $action,    ) {}    public function onCelebrationWasCreated(CelebrationWasCreated $event): void    {        $this->action->handle(            celebration: $event->celebration,        );    }}

我们的投影机 / 处理程序,无论你选择如何称呼它,都将从容器中为我们解析 – 然后它将寻找一个以 on 为前缀的方法,后跟事件名称本身。所以在我们的例子中,onCelebrationWasCreated。在我的示例中,我使用一个动作来执行事件中的实际逻辑 – 单个类执行一项可以轻松伪造或替换的工作。所以再一次,我们把树追到下一个班级。动作,这对我来说是这样的:

declare(strict_types=1);namespace DomainsCultureActions;use AppModelsCelebration;use DomainsCultureDataObjectsCelebration as CelebrationObject;use IlluminateDatabaseEloquentModel;use InfrastructureCultureActionsCreateNewCelebrationContract;final class CreateNewCelebration implements CreateNewCelebrationContract{    public function handle(CelebrationObject $celebration): Model|Celebration    {        return Celebration::query()->create(            attributes: $celebration->toArray(),        );    }}

这是当前执行的操作。如你所见,我的操作类本身实现了一个合同 / 接口。这意味着我将接口绑定到我的服务提供者中的特定实现。这使我可以轻松地创建测试替身 / 模拟 / 替代方法,而不会对需要执行的实际操作产生连锁反应。这不是严格意义上的事件溯源,而是通用编程。我们确实拥有的一个好处是我们的投影仪可以重放。因此,如果出于某种原因,我们离开了 Laravel Eloquent,也许我们使用了其他东西,我们可以创建一个新的操作 – 将实现绑定到我们的容器中,重放我们的事件,它应该都能正常工作。

在这个阶段,我们正在存储我们的事件并有办法改变我们的应用程序的状态 —— 但是我们做到了吗?我们需要告诉 Event Sourcing 库我们已经注册了这个 Projector/Handler 以便它知道在事件上触发它。通常我会为每个域创建一个 EventSourcingServiceProvider,这样我就可以在一个地方注册所有的处理程序。我的看起来如下:

declare(strict_types=1);namespace DomainsCultureProviders;use DomainsCultureHandlersCelebrationHandler;use IlluminateSupportServiceProvider;use SpatieEventSourcingFacadesProjectionist;final class EventSourcingServiceProvider extends ServiceProvider{    public function register(): void    {        Projectionist::addProjector(            projector: CelebrationHandler::class,        );    }}

剩下的就是确保再次注册此服务提供者。我为每个域创建一个服务提供者来注册子服务提供者 —— 但这是另一个故事和教程。

在这个阶段,我们正在存储我们的事件,并有一种办法改变我们的应用程序的状态——但是我们做到了吗?我们需要告诉 Event Sourcing 库,我们已经注册了 Projector/Handler 以便它知道在事件上触发它。通常,我会为每个域创建一个EventSourcingServiceProvider,以便可以在一个位置注册所有处理程序。如下:

declare(strict_types=1);namespace DomainsCultureProviders;use DomainsCultureHandlersCelebrationHandler;use IlluminateSupportServiceProvider;use SpatieEventSourcingFacadesProjectionist;final class EventSourcingServiceProvider extends ServiceProvider{    public function register(): void    {        Projectionist::addProjector(            projector: CelebrationHandler::class,        );    }}

剩下确保此服务提供者重新注册。我为每个域创建一个 Service Provider 来注册子服务提供者–但这是另一个故事和教程。

现在,当我们把它们放在一起时。我们可以要求我们的聚合创建一个庆祝活动,它将记录事件并将其保存在数据库中,并且作为副作用,我们的处理程序将被触发,随着新的变化改变应用程序的状态。

这似乎有点啰嗦,对吧?有没有更好的办法?可能,但在这一点上,我们知道何时更改了我们的应用程序状态。我们了解它们的制作原因。此外,由于我们的数据对象,我们知道谁进行了更改以及何时进行了更改。所以它可能不是最直接的方法,但它可以让我们更多地了解我们的应用程序。

你可以根据需要尽可能多地进行此操作,也可以将脚趾浸入事件溯源中,这是最有意义的。希望本教程为你展示了一条清晰实用的路径,让你从今天开始使用事件溯源。

如果看完后你觉得意犹未尽, Spatie 很大方的提供了一张 7 折优惠券,可以用在他们 Laravel 课程里的事件溯源部分,真是太棒了!访问 课程网站 并使用优惠券码 LARAVEL-NEWS-EVENT-SOURCING

你曾经使用过事件溯源吗?你是怎么处理的?在评论区告诉我们!

原文地址:https://laravel-news.com/event-sourcing-in-laravel译文地址:https://learnku.com/laravel/t/71001

更多编程相关知识,请访问:编程视频!!

以上就是一文详解Laravel中的事件溯源的详细内容,更多请关注php中文网其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
怎么查看小米10s电池状况_查看小米10s电池状况教程
上一篇 2025年11月5日 17:36:03
mysql if then 出错
下一篇 2025年11月5日 17:36:08

相关推荐

  • 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
  • 怎么在PHP代码中实现图片上传功能_PHP图片上传功能实现与安全处理教程

    首先创建含enctype的HTML表单,再用PHP接收文件,检查目录、移动临时文件,验证类型与大小,生成唯一文件名,并调整php.ini限制以确保上传成功。 如果您尝试在PHP项目中添加图片上传功能,但服务器无法正确接收或保存文件,则可能是由于表单配置、文件处理逻辑或安全限制的问题。以下是实现该功能…

    2026年5月10日
    100
  • 获取日期中的周数:CodeIgniter 教程

    本教程旨在帮助开发者在 CodeIgniter 框架中,从日期字符串中准确提取周数。我们将使用 PHP 内置的 DateTime 类,并提供详细的代码示例和注意事项,确保您能够轻松地在项目中实现此功能。 使用 DateTime 类获取周数 PHP 的 DateTime 类提供了一种便捷的方式来处理日…

    2026年5月10日
    000
  • php常量怎么用_PHP常量(define/const)定义与使用方法

    PHP中可通过define函数和const关键字定义常量,用于存储不可变值。define适用于全局作用域,支持动态名称和条件定义,如define(‘SITE_NAME’, ‘MyWebsite’);const在编译时生效,语法简洁但限制多,只能在类或全…

    2026年5月10日
    000
  • PHP动态生成表单输入与POST数据获取实践指南

    本教程详细阐述了如何在php中根据动态数据源(如数据库值)生成多个表单输入框,并演示了如何通过post方法准确无误地获取这些动态生成的输入值。文章强调了正确的输入框命名策略,避免了常见的命名误区,并提供了完整的代码示例,确保开发者能够高效处理动态表单数据。 动态生成表单输入 在Web开发中,我们经常…

    2026年5月10日
    000
  • JavaScript函数中插入加载动画(Spinner)的正确方法

    本文旨在解决在JavaScript函数中插入加载动画(Spinner)时遇到的异步问题。通过引入async/await和Promise.all,确保在数据处理完成前后正确显示和隐藏加载动画,提升用户体验。我们将提供两种实现方案,并详细解释其原理和优势。 在Web开发中,当执行耗时操作时,显示加载动画…

    2026年5月10日
    000
  • PHP多维数组到复杂XML结构的SOAP序列化实践

    本文旨在解决php多维数组向复杂soap xml结构序列化时遇到的“无法序列化结果”问题。通过深入理解soap xml的结构要求,包括命名空间和类型属性,文章将指导您如何构建符合特定xml schema的php关联数组。我们将利用`spatie/array-to-xml`库,详细演示其安装与使用方法…

    2026年5月10日
    000
  • 使用 Ajax 和 FormData 实现文件上传及文本数据提交的完整教程

    本文旨在解决在使用 Ajax 和 FormData 进行文件上传时,遇到的 $_POST 和 $_FILES 为空的问题。通过详细的代码示例和解释,我们将展示如何正确地构建 FormData 对象,并通过 Ajax 将文件和文本数据发送到服务器端,同时避免常见的错误配置,确保数据能够成功地被 PHP…

    2026年5月10日
    000
  • 虫虫漫画直接进入官网入口_虫虫漫画网页版清爽版

    虫虫漫画直接进入官网入口_虫虫漫画网页版清爽版虫虫漫画直接进入官网入口_虫虫漫画网页版清爽版虫虫漫画直接进入官网入口_虫虫漫画网页版清爽版虫虫漫画直接进入官网入口_虫虫漫画网页版清爽版

    虫虫漫画官网入口为www.ccmh.com,用户可直接通过浏览器访问,支持多端适配与账号同步功能,界面简洁无广告,提供海量国漫、日漫、韩漫资源,涵盖恋爱、玄幻等热门题材,更新及时,支持多种阅读模式及离线缓存,阅读体验流畅。 虫虫漫画直接进入官网入口在哪里?这是不少网友都关注的,接下来由PHP小编为大…

    2026年5月10日 用户投稿
    100
  • 从 JavaScript 获取 URL 并在 PHP DataGrid 中使用

    本文档旨在指导开发者如何从 JavaScript 函数中获取 URL,并将其动态应用于 PHP DataGrid。通过前端 JavaScript 动态生成 API 地址,并将其传递给后端的 PHP DataGrid,实现数据根据用户会话动态加载。 动态配置 DataGrid 的 URL 在构建动态 …

    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
  • PHP安全文件下载:防止直链与保护资源

    本文旨在解决通过检查元素获取直链下载文件的问题,并提供一种安全的PHP服务器端文件交付方案。核心思想是利用PHP作为文件代理,通过设置HTTP响应头直接将文件发送给用户,从而隐藏文件的实际存储路径,有效防止未经授权的直接链接访问。 客户端下载链接的风险与局限性 在构建下载页面时,开发者常常面临一个挑…

    2026年5月10日
    100
  • php超过字数怎么解密_用PHP分段处理超字数加密数据并解密教程【技巧】

    分段解密超长加密数据需先确定算法限制,再通过OpenSSL扩展支持,编写函数逐段解密并拼接结果。1、明确加密算法与密钥对应的分段大小;2、启用php.ini中openssl扩展并重启服务;3、自定义函数读取私钥、base64解码密文、循环截取块解密;4、确保去除密文换行符并按原加密块大小切分;5、解…

    2026年5月10日
    000
  • Python中如何实现过滤器模式?

    在Python中实现过滤器模式的过程中,我们可以利用Python的灵活性来创建一个既简单又强大的过滤系统。让我们从回答这个问题开始:Python中如何实现过滤器模式? 在Python中,过滤器模式可以通过定义一系列的过滤器类来实现,这些类能够根据特定条件对对象进行过滤。Python的函数式编程特性,…

    2026年5月10日
    100
  • php代码如何操作JSON数据_php代码解析和生成JSON的方法

    答案:PHP中处理JSON需使用json_encode()和json_decode()函数。1、将数组转为JSON字符串时,用json_encode()并检查返回值是否为false;2、解析JSON字符串时,调用json_decode()并设第二参数为true返回数组,false则返回对象;3、处理…

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

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

    2026年5月10日
    000
  • PHP代码注入检测日志分析_PHP代码注入日志检测方法详解

    答案:日志分析是发现PHP代码注入的关键手段,主要通过Web服务器访问日志、PHP错误日志、PHP-FPM日志及应用自定义日志等多源数据,结合grep、ELK、WAF等工具识别含eval()、system()、Base64编码、目录遍历等特征的异常请求,并建立基线、设置检测规则与自动化告警,配合事件…

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

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

    2026年5月10日
    000
  • PHP中通过键名高效关联与输出多维数组数据

    本教程旨在解决php开发中常见的数据关联与输出问题,特别是当需要将不同数组中通过共同键名关联的数据进行整合展示时。文章将详细阐述如何利用foreach循环的键值对特性,结合array_key_exists函数,实现从多个数组中提取并组合相关信息,从而避免不必要的嵌套循环,提升代码的清晰度和执行效率。…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信