This commit is contained in:
2026-06-24 20:45:37 +08:00
parent 69804ea595
commit b5e0026816
+61 -3
View File
@@ -215,11 +215,12 @@ class Router
$changedFiles = $container->get(HotReloadState::class)->consume(); $changedFiles = $container->get(HotReloadState::class)->consume();
// Worker 首次启动(无变更文件 + Master 已完成扫描): // Worker 首次启动(无变更文件 + Master 已完成扫描):
// RouterCollector 已由 Master 填充(含路由文件路由 + 注解路由),Worker 通过 fork 继承 // 重新 include 路由文件(Router::get/post 显式注册) + 基于 Master 扫描清单重建注解路由
// 重新 include 路由文件以更新 Router::get/post 等显式注册的路由(同 key 覆盖不产生重复) // 避免 opcache_compile_file,仅用 Reflection 重建路由,内存开销极小
// 注解路由(#[Route]/#[Get]/#[Post]等)由 Master 注入保留,无需重新扫描 app 目录
if (empty($changedFiles) && self::$initialScanDone) { if (empty($changedFiles) && self::$initialScanDone) {
$container->get(DataGrip::class)->reset(static::$type);
$this->read_dir_file(APP_PATH . 'routes'); $this->read_dir_file(APP_PATH . 'routes');
$this->rebuildAnnotationRoutes($container);
$this->reset($container); $this->reset($container);
$coordinator->done(); $coordinator->done();
return; return;
@@ -294,6 +295,63 @@ class Router
} }
/**
* 基于 Master 扫描清单重建注解路由(轻量级,无文件 I/O)
* 遍历 Scanner manifest 中的所有类,用 Reflection 重新发现 #[Route]/#[Get] 等注解
* 避免 Worker 重复执行 opcache_compile_file,但确保注解路由不丢失
*
* @param ContainerInterface $container
* @return void
*/
private function rebuildAnnotationRoutes(ContainerInterface $container): void
{
$scanner = $container->get(Kiri\Di\Scanner::class);
$scanConfig = array_merge(
config('servers.reload.scan', []),
config('site.scanner', [])
);
$scanner->setConfig($scanConfig);
// 获取 Master 扫描产生的完整清单,遍历所有已注册类重建注解路由
$manifestEntries = $scanner->getManifestClasses();
foreach ($manifestEntries as $path => $entry) {
$classes = is_array($entry) && isset($entry['classes']) ? $entry['classes'] : [];
foreach ($classes as $class) {
if (!class_exists($class)) {
continue;
}
try {
$reflect = $container->getReflectionClass($class);
if (!$reflect->isInstantiable() || $reflect->isTrait() || $reflect->isEnum() || $reflect->isInterface()) {
continue;
}
foreach ($reflect->getMethods() as $method) {
if ($method->isStatic() || $method->getDeclaringClass()->getName() !== $class) {
continue;
}
foreach ($method->getAttributes() as $attribute) {
$attrName = $attribute->getName();
if (!class_exists($attrName)) {
continue;
}
try {
$instance = $attribute->newInstance();
if ($instance instanceof Kiri\Di\Interface\InjectMethodInterface) {
$instance->dispatch($class, $method->getName());
}
} catch (\Throwable $e) {
\Kiri::getLogger()->error("AnnotationRoute rebuild [{$class}::{$method->getName()}]: " . $e->getMessage());
}
}
}
} catch (\Throwable $e) {
\Kiri::getLogger()->error("AnnotationRoute class [{$class}]: " . $e->getMessage());
}
}
}
}
/** /**
* @param $path * @param $path
* *