本文详解因函数参数名与函数名同名引发的变量遮蔽(shadowing)问题,导致 `typeerror: cityname is not a function` 错误,并提供可立即使用的修复方案与最佳实践。
在 Node.js 命令行应用中,使用 readlin
e-sync 实现交互式城市天气查询时,若在异步回调中递归调用同名函数(如 cityname()),却意外报错 TypeError: cityname is not a function,根本原因并非作用域缺失或声明提升问题,而是参数名与函数名冲突造成的变量遮蔽(variable shadowing)。
JavaScript 中,当函数形参名为 cityname 时,该参数会在整个函数作用域内优先于同名的外部函数声明。因此,在 getWeatherData(cityname, apiKey) 内部,cityname 指向的是传入的字符串参数(如 "Beijing"),而非全局函数 cityname() —— 导致 case 1: cityname(); 实际尝试调用一个字符串,从而抛出类型错误。
✅ 正确修复方式:重命名冲突参数,避免与函数名重复。例如将 cityname 参数改为 myCity、city 或 cityNameInput 等语义清晰且无歧义的名称。
以下是修复后的完整可运行代码(已优化可读性与健壮性):
import axios from "axios";
import readlineSync from "readline-sync";
// 启动入口
cityname();
function cityname() {
const apiKey = 'xyz';
const name = readlineSync.question('Enter City Name:\n');
return getWeatherData(name, apiKey);
}
function getWeatherData(city, apiKey) { // ✅ 参数名已更正:city ≠ cityname
axios
.get(`https://api.openweathermap.org/data/2.5/weather?q=${city}&appid=${apiKey}`)
.then((res) => {
const data = res.data;
const options = readlineSync.questionInt(
`Select an option below for ${city}:\n` +
`1. Change City\n` +
`0. Exit\n`
);
switch (options) {
case 1:
cityname(); // ✅ 此处正确调用函数,无遮蔽
break;
case 0:
console.log('Exiting the program...');
break;
default:
console.log('Invalid option. Please try again.');
// 可选:递归重试当前菜单,而非退出
getWeatherData(city, apiKey);
break;
}
})
.catch((error) => {
console.error('Failed to fetch weather data:', error.message);
console.log('Please check the city name and try again.');
cityname(); // 网络失败时也允许重新输入城市
});
}⚠️ 注意事项:
总结:命名一致性固然重要,但语义明确性与作用域安全性更为关键。将参数命名为 city 而非 cityname,既消除遮蔽风险,又更准确表达其数据类型(字符串),是兼顾可维护性与健壮性的最佳实践。