一文带你了解Laravel schedule调度的运行机制

本篇文章带大家聊聊laravel 中 schedule 调度的运行机制,希望对大家有所帮助!

一文带你了解Laravel schedule调度的运行机制

  Laravel 的 console 命令行极大的方便了 PHP 定时任务的设置以及运行。以往通过 crontab 配置定时任务过程相对比较繁琐,并且通过 crontab 设置的定时任务很难防止任务的交叠运行。

  所谓任务的交叠运行,是指由于定时任务运行时间较长,在 crontab 设置的运行周期不尽合理的情况下,已经启动的任务还没有结束运行,而系统又启动了新的任务去执行相同的操作。如果程序内部没有处理好数据一致性的问题,那么两个任务同时操作同一份数据,很可能会导致严重的后果。

runInBackgroundwithoutOverlapping

  为了防止任务的交叠运行,Laravel 提供了 withoutOverlapping() 方法;为了能让多任务在后台并行执行,Laravel 提供了 runInBackground() 方法。

runInBackground() 方法

  console 命令行中的每一个命令都代表一个 EventAppConsoleKernel 中的 schedule() 方法的作用只是将这些命令行代表的 Event 注册到 IlluminateConsoleSchedulingSchedule 的属性 $events 中。

// namespace IlluminateConsoleSchedulingSchedulepublic function command($command, array $parameters = []){    if (class_exists($command)) {        $command = Container::getInstance()->make($command)->getName();    }    return $this->exec(        Application::formatCommandString($command), $parameters    );}public function exec($command, array $parameters = []){    if (count($parameters)) {        $command .= ' '.$this->compileParameters($parameters);    }    $this->events[] = $event = new Event($this->eventMutex, $command, $this->timezone);    return $event;}

  Event 的运行方式有两种:ForegroundBackground 。二者的区别就在于多个 Event 是否可以并行执行。Event 默认以 Foreground 的方式运行,在这种运行方式下,多个 Event 顺序执行,后面的 Event 需要等到前面的 Event 运行完成之后才能开始执行。

  但在实际应用中,我们往往是希望多个 Event 可以并行执行,此时就需要调用 EventrunInBackground() 方法将其运行方式设置为 Background

  Laravel 框架对这两种运行方式的处理区别在于命令行的组装方式和回调方法的调用方式。

// namespace IlluminateConsoleSchedulingEventprotected function runCommandInForeground(Container $container){    $this->callBeforeCallbacks($container);    $this->exitCode = Process::fromShellCommandline($this->buildCommand(), base_path(), null, null, null)->run();    $this->callAfterCallbacks($container);}protected function runCommandInBackground(Container $container){    $this->callBeforeCallbacks($container);    Process::fromShellCommandline($this->buildCommand(), base_path(), null, null, null)->run();}public function buildCommand(){    return (new CommandBuilder)->buildCommand($this);}// namespace IlluminateConsoleSchedulingCommandBuilderpublic function buildCommand(Event $event){    if ($event->runInBackground) {        return $this->buildBackgroundCommand($event);    }    return $this->buildForegroundCommand($event);}protected function buildForegroundCommand(Event $event){    $output = ProcessUtils::escapeArgument($event->output);    return $this->ensureCorrectUser(        $event, $event->command.($event->shouldAppendOutput ? ' >> ' : ' > ').$output.' 2>&1'    );}protected function buildBackgroundCommand(Event $event){    $output = ProcessUtils::escapeArgument($event->output);    $redirect = $event->shouldAppendOutput ? ' >> ' : ' > ';    $finished = Application::formatCommandString('schedule:finish').' "'.$event->mutexName().'"';    if (windows_os()) {        return 'start /b cmd /c "('.$event->command.' & '.$finished.' "%errorlevel%")'.$redirect.$output.' 2>&1"';    }    return $this->ensureCorrectUser($event,        '('.$event->command.$redirect.$output.' 2>&1 ; '.$finished.' "$?") > '        .ProcessUtils::escapeArgument($event->getDefaultOutput()).' 2>&1 &'    );}

  从代码中可以看出,采用 Background 方式运行的 Event ,其命令行在组装的时候结尾会增加一个 & 符号,其作用是使命令行程序进入后台运行;另外,采用 Foreground 方式运行的 Event ,其回调方法是同步调用的,而采用 Background 方式运行的 Event ,其 after 回调则是通过 schedule:finish 命令行来执行的。

