【学习iOS高质量开发】——熟悉Objective-C

news2024/11/17 2:33:52

文章目录

  • 一、Objective-C的起源
    • 1.OC和其它面向对象语言
    • 2.OC和C语言
    • 3.要点
  • 二、在类的头文件中尽量少引用其他头文件
    • 1.OC的文件
    • 2.向前声明的好处
    • 3.如何正确引入头文件
    • 4.要点
  • 三、多用字面量语法,少用与之等价的方法
    • 1.何为字面量语法
    • 2.字面数值
    • 3.字面量数组
    • 4.字面量字典
    • 5.可变数组和字典
    • 6.局限性
    • 7.要点
  • 四、多用类型常量,少用#define预处理指令
    • 1.类型常量与#define
    • 2.static和const
    • 3.外界可见的常值变量
    • 4.要点
  • 五、用枚举表示状态、选项、状态码
    • 1.枚举表示状态
    • 2.枚举表示选项
    • 3.辅助宏
    • 4.枚举表示状态码
    • 5.要点


一、Objective-C的起源

1.OC和其它面向对象语言

Objective-C和Java、C++都是面向对象语言但是语法上有些许不同。OC使用“消息结构”而不是“函数调用”,这二者的区别主要体现在:使用消息结构的语言,其运行所应执行的代码由运行环境来决定;使用函数调用的语言,则由编译器决定。OC的重要工作都是由运行期组件来完成而不是编译器。

消息与函数调用之间的区别:

//Messaging(Objevtive-C)
Object *obj = [Object new];
[obj performWith:parameter1 and:parameter2];

//Function calling(C++)
Object *obj = new Object;
obj->perform(parameter1, parameter2);

运行期组件包含OC面向对象特性的全部数据结构和函数,它的本质就是一种动态库并且能够桥梁一样把我们所编写的代码连接起来。

2.OC和C语言

OC是C语言的超集,因此C语言中所有的功能在OC中依然可以使用,并且理解OC中的内存模型和引用计数的一个前提就是理解C语言的内存模型。OC的指针是指向对象的,想要声明一个变量并且指向某个对象可以使用如下语法:


NSString *someString = @"The string";

OC中所有的对象都必须这么声明,因为OC中的对象所占的内存是在堆区中而不会分配到栈区中,这样的好处有:

1.在堆区分配对象意味着对象的生命周期可以超出当前函数的作用域。如果对象被分配在栈区,当函数返回时,栈上的局部变量会被自动销毁,这将导致对象在栈上被销毁,进而引发悬垂指针的问题。
2.指针变量通常存储在栈中,对象在堆区分配可以被多个指针同时所指。
3.堆区可以容纳任意大小的对象,而栈区的大小通常是有限的。由于对象的大小不确定,将其分配到堆区可以更好地适应各种大小的对象需求。

在这里插入图片描述

3.要点

1.Objective-C为C语言添加了面向对象特性,是其超集。Objective-C使用动态绑定的消息结构,也就是说,在运行时才会检查对象类型。接收一条消息之后,究竟应执行何种代码,由运行期环境而非编译器来决定。

2.理解C语言的核心概念有助于写好Objective-C程序。尤其是要掌握内存模型和指针。

二、在类的头文件中尽量少引用其他头文件

1.OC的文件

与C和C++一样,Objective-C也使用“头文件”(header file)与“实现文件”(implementation file)来区隔代码。用Objective-C编写“类”(class)的标准方式:以类名做文件名,分别创建两个文件,头文件后缀用.h,实现文件后缀用.m。

//EOCPerson.h
#import <Foundation/Foundation.h>
@interface EOCPerson : NSObject
@end

//EOCPerson.m
#import "EOCPerson.h"
@implementation EOCPerson
//Implementation of methods
@end

如果创建一个新类例如EOCEmployer类然后将其设置为EOCPerson类的属性就需要在ECOPerson类的接口部分添加#import "EOCEmployer.h"。有时候我们使用EOCPerson类时不需要知道ECOEmployer类的全部细节只需要知道有这个类就行,此时就可以使用向前声明(forward declaring) 需要修改的就是在接口部分将#import "EOCEmployer.h"替换成@class EOCEmployer;

