Smarty初始化必须手动设置模板目录和编译目录,否则因路径不存在或权限不足导致报错;变量名严格区分大小写且不可含点号;foreach需显式声明key/item别名;调试模式需手动启用且仅限开发环境。
不显式指定 $smarty->setTemplateDir() 和 $smarty->setCompileDir(),Smarty 会尝试使用默认路径(如 templates/ 和 templates_c/),但这些路径往往不存在或权限不足,导致报错 SmartyException: unable to write file 或 template not found。
实操建议:
./templates/)存在且 Web 服务器用户(如 www-data、apache)有读取权限./templates_c/)必须可写,建议用 chmod 755 templates_c 或更严格地设为 775 并确认组属正确$smarty = new Smarty(); $smarty->setTemplateDir(__DIR__ . '/templates/'); $smarty->setCompileDir(__DIR__ . '/templates_c/'); $smarty->setCacheDir(__DIR__ . '/cache/'); // 如启用缓存,同样需指定且可写
在 PHP 中调用 $smarty->assign('user_name', $name),模板里必须用 {$user_name},写成 {$User_Name} 或 {$user.name} 都无法解析——Smarty 的变量命名规则是纯字母+数字+下划线,且严格区分大小写。
常见错误现象:
{$data.user.id})→ 实际应先在 PHP 层展开或用 -> 语法({$data->user->id}),或改用 assign() 传入扁平化变量{$arr.$key} 可行,但 {$arr.{$key}} 语法错误)→ 需用 {assign var="val" value=$arr[$key]} 中转Smarty 的 {foreach} 不像 PHP foreach 那样自动推导键/值,漏写 key 或 item 别名会导致语法错误或意外输出。例如 {foreach $users as $u} 是非法的,必须写成 {foreach $users as $u} → 实际应为 {foreach $users as $u}?不对,Smarty 正确写法是:{foreach $users as $u} 也不对——它要求明确关键字:
{foreach $users as $u}
{$u.name}
{/foreach} 这种写法在 Smarty 3+ 是允许的(等价于 item=$u),但若需 k
{foreach $users key=$i item=$u}
[{$i}] {$u.name}
{/foreach}
使用场景提醒:
key 就拿不到键名;不声明 item 就拿不到值(哪怕只写 item=$v 也比省略安全)$item,内层也用 $item 会导致覆盖 → 建议用语义化别名,如 $category / $product
{foreach} 不执行内容,无需额外判空,但若需 fallback 提示,得用 {foreachelse}
Smarty 自带的调试模板(按 Ctrl+Shift+I 弹出变量面板)不是默认打开的,必须调用 $smarty->debugging = true;,否则即使页面加载完成也看不到调试窗口。
但要注意:
debugging,它会暴露全部已分配变量、配置、模板路径,构成信息泄露风险$smarty->enableDebugging(false) 或部署时未注释掉该行,调试窗口不会出现,且无任何提示X-Debug 相关字段,导致调试窗口不弹出 → 优先检查浏览器控制台是否有 JS 报错,再确认 PHP 端是否真正执行了 $smarty->debugging = true