Objective-C const常量的优雅使用方法

目录
  • 正文
  • Objective-C 的常量声明方式
  • 在 Objective-C 中使用 let 来声明常量

正文

在编写代码时经常要使用常量,来替代 magic number。比较简单的做法是通过预处理指令 #define 来实现。

#define ANIMATION_DURATION 0.3

上述预处理指令会在编译时的预处理阶段会将代码中 ANIMATION_DURATION 字符串替换为 0.3。这种定义常量的方式比较简便,但是存在两个问题:

  • 丢失了类型信息。
  • 若该预处理指令声明在头文件中,引入该头文件的代码,ANIMATION_DURATION 都会被替换,可能出现冲突。

Objective-C 的常量声明方式

幸运的是,Objective-C 中提供了 const 关键字,可以用来定义常量。const 关键字可以对变量加以限定,使其值不能被改变,在整个作用域中都保持固定。

const NSTimeInterval kAnimationDuration = 0.3;

这种方式定义的常量包含类型信息,且在编译时即可检查是否与其他常量出现冲突。如果试图修改由 const 修饰符所声明的变量,那么编译器就会报错。

如果常量仅在某个实现文件中使用,还应该加上 static 关键字,否则会被视为全局常量。若不使用 static,编译器会为它创建一个外部符号,若另一个编译单元中也声明了同名变量,就会报错。

static const NSTimeInterval kAnimationDuration = 0.3;

当一个变量同时使用了 staticconst,那么编译器并不会创建符号,而是会像 #define 预处理指令一样,把所有遇到的变量替换为常值。

有时候需要把一个常量暴露给外界使用,比如通知,此类常量需放在全局符号表中。可以使用 extern 关键字,在头文件中进行声明:

// .h
extern NSString * const AFNetworkingTaskDidResumeNotification;
// .m
NSString * const AFNetworkingTaskDidResumeNotification = @"com.alamofire.networking.task.resume";

该常量在头文件中声明,在实现文件中定义。需要注意的是 const 写在指针类型的右边意味着该指针的指向不可被改变,若写在左边意味着该指针指向的内容不可被改变。

按上述方式实现并定义后,在编译时生成目标文件时,编译器会在数据段为字符串分配存储空间。

Foundation 框架中,苹果为了兼容 C++ 中对 extern 的使用,提供了宏:

#if defined(__cplusplus)
#define FOUNDATION_EXTERN extern "C"
#else
#define FOUNDATION_EXTERN extern
#endif
#define FOUNDATION_EXPORT FOUNDATION_EXTERN
#define FOUNDATION_IMPORT FOUNDATION_EXTERN

一个 C++ 程序中可能包含其他语言编写的部分代码,同样,C++ 编写的代码片段也可能被用在其他语言编写的代码中。但是,不同语言编写的代码相互调用是困难的,更何况用同一种语言编写,使用不同编译器进行编译的情况。因为,不同语言或者同种语言在不同编译器上编译时,在注册变量,传递参数和参数在栈上的布局上可能存在差异。

为了使它们遵守统一规则,可以使用 extern 指定一个编译和链接规约。extern "C" 指令中的 C,表示的是一种编译和链接规约,而不是一种语言。C 表示符合 C 语言的编译和链接规约的任何语言。

还要说明的是,extern "C" 指令指定的编译和链接规约,不会影响语义,只是改变编译和链接的方式。

FOUNDATION_EXPORTFOUNDATION_IMPORT 是用来兼容 Win32 应用程序的,移动端开发可以忽略。

所以上述对全局常量的声明,可以写成:

// .h
FOUNDATION_EXPORT NSString * const AFNetworkingTaskDidResumeNotification;
// .m
NSString * const AFNetworkingTaskDidResumeNotification = @"com.alamofire.networking.task.resume";

在 Objective-C 中使用 let 来声明常量

使用过 Swift 的同学,一定对其声明常量的方式的简洁性印象深刻,在 Swift 中声明常量的方式如下所示:

let kAnimationDuration = 0.3