withoutOverlapping() 方法

  在设置 Event 的运行周期时,由于应用场景的不断变化,很难避免某个特定的 Event 在某个时间段内需要运行较长的时间才能完成,甚至在下一个运行周期开始时还没有执行完成。如果不对这种情况进行处理,就会导致多个相同的 Event 同时运行,而如果这些 Event 当中涉及到对数据的操作并且程序中没有处理好幂等问题,很可能会造成严重后果。

  为了避免出现上述的问题,Event 中提供了 withoutOverlapping() 方法,该方法通过将 EventwithoutOverlapping 属性设置为 TRUE ,在每次要执行 Event 时会检查当前是否存在正在执行的相同的 Event ,如果存在,则不执行新的 Event 任务。

// namespace IlluminateConsoleSchedulingEventpublic function withoutOverlapping($expiresAt = 1440){    $this->withoutOverlapping = true;    $this->expiresAt = $expiresAt;    return $this->then(function () {        $this->mutex->forget($this);    })->skip(function () {        return $this->mutex->exists($this);    });}public function run(Container $container){    if ($this->withoutOverlapping &&        ! $this->mutex->create($this)) {        return;    }    $this->runInBackground                ? $this->runCommandInBackground($container)                : $this->runCommandInForeground($container);}

mutex 互斥锁

  在调用 withoutOverlapping() 方法时,该方法还实现了另外两个功能:一个是设置超时时间,默认为 24 小时;另一个是设置 Event 的回调。

⑴ 超时时间

  首先说超时时间,这个超时时间并不是 Event 的超时时间,而是 Event 的属性 mutex 的超时时间。在向 IlluminateConsoleSchedulingSchedule 的属性 $events 中注册 Event 时,会调用 Schedule 中的 exec() 方法,在该方法中会新建 Event 对象,此时会向 Event 的构造方法中传入一个 eventMutex ,这就是 Event 对象中的属性 mutex ,超时时间就是为这个 mutex 设置的。而 Schedule 中的 eventMutex 则是通过实例化 CacheEventMutex 来创建的。

// namespace IlluminateConsoleSchedulingSchedule$this->eventMutex = $container->bound(EventMutex::class)                                ? $container->make(EventMutex::class)                                : $container->make(CacheEventMutex::class);

  设置了 withoutOverlappingEvent 在执行之前,首先会尝试获取 mutex 互斥锁,如果无法成功获取到锁,那么 Event 就不会执行。获取互斥锁的操作通过调用 mutexcreate() 方法完成。

  CacheEventMutex 在实例化时需要传入一个 IlluminateContractsCacheFactory 类型的实例,其最终传入的是一个 IlluminateCacheCacheManager 实例。在调用 create() 方法获取互斥锁时,还需要通过调用 store() 方法设置存储引擎。