但是在ECOPerson类的实现部分要使用ECOEmpoyer类就必须知道它的全部细节因此就需要添加#import "EOCEmployer.h"。需要注意的是,将引入头文件的时机尽量延后,只在需要的时候才引入,这样就可以减少类的使用者所需引入头文件的数量。

2.向前声明的好处

向前声明解决了两个类相互引用的问题。例如要编译EOCEmployer类则编译器必须知道EOCPerson类,要编译EOCPerson类则编译器必须知道EOCEmployer类,所以需要在两个头文件中互相引用对方的头文件,但是这样就会导致循环引用从而引发两个类中有一个不能被正常编译。使用向前声明则不需要引入头文件也就不会通过头文件的循环引用导致出错。

3.如何正确引入头文件

1.如果你写的类继承于某个超类,则必须引入定义那个超类的头文件。

2.如果声明某个类遵循某个协议,那么这个协议就必须有完整的定义也就是要知道这个协议的全部细节,因此就不能使用向前声明。所以要实现属性、实例变量或者遵循协议而必须引入头文件的话,则应该将其移至“class-continuation分类中”。

4.要点

1.除非确有必要,否则不要引入头文件。一般来说,应在某个类的头文件中使用向前声明来提及别的类,并在实现文件中引入那些类的头文件。这样做可以尽量降低类之间的耦合(coupling)。

2.有时无法使用向前声明,比如要声明某个类遵循一项协议。这种情况下,尽量把“该类遵循的协议”的这条声明移至“class-continuation分类”中。如果不行的话,就把协议单独放在一个头文件中,然后将其引入。


三、多用字面量语法,少用与之等价的方法

1.何为字面量语法

在OC中初始化NSString对象一般用alloc及init方法,这种方法有点繁杂,因此从Objective-C 1.0的时候出现了非常简单的方式来创建NSString对象,就是“字符串字面量”,语法如下:

NSString *someString = @"Effective Objective-C 2.0";

这个格式跟C语言中字符串的初始化如出一辙。使用字面量语法可以缩减源代码长度,使其更为易读。

2.字面数值

传统的将数值包装成OC的NSNumber对象使用如下方法:

NSNumber *someNumber = [NSNumber numberWithInt:1];

使用字面量语法则是:

NSNumber *someNumber = @1;

并且能够以NSNumber实例表示的所有数据类型都可以使用字面量:

NSNumber *intNumber = @1;
NSNumber *floatNumber = @2.5f;
NSNumber *boolNumber = @YES;
NSNumber *charNumber = @'a';
//也适用下面的的表达式
int x = 5;
float y = 6.32f;
NSNumber *expressionNumber = @(x * y);

3.字面量数组

传统的创建数组和操作数组元素的方法如下:

NSArray *animals = [NSArray arrayWithObjects:@"cat", @"dog", @"mouse", @"badger", nil];

NSString *dog = [animals objectAtIndex:1];

而使用字面量语法来创建和操作数组元素的方式如下:

NSArray *animals = @[@"cat", @"dog", @"mouse", @"badger"];

NSString *dog = animals[1];

可以看到使用字面量语法更加简单并且可读性更强。还有一点,当数组中某个元素为nil时,使用传统方法创建的数组会依次处理各个参数直到遇到nil才停止,也就是不会报错但是数组个数为nil元素之前的个数。使用字面量语法则会抛出异常并且程序终止,这样程序的健壮性会更好。

4.字面量字典

传统的创建字典和根据键来取值的方法如下:

NSDictionary *personData = [NSDictionary dictionaryWithObjectivesAndKeys:@"Mett", @"firstName", @"Galloway", @"lastName", [NSNumber numberWithInt:28], @"age", nil];