之所以能如此简洁,是因为 Swift 具有 let 关键字和类型推断的能力,但其实在 Objective-C 中也可以通过类似的方式来书写常量。

Objective-C 中有一个关键字,是 __auto_type,可以实现类似 Swift 中类型推断能力的关键字,如下所示:

const __auto_type kAnimationDuration = 0.3;

可能对于简单的数据类型,这样的优势不是很明显,但是对于具有复杂泛型的类型来说,可以说优势很大了:

// 旧方式
NSArray<NSDictionary<NSString *, NSString *> *> *models = ...;
// 新方式
__auto_type models = ...;

同时,可以通过宏的方式,来减少 __auto_type 的书写,即可实现通过 let 声明常量,var 声明变量。其中 auto 关键字是为了兼容 C++。

#if defined(__cplusplus)
#define let auto const
#else
#define let const __auto_type
#endif
#if defined(__cplusplus)
#define var auto
#else
#define var __auto_type
#endif

声明了上面的宏之后,就可以直接使用了:

let kAnimationDuration = 0.3;

以上就是Objective-C const常量的优雅使用方法的详细内容,更多关于Objective-C const常量的资料请关注我们其它相关文章!

时间: 2022-08-07

Objective-C优雅使用KVO观察属性值变化

目录 引言 KVOController YYCategories 引言 KVO 是苹果为我们提供的一套强大的机制,用于观察属性值的变化,但是大家在日常开发中想必多少也感受到了使用上的一些不便利,比如: 添加观察者和移除观察者的次数需要一一对应,否则会 Crash. 添加观察者和接受到属性变更通知的位置是分开的,不利于判断上下文. 多次对同一个属性值进行观察,会触发多次回调,影响业务逻辑. 为了解决上述三个问题,业界提出了一些方便开发者的开源方案,我们一起来看一下. KVOController K

Objective-C之Category实现分类示例详解

目录 引言 编译时 运行时 引言 在写 Objective-C 代码的时候,如果想给没法获得源码的类增加一些方法,Category 即分类是一种很好的方法,本文将带你了解分类是如何实现为类添加方法的. 先说结论,分类中的方法会在编译时变成 category_t 结构体的变量,在运行时合并进主类,分类中的方法会放在主类中方法的前面,主类中原有的方法不会被覆盖.同时,同名的分类方法,后编译的分类方法会“覆盖”先编译的分类方法. 编译时 在编译时,所有我们写的分类,都会转化为 category_t 结

在Swift中使用KVO的细节以及内部实现解析(推荐)

KVO是什么? KVO 是 Objective-C 对观察者设计模式的一种实现.[另外一种是:通知机制(notification),详情参考:iOS 趣谈设计模式--通知]: KVO提供一种机制,指定一个被观察对象(例如A类),当对象某个属性(例如A中的字符串name)发生更改时,对象会获得通知,并作出相应处理:[且不需要给被观察的对象添加任何额外代码,就能使用KVO机制] 在MVC设计架构下的项目,KVO机制很适合实现mode模型和view视图之间的通讯. 例如:代码中,在模型类A创建属性数据

iOS自动移除KVO观察者的实现方法

问题 KVO即:Key-Value Observing, 直译为:基于键值的观察者. 它提供一种机制,当指定的对象的属性被修改后,则对象就会接受到通知. 简单的说就是每次指定的被观察的对象的属性被修改后,KVO就会自动通知相应的观察者了. KVO的优点:当有属性改变,KVO会提供自动的消息通知. 这样开发人员不需要自己去实现这样的方案:每次属性改变了就发送消息通知. 这是KVO机制提供的最大的优点. 因为这个方案已经被明确定义,获得框架级支持,可以方便地采用. 开发人员不需要添加任何代码,不需要

iOS 监听回调机制KVO实例