// namespace IlluminateFoundationConsoleKernelprotected function defineConsoleSchedule(){    $this->app->singleton(Schedule::class, function ($app) {        return tap(new Schedule($this->scheduleTimezone()), function ($schedule) {            $this->schedule($schedule->useCache($this->scheduleCache()));        });    });}protected function scheduleCache(){    return Env::get('SCHEDULE_CACHE_DRIVER');}// namespace IlluminateConsoleSchedulingSchedulepublic function useCache($store){    if ($this->eventMutex instanceof CacheEventMutex) {        $this->eventMutex->useStore($store);    }    /* ... ... */    return $this;}// namespace IlluminateConsoleSchedulingCacheEventMutexpublic function create(Event $event){    return $this->cache->store($this->store)->add(        $event->mutexName(), true, $event->expiresAt * 60    );}// namespace IlluminateCacheCacheManagerpublic function store($name = null){    $name = $name ?: $this->getDefaultDriver();    return $this->stores[$name] = $this->get($name);}public function getDefaultDriver(){    return $this->app['config']['cache.default'];}protected function get($name){    return $this->stores[$name] ?? $this->resolve($name);}protected function resolve($name){    $config = $this->getConfig($name);    if (is_null($config)) {        throw new InvalidArgumentException("Cache store [{$name}] is not defined.");    }    if (isset($this->customCreators[$config['driver']])) {        return $this->callCustomCreator($config);    } else {        $driverMethod = 'create'.ucfirst($config['driver']).'Driver';        if (method_exists($this, $driverMethod)) {            return $this->{$driverMethod}($config);        } else {            throw new InvalidArgumentException("Driver [{$config['driver']}] is not supported.");        }    }}protected function getConfig($name){    return $this->app['config']["cache.stores.{$name}"];}protected function createFileDriver(array $config){    return $this->repository(new FileStore($this->app['files'], $config['path'], $config['permission'] ?? null));}

  在初始化 Schedule 时会指定 eventMutex 的存储引擎,默认为环境变量中的配置项 SCHEDULE_CACHE_DRIVER 的值。但通常这一项配置在环境变量中并不存在,所以 useCache() 的参数值为空,进而 eventMutexstore 属性值也为空。这样,在 eventMutexcreate() 方法中调用 store() 方法为其设置存储引擎时,store() 方法的参数值也为空。

  当 store() 方法的传参为空时,会使用应用的默认存储引擎(如果不做任何修改,默认 cache 的存储引擎为 file)。之后会取得默认存储引擎的配置信息(引擎、存储路径、连接信息等),然后实例化存储引擎。最终,file 存储引擎实例化的是 IlluminateCacheFileStore

  在设置完存储引擎之后,紧接着会调用 add() 方法获取互斥锁。由于 store() 方法返回的是 IlluminateContractsCacheRepository 类型的实例,所以最终调用的是 IlluminateCacheRepository 中的 add() 方法。

// namespace IlluminateCacheRepositorypublic function add($key, $value, $ttl = null){    if ($ttl !== null) {        if ($this->getSeconds($ttl) store, 'add')) {            $seconds = $this->getSeconds($ttl);            return $this->store->add(                $this->itemKey($key), $value, $seconds            );        }    }    if (is_null($this->get($key))) {        return $this->put($key, $value, $ttl);    }    return false;}public function get($key, $default = null){    if (is_array($key)) {        return $this->many($key);    }    $value = $this->store->get($this->itemKey($key));    if (is_null($value)) {        $this->event(new CacheMissed($key));        $value = value($default);    } else {        $this->event(new CacheHit($key, $value));    }    return $value;}// namespace IlluminateCacheFileStorepublic function get($key){    return $this->getPayload($key)['data'] ?? null;}protected function getPayload($key){    $path = $this->path($key);    try {        $expire = substr(            $contents = $this->files->get($path, true), 0, 10        );    } catch (Exception $e) {        return $this->emptyPayload();    }    if ($this->currentTime() >= $expire) {        $this->forget($key);        return $this->emptyPayload();    }    try {        $data = unserialize(substr($contents, 10));    } catch (Exception $e) {        $this->forget($key);        return $this->emptyPayload();    }    $time = $expire - $this->currentTime();    return compact('data', 'time');}

  这里需要说明,所谓互斥锁,其本质是写文件。如果文件不存在或文件内容为空或文件中存储的过期时间小于当前时间,则互斥锁可以顺利获得;否则无法获取到互斥锁。文件内容为固定格式:timestampb:1

  所谓超时时间,与此处的 timestamp 的值有密切的联系。获取互斥锁时的时间戳,再加上超时时间的秒数,即是此处的 timestamp 的值。

  由于 FileStore 中不存在 add() 方法,所以程序会直接尝试调用 get() 方法获取文件中的内容。如果 get() 返回的结果为 NULL,说明获取互斥锁成功,之后会调用 FileStoreput() 方法写文件;否则,说明当前有相同的 Event 在运行,不会再运行新的 Event

  在调用 put() 方法写文件时,首先需要根据传参计算 eventMutex 的超时时间的秒数,之后再调用 FileStore 中的 put() 方法,将数据写入文件中。

