php变量值有8种底层类型:6种标量/复合类型(bool、int、float、string、array、object)和2种特殊类型(null、resource);mixed和never仅为静态分析语法糖,非运行时类型。

PHP变量能存什么?看底层类型而不是语法糖
PHP变量本身不声明类型,但值有明确类型——共8种,其中6种是标量/复合类型,2种是特殊类型。别被$a = "1"这种写法骗了,gettype($a)返回的是"string",不是“可以随便变”。类型决定你能对它做什么,比如strlen()只认string,传null会警告,传array直接报错。
-
bool:仅true或false,注意0、""、[]、null在松散比较中“像false”,但它们不是bool -
int和float:整数和浮点数,PHP 8.0+ 支持int字面量后加_(如1_000_000),但float精度有限,别用==比金额 -
string:字节序列,不是Unicode字符串;多字节操作必须用mb_*函数,strlen()返回字节数,不是字符数 -
array:实际是有序映射(ordered map),键可为int或string,值可为任意类型,包括array自身(嵌套无限制) -
object:实例化类后的结果,每个对象有独立属性表和方法表;stdClass是空壳,常用于json_decode()默认返回 -
callable:不是独立类型,而是is_callable()能识别的结构,比如["ClassName", "method"]、function() {}、"strlen" -
null:只有一个值null,表示“未初始化”或“显式清空”,isset()对它返回false -
resource:外部资源句柄(如fopen()返回的文件指针),不能用json_encode()序列化,也不能直接打印内容
为什么gettype()有时返回"unknown type"?
这通常发生在你把一个已关闭或无效的resource传给gettype(),比如$fp = fopen("missing.txt", "r"); fclose($fp); gettype($fp)。PHP无法识别该资源状态,就退回到"unknown type"。这不是bug,是设计使然——resource生命周期由C层管理,PHP变量只是个引用。
- 判断资源是否有效,用
is_resource($var) && get_resource_type($var) !== "Unknown" - 不要依赖
gettype()做类型分发逻辑,改用is_string()、is_array()等专用函数 -
var_dump()比gettype()更可靠,它会显示resource(5) of type (stream)这类细节
unset()之后变量真的“不存在”了吗?
不完全是。unset($var)只是断开变量名到值的绑定,如果该值还被其他变量引用(比如$b = &$a),值本身不会销毁。更关键的是:isset($var)返回false,但array_key_exists("var", $GLOBALS)仍可能返回true——因为$GLOBALS是超全局数组,它的键不会因unset而消失,只是对应值变成null。
- 检查变量是否被定义且非
null,用isset($var);只查是否定义,用array_key_exists("var", $GLOBALS)(限全局作用域) - 在函数内
unset($arg)不影响调用方的原始变量,除非是引用传参:function f(&$x) { unset($x); } -
unset()对array元素安全,但对object属性无效(除非属性是public且未定义__unset())
PHP 8 的mixed和never是新类型吗?
不是运行时类型,是类型声明语法糖,只在PHPStan、Psalm、PHP 8.0+的declare(strict_types=1)下起作用。运行时gettype($x)永远不会返回"mixed"或"never"——它们只存在于PHP解析器和静态分析器中。
立即学习“PHP免费学习笔记(深入)”;
-
function foo(): mixed表示“返回任意类型”,但调用方仍需自己判断返回值类型,PHP不自动做转换 -
function bar(): never表示“永不返回”,比如抛异常或die(),可用于控制流分析,但不改变运行时行为 - 误写
mixed当运行时类型会导致困惑,比如is_mixed($x)根本不存在,PHP没提供这个函数
json_encode()了”,其实问题不在变量,而在它当前绑定的那个resource或closure——这些类型天生不可序列化。










