diff --git a/core/lib/Drupal/Core/Controller/ControllerBase.php b/core/lib/Drupal/Core/Controller/ControllerBase.php
index a82770eaa5553cbd92107d09a846fcc72fa6ee72..a9569c622f55257a33a2d3f81b22cb55b728c64b 100644
--- a/core/lib/Drupal/Core/Controller/ControllerBase.php
+++ b/core/lib/Drupal/Core/Controller/ControllerBase.php
@@ -2,12 +2,12 @@
 
 namespace Drupal\Core\Controller;
 
+use Drupal\Core\DependencyInjection\AutowireTrait;
 use Drupal\Core\DependencyInjection\ContainerInjectionInterface;
 use Drupal\Core\Logger\LoggerChannelTrait;
 use Drupal\Core\Routing\RedirectDestinationTrait;
 use Drupal\Core\StringTranslation\StringTranslationTrait;
 use Drupal\Core\Url;
-use Symfony\Component\DependencyInjection\ContainerInterface;
 use Drupal\Core\Messenger\MessengerTrait;
 use Symfony\Component\HttpFoundation\RedirectResponse;
 
@@ -34,6 +34,7 @@
  */
 abstract class ControllerBase implements ContainerInjectionInterface {
 
+  use AutowireTrait;
   use LoggerChannelTrait;
   use MessengerTrait;
   use RedirectDestinationTrait;
@@ -102,13 +103,6 @@ abstract class ControllerBase implements ContainerInjectionInterface {
    */
   protected $formBuilder;
 
-  /**
-   * {@inheritdoc}
-   */
-  public static function create(ContainerInterface $container) {
-    return new static();
-  }
-
   /**
    * Retrieves the entity type manager.
    *
diff --git a/core/lib/Drupal/Core/DependencyInjection/AutowireTrait.php b/core/lib/Drupal/Core/DependencyInjection/AutowireTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..855bf20d82cd2acc517629746f4a1c734f312663
--- /dev/null
+++ b/core/lib/Drupal/Core/DependencyInjection/AutowireTrait.php
@@ -0,0 +1,47 @@
+<?php
+
+namespace Drupal\Core\DependencyInjection;
+
+use Symfony\Component\DependencyInjection\Attribute\Autowire;
+use Symfony\Component\DependencyInjection\ContainerInterface;
+use Symfony\Component\DependencyInjection\Exception\AutowiringFailedException;
+
+/**
+ * Defines a trait for automatically wiring dependencies from the container.
+ *
+ * This trait uses reflection and may cause performance issues with classes
+ * that will be instantiated multiple times.
+ */
+trait AutowireTrait {
+
+  /**
+   * Instantiates a new instance of the implementing class using autowiring.
+   *
+   * @param \Symfony\Component\DependencyInjection\ContainerInterface $container
+   *   The service container this instance should use.
+   *
+   * @return static
+   */
+  public static function create(ContainerInterface $container) {
+    $args = [];
+
+    if (method_exists(static::class, '__construct')) {
+      $constructor = new \ReflectionMethod(static::class, '__construct');
+      foreach ($constructor->getParameters() as $parameter) {
+        $service = (string) $parameter->getType();
+        foreach ($parameter->getAttributes(Autowire::class) as $attribute) {
+          $service = (string) $attribute->newInstance()->value;
+        }
+
+        if (!$container->has($service)) {
+          throw new AutowiringFailedException($service, sprintf('Cannot autowire service "%s": argument "$%s" of method "%s::_construct()", you should configure its value explicitly.', $service, $parameter->getName(), static::class));
+        }
+
+        $args[] = $container->get($service);
+      }
+    }
+
+    return new static(...$args);
+  }
+
+}
diff --git a/core/modules/node/src/Controller/NodeController.php b/core/modules/node/src/Controller/NodeController.php
index 25965374a72dd68c9594a542f286b383497f67bb..9cb1a50ad82c5c2c3a0e151cb61ae8e22468e76f 100644
--- a/core/modules/node/src/Controller/NodeController.php
+++ b/core/modules/node/src/Controller/NodeController.php
@@ -13,7 +13,6 @@
 use Drupal\node\NodeStorageInterface;
 use Drupal\node\NodeTypeInterface;
 use Drupal\node\NodeInterface;
-use Symfony\Component\DependencyInjection\ContainerInterface;
 
 /**
  * Returns responses for Node routes.
@@ -57,17 +56,6 @@ public function __construct(DateFormatterInterface $date_formatter, RendererInte
     $this->entityRepository = $entity_repository;
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public static function create(ContainerInterface $container) {
-    return new static(
-      $container->get('date.formatter'),
-      $container->get('renderer'),
-      $container->get('entity.repository')
-    );
-  }
-
   /**
    * Displays add content links for available content types.
    *
diff --git a/core/modules/system/tests/modules/system_test/src/Controller/BrokenSystemTestController.php b/core/modules/system/tests/modules/system_test/src/Controller/BrokenSystemTestController.php
new file mode 100644
index 0000000000000000000000000000000000000000..8ebfa84c38137d40195edb3972328d9e5efac848
--- /dev/null
+++ b/core/modules/system/tests/modules/system_test/src/Controller/BrokenSystemTestController.php
@@ -0,0 +1,23 @@
+<?php
+
+namespace Drupal\system_test\Controller;
+
+use Drupal\Core\Controller\ControllerBase;
+use Drupal\Core\Lock\LockBackendInterface;
+
+/**
+ * A controller that does not specify its autowired dependencies correctly.
+ */
+class BrokenSystemTestController extends ControllerBase {
+
+  /**
+   * Constructs the BrokenSystemTestController.
+   *
+   * @param \Drupal\Core\Lock\LockBackendInterface $lock
+   *   The lock service.
+   */
+  public function __construct(
+    protected LockBackendInterface $lock,
+  ) {}
+
+}
diff --git a/core/modules/system/tests/modules/system_test/src/Controller/SystemTestController.php b/core/modules/system/tests/modules/system_test/src/Controller/SystemTestController.php
index b806e582d6a45f3231b95d4294fd13584d35f28a..f5cf04c5bdce18d5a649c07f66a73fc077206caa 100644
--- a/core/modules/system/tests/modules/system_test/src/Controller/SystemTestController.php
+++ b/core/modules/system/tests/modules/system_test/src/Controller/SystemTestController.php
@@ -11,11 +11,11 @@
 use Drupal\Core\Render\Markup;
 use Drupal\Core\Session\AccountInterface;
 use Drupal\Core\Url;
+use Symfony\Component\DependencyInjection\Attribute\Autowire;
 use Symfony\Component\HttpFoundation\RedirectResponse;
 use Symfony\Component\HttpFoundation\Request;
 use Symfony\Component\HttpFoundation\Response;
 use Drupal\Core\Lock\LockBackendInterface;
-use Symfony\Component\DependencyInjection\ContainerInterface;
 
 /**
  * Controller routines for system_test routes.
@@ -71,7 +71,15 @@ class SystemTestController extends ControllerBase implements TrustedCallbackInte
    * @param \Drupal\Core\Messenger\MessengerInterface $messenger
    *   The messenger service.
    */
-  public function __construct(LockBackendInterface $lock, LockBackendInterface $persistent_lock, AccountInterface $current_user, RendererInterface $renderer, MessengerInterface $messenger) {
+  public function __construct(
+    #[Autowire(service: 'lock')]
+    LockBackendInterface $lock,
+    #[Autowire(service: 'lock.persistent')]
+    LockBackendInterface $persistent_lock,
+    AccountInterface $current_user,
+    RendererInterface $renderer,
+    MessengerInterface $messenger,
+  ) {
     $this->lock = $lock;
     $this->persistentLock = $persistent_lock;
     $this->currentUser = $current_user;
@@ -79,19 +87,6 @@ public function __construct(LockBackendInterface $lock, LockBackendInterface $pe
     $this->messenger = $messenger;
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public static function create(ContainerInterface $container) {
-    return new static(
-      $container->get('lock'),
-      $container->get('lock.persistent'),
-      $container->get('current_user'),
-      $container->get('renderer'),
-      $container->get('messenger')
-    );
-  }
-
   /**
    * Tests main content fallback.
    *
diff --git a/core/tests/Drupal/KernelTests/Core/Controller/ControllerBaseTest.php b/core/tests/Drupal/KernelTests/Core/Controller/ControllerBaseTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..8a74fd0e357d5d835112d8e3eb77f27151466b3f
--- /dev/null
+++ b/core/tests/Drupal/KernelTests/Core/Controller/ControllerBaseTest.php
@@ -0,0 +1,50 @@
+<?php
+
+namespace Drupal\KernelTests\Core\Controller;
+
+use Drupal\KernelTests\KernelTestBase;
+use Drupal\system_test\Controller\BrokenSystemTestController;
+use Drupal\system_test\Controller\SystemTestController;
+use Symfony\Component\DependencyInjection\Exception\AutowiringFailedException;
+
+/**
+ * Tests \Drupal\Core\Controller\ControllerBase.
+ *
+ * @coversDefaultClass \Drupal\Core\Controller\ControllerBase
+ * @group Controller
+ */
+class ControllerBaseTest extends KernelTestBase {
+
+  /**
+   * Modules to enable.
+   *
+   * @var array
+   */
+  protected static $modules = ['system_test', 'system'];
+
+  /**
+   * @covers ::create
+   */
+  public function testCreate() {
+    $controller = $this->container->get('class_resolver')->getInstanceFromDefinition(SystemTestController::class);
+
+    $property = new \ReflectionProperty(SystemTestController::class, 'lock');
+    $this->assertSame($this->container->get('lock'), $property->getValue($controller));
+
+    $property = new \ReflectionProperty(SystemTestController::class, 'persistentLock');
+    $this->assertSame($this->container->get('lock.persistent'), $property->getValue($controller));
+
+    $property = new \ReflectionProperty(SystemTestController::class, 'currentUser');
+    $this->assertSame($this->container->get('current_user'), $property->getValue($controller));
+  }
+
+  /**
+   * @covers ::create
+   */
+  public function testCreateException() {
+    $this->expectException(AutowiringFailedException::class);
+    $this->expectExceptionMessage('Cannot autowire service "Drupal\Core\Lock\LockBackendInterface": argument "$lock" of method "Drupal\system_test\Controller\BrokenSystemTestController::_construct()", you should configure its value explicitly.');
+    $this->container->get('class_resolver')->getInstanceFromDefinition(BrokenSystemTestController::class);
+  }
+
+}