// namespace IlluminateCacheRepositorypublic function put($key, $value, $ttl = null){    /* ... ... */    $seconds = $this->getSeconds($ttl);    if ($seconds forget($key);    }    $result = $this->store->put($this->itemKey($key), $value, $seconds);    if ($result) {        $this->event(new KeyWritten($key, $value, $seconds));    }    return $result;}// namespace IlluminateCacheFileStorepublic function put($key, $value, $seconds){    $this->ensureCacheDirectoryExists($path = $this->path($key));    $result = $this->files->put(        $path, $this->expiration($seconds).serialize($value), true    );    if ($result !== false && $result > 0) {        $this->ensureFileHasCorrectPermissions($path);        return true;    }    return false;}protected function path($key){    $parts = array_slice(str_split($hash = sha1($key), 2), 0, 2);    return $this->directory.'/'.implode('/', $parts).'/'.$hash;}// namespace IlluminateConsoleSchedulingSchedulepublic function mutexName(){    return 'framework'.DIRECTORY_SEPARATOR.'schedule-'.sha1($this->expression.$this->command);}

  这里需要重点说明的是 $key 的生成方法以及文件路径的生成方法。$key 通过调用 EventmutexName() 方法生成,其中需要用到 Event$expression$command 属性。其中 $command 为我们定义的命令行,在调用 $schedule->comand() 方法时传入,然后进行格式化,$expression 则为 Event 的运行周期。

  以命令行 schedule:test 为例,格式化之后的命令行为 `/usr/local/php/bin/php` `artisan` schedule:test,如果该命令行设置的运行周期为每分钟一次,即 * * * * * ,则最终计算得到的 $key 的值为 framework/schedule-768a42da74f005b3ac29ca0a88eb72d0ca2b84be 。文件路径则是将 $key 的值再次进行 sha1 计算之后,以两个字符为一组切分成数组,然后取数组的前两项组成一个二级目录,而配置文件中 file 引擎的默认存储路径为 storage/framework/cache/data ,所以最终的文件路径为 storage/framework/cache/data/eb/60/eb608bf555895f742e5bd57e186cbd97f9a6f432 。而文件中存储的内容则为 1642122685b:1

⑵ 回调方法

  再来说设置的 Event 回调,调用 withoutOverlapping() 方法会为 Event 设置两个回调:一个是 Event 运行完成之后的回调,用于释放互斥锁,即清理缓存文件;另一个是在运行 Event 之前判断互斥锁是否被占用,即缓存文件是否已经存在。

  无论 Event 是以 Foreground 的方式运行,还是以 Background 的方式运行,在运行完成之后都会调用 callAfterCallbacks() 方法执行 afterCallbacks 中的回调,其中就有一项回调用于释放互斥锁,删除缓存文件 $this->mutex->forget($this) 。区别就在于,以 Foreground 方式运行的 Event 是在运行完成之后显式的调用这些回调方法,而以 Background 方式运行的 Event 则需要借助 schedule:finish 来调用这些回调方法。

  所有在 AppConsoleKernel 中注册 Event,都是通过命令行 schedule:run 来调度的。在调度之前,首先会判断当前时间点是否满足各个 Event 所配置的运行周期的要求。如果满足的话,接下来就是一些过滤条件的判断,这其中就包括判断互斥锁是否被占用。只有在互斥锁没有被占用的情况下,Event 才可以运行。

