js 把一个函数赋给一个变量时带括号与不带括号的区别

本文探讨了JavaScript中函数作为变量赋值的不同方式及其效果。解析了直接赋值与执行后赋值的区别,并通过实例说明了闭包及作用域链的概念。
function test(){
        return "hello";
}
var aaa = test;
console.log(aaa );
console.log(aaa ());

第一种情况://function test(){return “hello”;} //此种写法是将函数赋值给变量,也就是将内存给这段函数的首地址赋值给变量

第二种情况://hello 此种写法是将函数的返回值赋值给变量

function hi(){  
  var a = 1;  
  return function(){
   console.log(a++);
  };  
};         
var aaa = hi();
var bbb = hi;
aaa();
aaa();
bbb();

这段代码中 aaa与bbb的区别是什么?

aaa 是将 hi() 的运行结果赋值给它,即 return 返回的匿名函数,此时有一个闭包,则每次调用 aaa 时都访问的同一个 a,aaa() 第一次运行结果为 1,第二次为2
而 bbb 将是将 hi 这个函数名赋值给它,则调用 bbb() 后返回一个函数表达式,即function(){console.log(a++)};

bbb就是hi本身,换句话说就是外层的function。bbb()的结果应该跟hi()的结果一样。
aaa是hi()调用之后返回的那个东西,也就是说它是里层的function,a是它闭包里面的一个对象。aaa()的调用结果是console.log(a++)。

总结:加括号是把函数执行后的值赋值给变量;不加括号是把函数本身赋值给变量