监听某个对象,如果这个对象的数据发生变化,会发送给监听者从而触发回调函数 [self.bean addObserver:self forKeyPath:@"data" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:NULL]; 这个就是注册监听,这个@"data"作为标识符方便回调函数辨认 -(void)observeValueForKeyPath:(NSStrin

JavaScript监听触摸事件代码实例

这篇文章主要介绍了JavaScript监听触摸事件代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 监听 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <meta name="viewport" content="initial-scale=1, maximum-scale=1, use

android 控件同时监听单击和双击实例

不适用click而用touch 自定义监听: class myOnGestureListener extends GestureDetector.SimpleOnGestureListener { @Override public boolean onDoubleTap(MotionEvent e) { //点赞 mLoadingListener.onFinishedLoading("0");//取消点赞 是一个接口 //已经点赞 更换图片 1:已经点赞 0 :没有点赞 if (lik

JS实现移动端实时监听输入框变化的实例代码

如果是在pc端,监听输入框你可能想到focus.blur.hover.onkeyup这些事件,但是如果是在移动端使用使用这些事件用户体验极差,因为你要用户收起键盘并且再点个空白处才生效,甚至还存在很大的兼容差异.那么怎么解决呢? 方案一 以前做一个简单的模糊搜索框的需求,大致思路是定义一个1s左右执行的定时器,定时器函数里面的逻辑是 判断文本框内容的长度,如果跟上一次长度有变化,就走一遍查询,如果没有变化不执行任何操作. 方案二 利用jquery提供的 input propertychange

iOS监听手机锁屏状态

iPhone的锁屏监测分为两种方式监听: 1. 程序在前台,这种比较简单.直接使用Darwin层的通知就可以了: #import <notify.h> #define NotificationLock CFSTR("com.apple.springboard.lockcomplete") #define NotificationChange CFSTR("com.apple.springboard.lockstate") #define Notifica

java 回调机制的实例详解

java 回调机制的实例详解 序言 最近接触到了回调机制(CallBack).初识时感觉比较混乱,而且在网上搜索到的相关的讲解,要么一言带过,要么说的比较单纯的像是给CallBack做了一个定义.当然了,我在理解了回调之后,再去看网上的各种讲解,确实没什么问题.但是,对于初学的我来说,缺了一个循序渐进的过程.此处,将我对回调机制的个人理解,按照由浅到深的顺序描述一下,如有不妥之处,望不吝赐教! 开始之前,先想象一个场景:幼稚园的小朋友刚刚学习了10以内的加法. 第1章. 故事的缘起 幼师在黑板上

Jquery实时监听input value的实例

实例如下: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</title> </head> <body id="lia-body"> <div class="lia-content"> <div class

addeventlistener监听scroll跟touch(实例讲解)

这三个事件只在手机上生效 touchstart:手指开始触屏 touchmove:手指移动 touchend:手指触屏结束 这个事件在手机上跟在pc端都生效 scroll事件 addeventlistener(name,callback,optional,useCapture) useCapture:事件是否用捕获事件,从外到里,true                         默认为false:使用冒泡事件,从里到外 optional:{      passive:false时,阻止默

Spring boot + LayIM + t-io 实现文件上传、 监听用户状态的实例代码

前言 今天的主要内容是:LayIM消息中图片,文件的上传对接.用户状态的监听.群在线人数的监听.下面我将挨个介绍. 图片上传 关于Spring boot中的文件上传的博客很多,我也是摘抄了部分代码.上传部分简单介绍,主要介绍在开发过程中遇到的问题.首先我们看一下LayIM的相应的接口: layim.config({ //上传图片接口 ,uploadImage: {url: '/upload/file'} //上传文件接口 ,uploadFile: {url: '/upload/file'} //

使用 vue 实例更好的监听事件及vue实例的方法

文章举例说明一下在 vue 中如何更好的监听浏览器事件.原文介绍了一种新增 vue 实例的方法,单独监听事件.这样代码书写较为简练,容易管理. 当监听如下事件的传统做法是: window.scrollX window.scrollY window.innerHeight window.innerWidth 通常需要书写很多代码: created () { this.$el.addEventListener('click', this.someMethod) }, destroyed () { t