NSString *lastName = [personData objectForKey:@:lastName"];

而使用字面量语法创建字典和根据键来取值的方法如下:

NSDictionary *personData = @{@"firstName": @"Matt", @"lastName": @"Galloway", @"age": @28};

NSString *lastName = personData[@"lastName"];

可以看到使用字面量语法更加简单并且更符合阅读习惯。和字面量创建数组一样,使用字面量语法创建字典遇到nil则会抛出异常并且程序终止。

5.可变数组和字典

通过取下标操作,可以访问数组中的某个元素或者字典中的某个键对应的元素,如果数组和字典是可变的(mutable),那么也能通过下标修改其中的元素值,例如:

mutableArray[1] = @"dog";
mutableDictionary[@"lastName"] = @"Galloway";

6.局限性

字面量语法有个小小的限制,就是除了字符串以外,所创建出来的对象必须属于Foundation框架才行。如果自定义了这些类的子类,则无法用字面量语法创建其对象。并且使用字面量语法创建出来的字符串、数组、字典对象都是不可变的,要变成可变对象则需要拷贝一份:

NSMutableArray *mutable = [@[@1, @2, @3, @4, @5] mutableCopy];

7.要点

1.应该使用字面量语法来创建字符串、数值、字典。与创建此类对象的常规方法相比,这么做更加简明扼要。

2.应该通过取下标操作来访问数组下标或者字典中的键对应的元素。

3.用字面量语法创建数组或者字典时,若值中有nil,则会抛出异常。因此,务必确保值里不含nil。

四、多用类型常量,少用#define预处理指令

1.类型常量与#define

通常在定义常量时会用到#define,例如#define ANIMATION_DURATION 0.3, 但是使用这种方法定义的常量不会包含类型信息并且还有可能在编译之后运行时被修改从而导致问题。因此还有更好的方法来进行替换,例如下面这个方法:

static const NSTimeInterval kAnimationDuration = 0.3;

这种方式定义的常量包含类型信息,其好处是清楚的描述了常量的含义。由此可知,该常量类型为NSTimeInterval,这有助于其编写开发文档。

常量命名规则:
如果常量局限在实现文件中则在常量前加k;如果常量在类之外可见,则通常将类名作为该常量的前缀。

常量定义的位置也非常重要,我们最好不要将常量定义在头文件中,若你定义在头文件中,又被其他的文件引用了,那么该这个文件中的这个常量都会被其替换掉,所以最好不要在头文件中定义常量。

2.static和const

变量一定要同时使用static和const来声明,如果试图修改由const修饰符所声明的变量,那么编译器就会报错。static修饰符则意味着该变量仅在定义此变量的编译单元中可见。假如声明此变量时不加static,则编译器会为它创建一个“外部符号”。此时若是另一个编译单元中也声明了同名变量,那么编译器就会抛出错误信息。

3.外界可见的常值变量

有时候我们需要对外公开某个常量,比如说,用一个对象派发通知,让其他想要接收通知的对象向该对象注册,派发通知的时候需要字符串表示通知名称,我们定义一个常量,外界就可以直接使用这个常值变量来注册自己想要接收的通知即可,而不用知道实际字符串的值。此类常量需放在“全局符号表”中,以便可以在定义该常量的编译单元之外使用。方式如下:

//In the header file
extern NSString *const EOCStringConstant;

//In the implementation file
NSString *const EOCStringConstant = @"VALUE";

这个常量在头文件中声明并且在实现文件中定义。 注意const在常量类型中的位置。因为常量定义应该从右向左解读,所以它是一个不可变的指向NSString *类型的指针,这个指针的指向不允许被改变。extern就是告诉编译器,在全局符号表中将会有一个名叫EOCStringConstant的符号,也就是说,编译器无需查看其定义,即允许代码使用此常量。因为符号要放在全局符号表里,所以我们就得更加注意其命名,为避免冲突,最好是用与之相关的类名做前缀。

4.要点

1.不能用预处理指令定义常量。这样定义出来的常量不含类型信息,编译器只是会在编译前据此执行查找与替换操作。即使有人重新定义了常量值,编译器也不会产生警告信息,这将导致应用程序中的常量值不一致。

2.在实现文件中使用static const来定义“只在编译单元内可见的常量”。由于此常量不在全局符号表中,所以无需为其名称加前缀。

3.在头文件中使用extern来声明全局变量,并在相关实现文件中定义其值。这种常量要出现在全局符号表中,所以其名称应该加以区隔,通常用与之相关的类名做前缀。

五、用枚举表示状态、选项、状态码

1.枚举表示状态

枚举是一种常量命名方式。某个对象所经历的各种状态就可以定义为一个简单的枚举集。例如用下列枚举表示“嵌套字连接”的状态:

enum EOCConnectionState {
    EOCConnectionStateDisConnected,
    EOCConnectionStateConnecting,
    EOCConnectionStateConnected,
};

编译器一般从0开始每个枚举加1来为枚举分配一个独有的编号,并且实现枚举所用的数据类型取决于编译器。定义枚举变量的方式通常如下:

enum EOCConnectionState = EOCConnectionStateDisConnected;

可以看到不是很简洁,因此为了操作起来更加方便可以使用typedef关键字重新定义枚举类型的方式:

typedef enum EOCConnectionState EOCConnectionState;

C++11标准修改后,可以指明用何种底层数据类型来保存枚举类型的变量,这样就可以向前声明枚举变量了。其语法如下:

enum EOCConnectionState : NSInteger {/*...*/};

上面这个枚举类型数据的底层数据类型就是NSInteger了,我们也可以使用向前声明的方法:

enum EOCConnectionState : NSInteger;

还可以收订指定某个枚举成员的所对应的值,那么在它之后的枚举类型的值就是在该值的基础上依次加一。

enum EOCConnectionState {
    EOCConnectionStateDisConnected = 1,
    EOCConnectionStateConnecting,
    EOCConnectionStateConnected,
};

2.枚举表示选项

使用枚举可以定义选项并且用“按位或操作符”可以组合多个选项,用“按位与操作符”可以判断是否启用某个选项。例如,iOS UI框架中有如下枚举类型,用来表示某个试图应该如何在水平或者垂直方向上调整方向:

enum UIViewAutoresizing {
    UIViewAutoresizingNone = 0,
    UIViewAutoresizingFlexibleLeftMargin = 1 << 0,
    UIViewAutoresizingFlexibleWidth = 1 << 1,
    UIViewAutoresizingFlexibleRightMargin = 1 << 2,
    UIViewAutoresizingFlexibleTopMargin = 1 << 3,
    UIViewAutoresizingFlexibleHeight = 1 << 4,
    UIViewAutoresizingFlexibleBottomMargin = 1 << 5,
};

上面操作代码如下:

enum UIViewAutoresizing resizing = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;

if (resizing & UIViewAutoresizingFlexibleWidth) {
    //UIViewAutoresizingFlexibleWidth is set
}

在这里插入图片描述

3.辅助宏

Foundation框架下定义了一些辅助宏来定义枚举类型从而指定用于保存枚举值的底层数据类型。这些宏在支持新语法的编译器上使用新式语法,否则是旧式语法。

4.枚举表示状态码

可以把逻辑含义相似的一组状态码放入一个枚举集中,而不需要用#define来定义。比如在创建UI元素的时候使用不同的样式,可以把这些样式声明为枚举类型。在switch语句里,枚举类型可以这样定义:

typedef NS_ENUM (NSUInteger, EOCConnectionState) {
    EOCConnectionStateDisConnected,
    EOCConnectionStateConnecting,
    EOCConnectionStateConnected,
};

switch (_currentState) {
    EOCConnectionStateDisConnected:
        //Handle disconnected state
        break;
    EOCConnectionStateConnecting:
        //Handle connecting state
        break;
    EOCConnectionStateConnected:
        //Handle connected state
        break;
}

通常我们喜欢在switch语句的最后加上default分支,但是定义状态码的话最好不要用。因为你本就是枚举变量,使用default分支如果用户没有使用枚举中的类型,那么这个switch语句也会进入default分支,会导致程序出现问题。

5.要点

1.应该用枚举来表示状态机的状态、传递给方法的选项以及状态码等值,给这些值起个易懂的名字。

2.如果把传递给某个方法的选项表示为枚举类型,而多个选项又可同时使用,那么就将各选项值定义为2的幂,以便通过按位或操作将其组合起来。

3.用NS_ENUM与NS_OPTIONS宏来定义枚举类型,并指明其底层数据类型。这样做可以确保枚举是用开发者所选的底层数据类型实现出来的,而不会采用编译器所选的类型。

4.在处理枚举类型的switch语句中不要实现default分支。这样的话,加入新枚举之后,编译器就会提示开发者:switch语句并未处理所有枚举。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1390638.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

chrome浏览器开启硬件加速无法打开提示“此设置有你的管理员管理“

chrome浏览器开启硬件加速无法打开提示"此设置有你的管理员管理" winR 输入regedit 打开注册表注册表搜索 计算机\HKEY_LOCAL_MACHINE\SOFTWARE\Policies\Google\Chrome删除 HardwareAccelerationModeEnabled重启Chrome浏览器。打开Chrome浏览器&#xff0c;查看设置…

【Java数据结构 -- 实现双链表的接口方法】

双链表 1.双链表2.双链表的创建3.双链表的头插节点4.双链表尾插5.双链表根据索引找节点6.双链表根据索引插入节点7.双链表删除值为key的节点8.删除所有值为key的节点9.双链表是否包含值为key节点10.双链表大小11.清空双链表12.打印双链表 1.双链表 双链表是一种数据结构&#…

【MATLAB】 SSA奇异谱分析信号分解算法

有意向获取代码&#xff0c;请转文末观看代码获取方式~ 1 基本定义 SSA奇异谱分析&#xff08;Singular Spectrum Analysis&#xff09;是一种处理非线性时间序列数据的方法&#xff0c;可以对时间序列进行分析和预测。 它基于构造在时间序列上的特定矩阵的奇异值分解&#…

部署YUM仓库及NFS共享存储

引言&#xff1a; 学习YUM 软件仓库&#xff0c;可以完成安装、卸载、自动升级 rpm 软件包等任务&#xff0c;能够自动 查找并解决 rpm 包之间的依赖关系&#xff0c;而无须管理员逐个、手工地去安装每个 rpm 包&#xff0c;使管理员在维护大量 Linux 服务器时更加轻松自如。特…

20240116-【UNITY 学习】增加滑动功能

替换脚本PlayerMovement_02.cs using System.Collections; using System.Collections.Generic; using UnityEngine;public class PlayerMovement_03 : MonoBehaviour {private float moveSpeed; // 玩家移动速度public float walkSpeed 7; // 行走速度public float sprintSpee…

竞赛保研 基于深度学习的水果识别 设计 开题 技术

1 前言 Hi&#xff0c;大家好&#xff0c;这里是丹成学长&#xff0c;今天做一个 基于深度学习的水果识别demo 这是一个较为新颖的竞赛课题方向&#xff0c;学长非常推荐&#xff01; &#x1f9ff; 更多资料, 项目分享&#xff1a; https://gitee.com/dancheng-senior/pos…

图像处理:孤立点的检测

图像处理-孤立点的检测 孤立点的检测在图像处理中通常涉及到检测图像中的突变或者边缘&#xff0c;而使用二阶导数是一种常见的方法。一阶导数可以帮助找到图像中的边缘&#xff0c;而二阶导数则有助于检测边缘上的峰值&#xff0c;这些峰值可能对应于孤立点或者特殊的图像结构…

2024美赛数学建模思路 - 案例:FPTree-频繁模式树算法

文章目录 算法介绍FP树表示法构建FP树实现代码 建模资料 ## 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 算法介绍 FP-Tree算法全称是FrequentPattern Tree算法&#xff0c;就是频繁模式树算法&#xff0c…

帆软笔记-决策表报对象使用(两表格联动)

效果描述如下&#xff1a; 数据库中有个聚合商表&#xff0c;和一个储能表&#xff0c;储能属于聚合商&#xff0c;桩表中有个字段是所属聚合商。 要求帆软有2个表格&#xff0c;点击某个聚合商&#xff0c;展示指定的储能数据。 操作&#xff1a; 帆软选中表格单元&#xf…

Windows Server 2019配置DNS服务器

正文共&#xff1a;1234 字 31 图&#xff0c;预估阅读时间&#xff1a;1 分钟 我们在给Windows Server添加角色和功能时&#xff0c;会发现有一项“远程桌面服务安装”&#xff0c;它的介绍为“为虚拟桌面基础结构&#xff08;Virtual Desktop Infrastructure&#xff0c;VDI&…

PyTorch Tutorial 2.0

这里是对于PyTorch Tutorial-CSDN博客的补充&#xff0c;但是与其相关的NLP内容无关&#xff0c;只是一些基础的PyTorch用法的记录&#xff0c;主要目的是能够自己生成一些模拟的数据集。先介绍随机数的目的是因为based on随机数方法。 当然在看随机数的方法的时候&#xff0c…

Python 最新版本 3.12.1 环境配置(windows)

文章目录 python 3.12.1环境安装3.12.1 网盘下载3.12.1 官网下载 python 安装完成测试第一个 python 程序Hello Python python 3.12.1环境安装 3.12.1 网盘下载 python 3.12.1 百度网盘地址&#xff1a;https://pan.baidu.com/s/1SAcH_uH0T3DiERn6AZeQlg?pwd4242 提取码&a…

java-Lambda 语法总结

文章目录 Lambda 语法概览Lambda 表达式语法1.Lambda 表达式与函数接口2.Lambda 遇上 this final Lambda 语法概览 String(] names {”Justi n ”,”caterpillar”,”Bush " }; Arrays . sort (names, new Compara tor<String> () { publ int compare (String na…

伪装目标检测模型论文阅读之:Zoom in and out

论文链接&#xff1a;https://arxiv.org/abs/2203.02688 代码;https://github.com/lartpang/zoomnet 1.摘要 最近提出的遮挡对象检测&#xff08;COD&#xff09;试图分割视觉上与其周围环境融合的对象&#xff0c;这在现实场景中是非常复杂和困难的。除了与它们的背景具有高…

布隆过滤器四种实现(Java,Guava,hutool,Redisson)

1.背景 为预防大量黑客故意发起非法的时间查询请求&#xff0c;造成缓存击穿&#xff0c;建议采用布隆过滤器的方法解决。布隆过滤器通过一个很长的二进制向量和一系列随机映射函数&#xff08;哈希函数&#xff09;来记录与识别某个数据是否在一个集合中。如果数据不在集合中…

两个方法实现echarts散点图的高光圆点

一、效果图&#xff1a; 二、代码 方法一&#xff1a;通过series的itemStyle进行设置&#xff0c;type为scatter 在 ECharts 中&#xff0c;要在二维散点图上实现看似 3D 的高光圆点效果&#xff0c;可以通过自定义散点图的 itemStyle 属性来实现。虽然无法直接创建真正的 3D…

Flume 之自定义 Source

1、简介 Flume 自带 Source 有 Avro、Thrift、Netcat、Taildir、Kafka、Http等&#xff0c;有些场合比如我们指定访问接口获取数据当做 Flume 的 Source&#xff0c;像这种定制化的 Source 需要我们自己实现&#xff0c;下面我将介绍如何自定义实现 Source。 2、自定义实现 Fl…

Linux中放大字体

环境&#xff1a;VMware17Pro&#xff0c;Ubuntu22.04 在显示设置外观中只看到图标放大的调整&#xff0c;没看到字体大小设置 不按照常规设置&#xff0c;点开下面的辅助功能->大号文本&#xff08;没有设置具体字号的选项&#xff0c;但是可以放大&#xff09; 效果图如下…

五、带登录窗体的demo

做了一个简单的带登录窗体的demo&#xff0c;有用户名和密码不能为空的验证&#xff0c;原理是在main.cpp的主函数入口处&#xff1a; 1、将默认的MainWindow主窗体注释。 2、新建一个formlogin登录窗体&#xff0c;在主函数中先运行登录窗体。 3、在登录窗体中引用MainWind…

Monorepo-uniapp 构建分享

Monorepo uniapp 构建灵感&#xff1a;刚好要做一个项目&#xff0c;于是想到升级一下之前自己写的一个vue3tspiniauno的模版框架&#xff0c;其实那个框架也不错&#xff1b;只是感觉还差点东西&#xff0c;我已经用那个小框架写了两三个项目&#xff1b;轻巧实用。为什么选…