<think>我们正在讨论C/C++预处理指令中的宏定义。用户的问题是关于宏定义括号括号的情况。 在之前的对话中,我们讨论了`RECOVERY_TASK_LOCK`宏,并且用户现在提到“一个宏定义是()的,另一个括号的”。 我们需要解释宏定义中括号括号区别,以及它们各自的使用场景。 注意:宏定义中,括号通常表示这是一个函数式宏(function-like macro),可以接受参数。 而括号的宏是对象式宏(object-like macro),通常用于定义常量或简单的替换文本。 但是,用户似乎是在比较两个宏,一个括号一个括号。我们需要说明这两者的差异。 根据用户的问题,可能是在同一个宏名称上,有定义为括号函数式),有定义为括号(对象式),然后通过预处理指令(如#ifdef和#undef)来切换。 因此,我们需要解释: 1. 函数式宏和对象式宏的区别。 2. 为什么需要在同一个宏名称上切换这两种形式(可能的原因)。 3. 这样做的影响。 同,我们需要按照规则在最后提出3-5个相关问题。 注意:在之前的回答中,我们提到了使用`#ifdef`和`#undef`来重新定义宏,用户可能是针对这个场景提问。 示例: 括号的宏定义:`#define RECOVERY_TASK_LOCK 1` 或 `#define RECOVERY_TASK_LOCK do{...}while(0)` 括号的宏定义:`#define RECOVERY_TASK_LOCK() do{...}while(0)` 区别在于:括号的宏可以接受参数(即使参数为空,也需要括号),而括号的宏则能传递参数。 当我们在代码中使用宏,如果定义的是函数式宏(括号),那么使用要加括号(即使没有参数也要加空括号);如果是对象式宏,则直接使用名称。 因此,切换这两种形式会导致使用宏的代码语法发生变化。为了避免语法错误,通常会在同一个宏名称上切换这两种形式,除非使用宏的代码也做了相应调整。 但是,用户的问题可能源于实际代码中看到这样的切换,所以我们也要说明在什么情况下会这样做。 可能的情况: 1. 误操作:小心将同一个宏定义为两种同的形式,这会导致错误。 2. 设计需求:某个宏在一种配置下需要参数,另一种配置下需要,但这种情况很少见,通常用同的宏名。 更常见的情况是:我们可能将宏从对象式改为函数式,或者反过来,并且通过条件编译来切换。但是,这要求使用宏的代码也要根据宏的定义方式变化,因此通常推荐在同一个宏名上切换类型。 因此,我们可以这样组织回答: 1. 解释两种宏定义的区别。 2. 说明在同一个宏名上切换两种形式的后果(需要调用代码也相应改变,否则编译错误)。 3. 给出一个切换的例子,并说明如何避免问题(比如使用条件编译来区分调用方式)。 但是,如果用户的问题只是单纯询问两个同的宏(一个括号一个括号)的作用,那么我们可以解释它们各自的使用场景。 根据上下文,用户的问题是在之前讨论`RECOVERY_TASK_LOCK`宏的重定义,所以这里我们假设用户是在问:为什么在重定义的候,可能一个版本括号,另一个括号? 因此,重点在于:在重定义宏,改变宏的类型(从对象式改为函数式或反之)会来什么影响?如何处理? 回答结构: ### 宏定义括号括号的本质区别 1. **对象式宏 (Object-like Macro)** 定义示例:`#define RECOVERY_TASK_LOCK pthread_mutex_lock(&mutex)` 使用方式:直接写宏名,例如:`RECOVERY_TASK_LOCK;` 特点:仅做文本替换,支持参数传递。 2. **函数式宏 (Function-like Macro)** 定义示例:`#define RECOVERY_TASK_LOCK() pthread_mutex_lock(&mutex)` 使用方式:必须加括号,例如:`RECOVERY_TASK_LOCK();` 特点:可接受参数(此处为空参数列表),语法类似函数。 ### 切换宏类型的问题 - 若旧版宏为对象式,切换为函数式后: - 原有代码 `RECOVERY_TASK_LOCK` (无括号)将无法编译(预处理器会将其视为未定义的对象式宏,而实际上现在定义为函数式宏) - 必须修改所有调用点为 `RECOVERY_TASK_LOCK()` - 反之亦然:函数式宏切换到对象式宏,原有括号的调用也会出错。 ### 解决方案 1. **版本兼容性设计** 通过条件编译统一调用接口(推荐混合使用): ```c #ifdef USE_FUNCTION_STYLE #define RECOVERY_TASK_LOCK() /* 函数式实现 */ #else #define RECOVERY_TASK_LOCK /* 对象式实现 */ #endif ``` 但此方案要求调用者提前知晓宏类型,实际难以维护。 2. **重构策略**(推荐) - **保留原名称用于主流方案**:例如始终使用函数式宏(括号) - **废弃方案用新名称**:如需对象式宏,则定义新宏名(如 `RECOVERY_TASK_LOCK_V1`) ### 典型应用场景 - **空参数函数式宏**:用于替代代码块,避免变量污染(通过 `do { ... } while(0)` 包裹) ```c #define LOCK() do { \ pthread_mutex_lock(&mutex); \ } while(0) ``` - **对象式宏**:定义常量或简单表达式(如 `#define MAX_RETRY 3`) ### 结论 在同一个宏名上切换括号括号的定义会导致**调用语法兼容**。若必须切换,应同步修改所有调用点,或通过设计隔离同版本(如命名区分)。 最后,提出相关问题。 §§相关问题§§ 1. 函数式宏的空参数列表是否允许省略括号? 2. 为什么在宏定义中常用 `do { ... } while(0)` 包裹代码块? 3. 如何避免因宏类型切换导致的代码兼容性问题? 4. 在C++中,哪些特性可以替代函数式宏以实现类型安全? </think>### 宏定义括号括号的本质区别 #### 1. **函数式宏 vs 对象式宏** ```c // 括号函数式宏(可参数) #define MACRO_WITH_BRACKETS() do_action() // 括号:对象式宏(常量/简单替换) #define MACRO_WITHOUT_BRACKETS 0xFFFF ``` #### 2. **核心差异** | 特性 | 括号 `()` | 括号 | |--------------------|----------------------|----------------------| | **参数支持** | 可接受参数(如 `MACRO(arg)`) | 能接受参数 | | **预处理行为** | 按函数模板展开 | 直接文本替换 | | **使用场景** | 复杂逻辑/代码块封装 | 常量定义/简单标识符 | | **空参数兼容性** | 支持空参数(`MACRO()`)| 支持 | #### 3. **典型错误示例** ```c // 错误:括号宏尝试接受参数 #define LOCK mutex_lock(&m) LOCK(123); // 编译失败:宏接受参数 // 正确:括号定义 #define LOCK() mutex_lock(&m) LOCK(); // 合法调用 ``` #### 4. **跨版本兼容设计** 通过预处理指令实现无缝切换: ```c // 旧版对象式宏 #define RECOVERY_LOCK 1 // 新版切换为函数式宏 #ifdef RECOVERY_LOCK #undef RECOVERY_LOCK #endif #define RECOVERY_LOCK() custom_lock_impl() ``` --- ### 关键设计原则 1. **语法明确性** 括号宏强制函数式调用语义(如 `MACRO()` 即使无参数也需括号) 2. **参数安全** 函数式宏可通过 `##__VA_ARGS__` 处理可变参数(如日志宏) 3. **代码可读性** 括号宏明确区分宏变量: ```c int FLAG = 1; // 变量 DO_ACTION(); // 明显是宏调用 ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值