← Home ← Codex ← DEBT
Browse by Category
+ added · updated 7d
← Back to glossary

callable vs Closure vs First-Class Callable

PHP PHP 7.1+ Intermediate
debt(d5/e3/b5/t7)
d5 Detectability Operational debt — how invisible misuse is to your safety net

Closest to 'specialist tool catches it' (d5). The detection_hints list phpstan and psalm — both specialist static analysis tools — as the means to catch callable vs Closure misuse. The code_pattern `callable $` confirms these tools flag the type hint, but it won't be caught by a default linter or compiler error.

e3 Effort Remediation debt — work required to fix once spotted

Closest to 'simple parameterised fix' (e3). The quick_fix says to replace callable type hints with Closure and use strlen(...) or Closure::fromCallable() — these are mostly find-and-replace substitutions within call sites, but touching multiple type hints across a component is a small refactor rather than a one-line patch.

b5 Burden Structural debt — long-term weight of choosing wrong

Closest to 'persistent productivity tax' (e5 → b5). The term applies_to web, cli, and queue-worker contexts, meaning the callable type hint pattern can appear anywhere callbacks are used across the codebase. Using callable instead of Closure silently undermines static analysis across all these contexts, taxing many work streams without dominating the architecture.

t7 Trap Cognitive debt — how counter-intuitive correct behaviour is

Closest to 'serious trap' (t7). The misconception field directly states: 'callable and Closure are the same — callable is a loose type that accepts strings and arrays too. Closure is a specific typed class.' This contradicts how developers who know typed languages reason about function types. The common mistake of using string-based callables that break on refactoring compounds the trap, making it a serious contradiction of developer expectations.

About DEBT scoring →

TL;DR

PHP has three callable forms: loose callable (string/array), typed Closure, and PHP 8.1 first-class callables (strlen(...)) — prefer Closure or first-class callables for type safety.

Explanation

callable is a loose type hint that accepts strings ('strlen'), arrays ([$obj, 'method']), and Closures. It provides no IDE autocompletion or static analysis. Closure is a specific class for anonymous functions — preferred for type-safe callbacks. PHP 8.1 introduced first-class callables: strlen(...) creates a Closure from any callable. Closure::fromCallable() does the same in PHP 7.1+. For strict code: never type-hint callable, always use Closure or a specific interface. PHPStan/Psalm can check Closure parameter types.

Common Misconception

callable and Closure are the same — callable is a loose type that accepts strings and arrays too. Closure is a specific typed class.

Why It Matters

callable type hints prevent static analysis from verifying callback signatures — using Closure enables type checking of the callback's parameters and return type.

Common Mistakes

  • Type-hinting callable instead of Closure — loses static analysis.
  • Using string-based callables ('strlen') that break with refactoring.
  • Not knowing strlen(...) syntax (PHP 8.1) for converting built-ins to Closures.

Code Examples

✗ Vulnerable
function process(callable $callback, array $items): array {
    return array_map($callback, $items); // No type safety on callback
}
process('strtolower', $items); // Works but fragile
✓ Fixed
// PHP 7.1+:
$fn = Closure::fromCallable('strtolower');

// PHP 8.1+ first-class callables:
$fn = strtolower(...);
$fn = $obj->method(...);
$fn = ClassName::staticMethod(...);

// Type-safe callback parameter:
function process(Closure $callback, array $items): array {
    return array_map($callback, $items);
}

Added 22 Mar 2026
Views 47
Rate this term
No ratings yet
🤖 AI Guestbook educational data only
| |
Last 30 days
0 pings T 1 ping W 1 ping T 0 pings F 0 pings S 0 pings S 0 pings M 0 pings T 0 pings W 2 pings T 1 ping F 1 ping S 0 pings S 2 pings M 1 ping T 1 ping W 0 pings T 1 ping F 1 ping S 0 pings S 0 pings M 0 pings T 0 pings W 0 pings T 1 ping F 0 pings S 0 pings S 0 pings M 1 ping T 0 pings W
No pings yet today
PetalBot 1
Amazonbot 9 Scrapy 7 ChatGPT 3 Unknown AI 3 Perplexity 3 Ahrefs 3 SEMrush 3 Google 2 Meta AI 2 Claude 2 Bing 1 PetalBot 1
crawler 35 crawler_json 3 pre-tracking 1
DEV INTEL Tools & Severity
🔵 Info ⚙ Fix effort: Low
⚡ Quick Fix
Replace callable type hints with Closure. Use strlen(...) syntax (PHP 8.1) or Closure::fromCallable() to convert built-in functions to typed Closures.
📦 Applies To
PHP 7.1+ web cli queue-worker
🔗 Prerequisites
🔍 Detection Hints
callable \$
Auto-detectable: ✓ Yes phpstan psalm
🤖 AI Agent
Confidence: Medium False Positives: Medium ✓ Auto-fixable Fix: Low Context: Function


✓ schema.org compliant