// namespace IlluminateConsoleSchedulingScheduleRunCommandpublic function handle(Schedule $schedule, Dispatcher $dispatcher){    $this->schedule = $schedule;    $this->dispatcher = $dispatcher;    foreach ($this->schedule->dueEvents($this->laravel) as $event) {        if (! $event->filtersPass($this->laravel)) {            $this->dispatcher->dispatch(new ScheduledTaskSkipped($event));            continue;        }        if ($event->onOneServer) {            $this->runSingleServerEvent($event);        } else {            $this->runEvent($event);        }        $this->eventsRan = true;    }    if (! $this->eventsRan) {        $this->info('No scheduled commands are ready to run.');    }}// namespace IlluminateConsoleSchedulingSchedulepublic function dueEvents($app){    return collect($this->events)->filter->isDue($app);}// namespace IlluminateConsoleSchedulingEventpublic function isDue($app){    /* ... ... */    return $this->expressionPasses() &&           $this->runsInEnvironment($app->environment());}protected function expressionPasses(){    $date = Carbon::now();    /* ... ... */    return CronExpression::factory($this->expression)->isDue($date->toDateTimeString());}// namespace CronCronExpressionpublic function isDue($currentTime = 'now', $timeZone = null){   /* ... ... */       try {        return $this->getNextRunDate($currentTime, 0, true)->getTimestamp() === $currentTime->getTimestamp();    } catch (Exception $e) {        return false;    }}public function getNextRunDate($currentTime = 'now', $nth = 0, $allowCurrentDate = false, $timeZone = null){    return $this->getRunDate($currentTime, $nth, false, $allowCurrentDate, $timeZone);}

  有时候,我们可能需要 kill 掉一些在后台运行的命令行,但紧接着我们会发现这些被 kill 掉的命令行在一段时间内无法按照设置的运行周期自动调度,其原因就在于手动 kill 掉的命令行没有调用 schedule:finish 清理缓存文件,释放互斥锁。这就导致在设置的过期时间到达之前,互斥锁会一直被占用,新的 Event 不会再次运行。

【相关推荐:laravel视频教程】

以上就是一文带你了解Laravel schedule调度的运行机制的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
收集企业发展需求怎么写
上一篇 2025年11月13日 15:55:42
客户需求收集表怎么写的
下一篇 2025年11月13日 15:55:57

相关推荐

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

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

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

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

    2026年5月10日
    000
  • 深入理解 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
  • 优化 Laravel Eloquent 查询:高效构建用户排行榜数据

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

    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
  • PHP中批量为嵌套数组元素添加公共属性的教程

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

    2026年5月10日
    000
  • PHP框架的社区支持存在哪些痛点?

    php框架社区支持的痛点包括:文档匮乏或过时(1)、响应缓慢(2)、社区分散(3)。实战案例表明这些痛点可能导致开发进度受阻。改善方法包括:提供全面的文档、建立响应迅速的官方论坛、创建一个集成的社区平台。 PHP 框架社区支持存在的痛点及实战案例 PHP 框架为 Web 开发提供了强大的基础,但其社…

    2026年5月10日
    100
  • Laravel 8中Firebase Storage文件条件删除策略与实践

    本文针对Laravel 8环境下Firebase Storage无法直接按目录批量或条件删除文件的限制,提出了一套基于元数据管理的解决方案。通过在数据库中记录文件信息,结合Laravel的Artisan命令和Cron任务,实现对过期文件的精准识别与逐个删除,确保存储资源的有效管理。 Firebase…

    2026年5月10日
    000
  • php怎么安装_在云服务器上部署PHP环境的步骤

    答案:在云服务器上部署PHP环境需搭建LEMP栈(Linux+Nginx+MySQL+PHP-FPM),依次更新系统、安装Nginx、MariaDB、PHP-FPM及扩展,配置Nginx解析PHP并测试,最后通过权限控制、安全配置、防火墙和HTTPS等措施保障环境安全稳定。 在云服务器上部署PHP环…

    2026年5月10日
    000
  • Laravel 产品多图上传错误:foreach() 参数类型问题解决方案

    本文旨在解决 Laravel 应用中产品多图上传时遇到的 “foreach() argument must be of type array|object, null given” 错误。通过检查并确保循环遍历的变量为数组类型,避免因空值导致的错误,并提供代码示例和注意事项,…

    2026年5月10日
    200
  • PHP源码命令行工具开发_PHP源码命令行工具开发教程

    答案是使用PHP开发命令行工具需依托CLI SAPI,结合Composer管理依赖,并推荐采用Symfony Console等组件库来构建。首先确保PHP支持CLI模式,通过编写基础脚本并利用$argv和getopt()处理参数,但更优方式是引入Symfony Console组件进行命令定义与输入输…

    2026年5月10日
    000
  • PHP怎么运行创建_php脚本创建与执行流程解析

    PHP脚本需在服务器环境中通过解释器运行,不能双击执行。首先搭建环境(如XAMPP),然后编写.php文件并保存至服务器根目录,接着通过浏览器访问或命令行执行php命令运行脚本,服务器会调用PHP解释器解析代码并返回结果。 PHP脚本的运行依赖于服务器环境和解释器,不是直接像可执行程序那样双击运行。…

    2026年5月10日
    100
  • php中get_parent_class获取父类名_php在继承链中定位父类的应用场景

    get_parent_class函数用于获取类的父类名称,接收类名字符串返回父类名或false。示例中Dog类继承Animal,调用get_parent_class(__CLASS__)输出Animal。应用场景一:条件性调用父类方法,如构造函数中判断是否存在父类并调用其方法,提升灵活性。应用场景二…

    2026年5月10日
    100
  • 使用Laravel Blade动态渲染带标题的表格数据

    本文旨在详细指导如何在Laravel Blade模板中,利用`@foreach`循环和正确的索引策略,高效且准确地从嵌套数组结构中提取数据,并将其渲染成一个结构清晰、内容匹配的HTML表格,避免数据重复和错位问题。 在Web开发中,经常需要根据后端提供的数据动态生成HTML表格。特别是在处理具有行标…

    2026年5月10日
    000
  • Laravel模型中实现多语言数据自动过滤:重写newQuery()方法

    本教程详细介绍在laravel多语言应用中,如何通过重写模型(model)的`newquery()`方法,实现数据查询时自动根据当前应用语言环境进行过滤。这种方法提供了一种优雅且dry(don’t repeat yourself)的解决方案,避免了在每次数据查询时手动添加语言条件,确保了…

    2026年5月10日
    000
  • php学习有哪些

    PHP 学习途径:入门途径:在线教程:Codecademy、Udemy、Coursera 等书籍:《Head First PHP & MySQL》、《PHP in Action》官方文档:PHP 官方文档进阶学习:框架:Laravel、CodeIgniter 等数据库:MySQL、Postg…

    2026年5月10日
    100
  • 在 Laravel 中同时存储原始图片和 WebP 转换图片

    本文详细介绍了在 Laravel 应用中如何高效地处理图片上传,实现同时保存原始图片(如 JPG/PNG)及其 WebP 转换版本。通过利用 PHP 原生 GD 库功能,我们能够克服 Intervention Image 在特定场景下的路径写入问题,确保原始图片和优化后的 WebP 格式文件都能正确…

    2026年5月10日
    000
  • 解决AJAX响应中PHP输出JSON后出现多余HTML的问题

    本文旨在解决PHP脚本通过AJAX响应返回JSON数据时,出现JSON数据后方意外附带HTML内容的问题。通过在PHP脚本中JSON编码输出后立即使用die()或exit()函数,可以有效阻止后续不必要的输出,确保客户端接收到纯净、可解析的JSON响应,从而避免解析错误,提升前后端通信的健壮性。 理…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信