Objective-C 学习笔记 | 基础

Objective-C 学习笔记 | 基础

参考书:《Objective-C 编程(第2版)》

第1部分 入门

Objective-C语言是以C语言为基础的,但增加了对面向对象编程的支持。Objective-C语言是用来开发在苹果iOS以及OS X操作系统上运行的应用的编程语言。

第2部分 如何编程

该部分讲解了C语言编程的必要知识,这里只记录Objective-C新增内容。

NSInteger和NSUInteger

NSUInteger是无符号的整型,NSInteger是有符号的整型。

NSInteger是一个封装,它会识别当前操作系统的位数,自动返回最大的类型。
定义的代码类似于下:

#if __LP64__ || TARGET_OS_EMBEDDED || TARGET_OS_IPHONE || TARGET_OS_WIN32 || NS_BUILD_32_LIKE_64
typedef long NSInteger;
typedef unsigned long NSUInteger;
#else
typedef int NSInteger;
typedef unsigned int NSUInteger;
#endif

这样做让NSInteger和NSUInteger变得通用,不用考虑设备是32位还是64位。

如果使用printf()来输出这两种类型的变量,需要先将NSInteger转换成long,NSUInteger转换成unsigned long。

第3部分 Objective-C与Foundation

该部分开始介绍Objective-C。编写Objective-C程序时,要使用Foundation框架。框架(framework)是由很多类(Class)组成的库。这里记录一些新的知识点。

#import和#include的区别

#include指令告诉编译器做呆板的复制粘贴,将包含的内容粘贴到目标文件中来。而#import指令则让编辑器先检查之前是否导入过这个文件,或是已经被包含到目标文件中了。因此,#import指令导入更快、更有效率。

类与实例、方法与消息

Objective-C也有类和对象的概念,Objective-C的方法和消息与函数类似。如需执行方法中的代码,首先需要发送一条消息给包含这个方法的对象或类。消息发送(指令)必须写在一对方括号中,并且必须包含接收方(receiver)和选择器(selector),如下图所示:

请添加图片描述

date是一个类方法。date方法执行后,NSDate类会在堆上new一个NSDate实例:now,并初始化为当前的日期/时间,然后返回新对象(now)的地址。

请添加图片描述

有了NSDate实例:now之后,我们可以给这个对象发一个实例方法,比如:timeIntervalSince1970。通常来说,实例方法会提供实例中实例变量的信息,或是对实例的实例变量进行操作。

测试程序:

//
//  main.m
//  TimeAfterTime
//
//  Created by 刘文晨 on 2024/6/5.
//#import <Foundation/Foundation.h>int main(int argc, const char * argv[]) {@autoreleasepool {// 向 NSDate 类发送 date 消息,让它执行 date 方法NSDate *now = [NSDate date];// 打印实例的地址NSLog(@"This NSDate object lives at %p", now);// %@ 会输出相应对象的“描述信息”NSLog(@"The date is %@", now);// 实例方法double seconds = [now timeIntervalSince1970];NSLog(@"It has benen %f seconds since the start of 1970.", seconds);}return 0;
}

注意:

  1. 类方法和类对应,实例方法和实例对应。receiver和selector不匹配就会出错。
  2. Objective-C是区分大小写的,方法名也是区分大小写的。
  3. Objective-C语言命名习惯为“驼峰式”或“前缀大写的驼峰式”。

alloc和init

消息可以以嵌套的形式连续发送,而唯一必须要以嵌套的形式发送的消息是alloc和init。

每个类都有一个alloc类方法,它创建一个新的对象并返回指向该对象的指针。通过alloc类方法创建出来的对象,必须要经过初始化才能使用。每个类也都有一个init实例方法,它用来初始化实例。

NSDate *now = [[NSDate alloc] init]; // 消息嵌套,和下面的效果一样
NSDate *now = [NSDate date]; // date 方法代码最少,称之为便利方法

nil

nil就是空指针,不指向任何对象。

在Objective-C中,可以向nil发送消息,是合法的,但得到的返回值没有意义。

注意,如果程序向某个对象发送了消息,但不符合预期,应该检查receiver是否为nil。

id

id 类型的含义是:可以指向任意类型的Objective-C对象的指针。‘

id delegate; // id 已经隐含了 * 的作用

类似于 auto(?)

ARC

ARC(automatic reference counting,自动引用计数)为Objective-C提供了一种自动销毁不被引用的对象的机制。当项目开启了ARC,编译器会自动给项目添加代码来计算每个对象的引用数,即每个对象都会对指向自己的指针计数。当引用数为0时, 程序会自动销毁该对象,释放内存。

类似于 shared_ptr。

在Objective-C加入ARC之前,程序员必须手动维护引用计数。

[anObject release]; // abOject 会失去一个拥有方
[anObject retain]; // abOject 会得到一个拥有方
[anObject autorelease]; // abOject 会在 autorelease 池(对象)被排干(drain)的时候收到 release 消息

虽然 ARC 会自动使用 autorelease 池,但是必须由程序创建并排空相应的 autoreleasepool 池,语法如下:

// 创建 autorelease 池(对象)
@autoreleasepool {...
} // autorelease 池被排空

字面量语法

格式:

NSString *lament = @"Hello World!";
NSArray *dataList = @[now, tomorrow, yesterday]; // 这三个都是 NSDate 对象

字面量语法是Objective-C语言的一种缩写,可以以不明确发送消息的方式创建实例。

还没有字面量语法的时候,只能使用 arrayWithObjects: 类方法来创建NSArray实例:

NSArray *dataList = [NSDate arrayWithObjects:now, tomorrow, yesterday, nil]; // nil 是结束标记,让方法停止运行

self

Objective-C的方法都包含一个隐含的局部变量 self。self 是指针,指向运行当前方法的对象。当某个对象要向自己发送消息时,就需要使用 self。它有 2 个简单的用法:

  1. 调用自身的存取方法,避免直接存取实例变量。
  2. 将 self 作为实参传给其他方法,以便其访问“当前的”对象。

NSObject 协议中的 self:

#include <objc/objc.h>
#include <objc/NSObjCRuntime.h>@class NSString, NSMethodSignature, NSInvocation;
@protocol NSObject- (BOOL)isEqual:(id)object;
@property (readonly) NSUInteger hash;
@property (readonly) Class superclass;
......
- (Class)class OBJC_SWIFT_UNAVAILABLE("use 'anObject.dynamicType' instead");
- (instancetype)self;
......
@end

self最终返回的结果就是instancetype类型的代理方法,它是动态类型,最终运行时才会确定,实例方法返回实例类型、静态方法返回的是Class。

super

super 的作用:直接调用父类中的某个方法,或者说从父类开始查找与之匹配的实现。

使用场合:子类重写父类的方法时想保留父类的一些行为。

结构体 objc_super 的官方解释:The compiler generates an objc_super data structure when it encounters the super keyword as the receiver of a message. It specifies the class definition of the particular superclass that should be messaged.

#include <objc/objc.h>
#include <objc/runtime.h>#pragma GCC system_header#ifndef OBJC_SUPER
#define OBJC_SUPER/// Specifies the superclass of an instance. 
struct objc_super {/// Specifies an instance of a class.__unsafe_unretained id receiver;/// Specifies the particular superclass of the instance to message. 
#if !defined(__cplusplus)  &&  !__OBJC2____unsafe_unretained Class class;
#else__unsafe_unretained Class super_class;
#endif
};

当遇到super关键字时,编译器会生成一个objc_super结构体,作为消息的接收者,objc_super结构体使得接收消息的父类的定义被明确化。

isa指针

任何一个对象的isa指针都会指向创建该对象的类。

给对象发送消息的时候,对象就会通过isa指针找到该对象的类并查询是否有该消息名的方法。如果没有,就会继续查询它的父类,直到找到名为消息名的方法,或者到达继承链的顶端(NSObject)为止。

description方法和%@转换说明

格式说明符%@让对象描述自己,实际上在处理%@时,程序会向相应的指针变量所指的对象发送 description 消息。

description 方法会返回一个描述类实例的字符串。description 是一个 NSObject 方法,所以所有的对象都有这个方法。默认的 NSObject 实现会以字符串的形式返回该对象在内存的地址。不同的类可以重写(override)description 方法,来最有效地描述实例。

强引用循环和弱引用

两个对象互相拥有的关系将导致相关对象都无法释放,这种情况叫强引用循环,这是导致内存泄露的常见原因。

请添加图片描述

Xcode的Instruments中的Leaks组件可以找出程序中的强引用循环:

请添加图片描述

通过弱引用,可以解决该问题。弱引用是不说明所有权的指针,把 BNRAsset 的 holder 属性改成 weak,就能让 BNRAsset 对象不拥有它的 holder(BNREmployee 对象)。代码如下:

#import <Foundation/Foundation.h>
@class BNREmployee;NS_ASSUME_NONNULL_BEGIN@interface BNRAsset : NSObject@property (nonatomic, copy) NSString *label;
@property (nonatomic, weak) BNREmployee *holder;
@property (nonatomic) unsigned int resaleValue;@endNS_ASSUME_NONNULL_END

完整程序见于:UestcXiye/Objective-C-Practice。

弱引用的自动置零特性:强引用会保留对象的拥有方,使其不被释放。而弱引用则不会保留,因此标为弱引用的实例变量与属性指向的对象可能会消失,如果发生了这种情况,那么这个实例变量或属性会被设为 nil。

Collection 类

Collection 类的实例用于保存指向其他对象的指针。主要分为三种:

  1. NSArray 及其子类 NSMutableArray
  2. NSSet/NSMutableSet
  3. NSDictionary/NSMutableDictionary

注意以下 4 点:

  1. Collection 对象只能保存对象的指针,不能保存基本类型变量或指向结构的指针,需要先将这些 C 语言基本类型封装成对象,再存入 Collection 对象。

    例如:float变量、int变量等要先转换成NSNumber再存入Collection对象,或者直接用NSNumber字面量实例。结构可以用NSValue(它是NSNumber的父类)实例来封装。

  2. 其中 NSArray、NSSet、NSDictionary 具有不可修改性。使用它们可以节约内存提高性能,因为它们的 copy 方法仅仅返回指向自己的指针,不会做拷贝之类的事情。

  3. 向可改变的 Collection 对象中加入某个对象时,Collection 对象会成为该对象的拥有方;同理,移除对象时,Collection 对象就不再是该对象的拥有方了。对于不可改变的 Collection 对象而言,创建时就拥有其中所有对象的所有权,而 Collection 对象被释放时,它就放弃其中所有对象的所有权。

  4. Collection 对象不能保存 nil,需要将“空”包装成一个对象,可以使用 NSNull 类。

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

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

相关文章

植物大战僵尸杂交版 2.0 下载及配置图文教程

文章目录 Part.I IntroductionPart.II 下载Chap.I 下载地址Chap.II 网盘直链下载 Part.III 配置Chap.I 解压与安装Chap.II 加载存档Chap.III 其他设置 Reference Part.I Introduction 最近看大仙儿直播植物大战僵尸&#xff0c;觉得挺好玩的。它大概长这样&#xff1a; 就上网…

使用proteus仿真51单片机的流水灯实现

proteus介绍&#xff1a; proteus是一个十分便捷的用于电路仿真的软件&#xff0c;可以用于实现电路的设计、仿真、调试等。并且可以在对应的代码编辑区域&#xff0c;使用代码实现电路功能的仿真。 汇编语言介绍&#xff1a; 百度百科介绍如下&#xff1a; 汇编语言是培养…

GPT-4o:OpenAI的最新篇章与深度探索

引言&#xff1a; 在人工智能领域&#xff0c;自然语言处理&#xff08;NLP&#xff09;技术持续引领着技术创新的步伐。自2023年引入以来&#xff0c;GPT系列模型一直以其卓越的语言生成能力而闻名&#xff0c;近期的迭代——GPT-4o&#xff0c;更是为这一领域的研究和应用带…

指令调度和延迟分支

指令调度和延迟分支是计算机体系结构中的两个重要概念,特别是在处理CPU效率优化方面。以下是关于这两个概念的详细解释: 指令调度(Instruction Scheduling) 定义: 指令调度是指对程序块或过程中的操作进行排序以有效利用处理器资源的任务。其目的是通过重排指令,提高指…

Python GUI编程:深入探索现代GUI库及其创新应用

目录 引言 Python GUI库概览 1. Tkinter 2. PyQt/PySide 3. wxPython 4. Kivy 5. PyGTK 6.FLTK (pyFLTK) 创新应用案例 1. 交互式数据分析工具 2. 智能物联网(IoT)仪表板 3. 增强现实(AR)辅助设计软件 4. 跨平台的科学计算软件 5. 交互式教育软件 实战示例1&…

Outlook设置邮箱签名

设置Outlook邮箱签名的步骤可以根据你所使用的Outlook版本&#xff08;如Windows、Mac或网页版&#xff09;有所不同。以下是针对不同版本Outlook设置签名的详细步骤&#xff1a; Windows版Outlook 启动Outlook&#xff1a;打开Windows中的Outlook应用程序。进入签名设置&…

Hive 面试题(八)

1. 简述Hive的三种自定义函数是什么&#xff1f;实现步骤与流程&#xff1f;它们之间的区别&#xff1f;作用是什么 &#xff1f; Hive提供了三种类型的自定义函数&#xff08;UDF&#xff09;&#xff1a;用户定义的标量函数&#xff08;UDF&#xff09;、用户定义的聚合函数…

.NET MAUI 了解MVVM

MVVM 模式中有三个核心组件&#xff1a;模型、视图和视图模型。 每个组件的用途不同。 下图显示了这三个组件之间的关系。 视图 视图负责定义用户在屏幕上看到的结构、布局和外观。 理想情况下&#xff0c;每个视图在 XAML 中定义&#xff0c;代码隐藏有限&#xff0c;不包含业…

linux shell实现打印国际象棋棋盘

chess.sh #!/bin/bashfor i in {1..8} dofor j in {1..8}dosum$[ij]if [ $[sum%2] -eq 0 ];thenecho -ne "\033[46m \033[0m"elseecho -ne "\033[47m \033[0m"fidoneecho done验证&#xff1a;

微信小程序学习笔记(4)

文章目录 1、< template >< / template >2、样式导入i、wxmlii、wxss 3、flex布局i、容器属性ii、项目属性 1、< template >< / template > 模板可以重复调用 首先要定义一个模板&#xff1a; <template name"test"><view>{{…

深入了解Git:从数据模型到集成IDEA

Git是现代软件开发中不可或缺的版本控制工具。理解Git的数据模型、暂存区、命令行接口&#xff0c;并将其集成到IDE&#xff08;如IntelliJ IDEA&#xff09;&#xff0c;可以显著提升开发效率。本文将从底层开始&#xff0c;逐步深入Git的各个方面&#xff0c;并介绍如何将其集…

AbstractMap和SimpleEntry

一、AbstractMap 位置&#xff1a;在java.util包 二、SimpleEntry 1、概述 继承了Map中的内部接口Entry<K,V> SimpleEntry<K,V>不仅继承了Map.Entry<K,V>&#xff0c;还继承了序列化的接口 2、构造方法 方法说明SimpleEntry(K key,V value)通过键值对初…

RabbitMQ-工作模式(Publish模式Routing模式)

文章目录 发布/订阅&#xff08;Publish/Subscribe&#xff09;交换机临时队列绑定总体代码示例 路由&#xff08;Routing&#xff09;绑定直连交换机多重绑定发送日志订阅总体代码示例 更多相关内容可查看 发布/订阅&#xff08;Publish/Subscribe&#xff09; 构建一个简单的…

(delphi11最新学习资料) Object Pascal 学习笔记---第14章泛型第3节(接口引用 vs 泛型接口约束)

14.3.4 接口引用 vs 泛型接口约束 ​ 在上一个例子中&#xff0c;我定义了一个泛型类&#xff0c;可以与实现特定接口的任何对象一起使用。我也可以通过创建基于接口引用的标准&#xff08;非泛型&#xff09;类来获得类似的效果。实际上&#xff0c;我可以定义一个类&#xf…

前端构建工具总结

前端构建工具是前端开发中的重要组成部分&#xff0c;它们能够帮助开发者自动化地处理、优化和打包前端资源&#xff0c;提高开发效率和项目的可维护性。以下是对一些常用前端构建工具的总结&#xff1a; Webpack 功能&#xff1a;Webpack是一个强大的前端构建工具&#xff0c…

[EFI]ASUS Vivobook 16x M1603QA 电脑 Hackintosh 黑苹果efi引导文件

黑果魏叔提供硬件型号驱动情况 主板ASUS Vivobook 16x M1603QA 处理器AMD Ryzen 5 5600H已驱动 内存8GB DDR4 on board 8GB DDR4 SO-DIMM已驱动 硬盘SSD INTEL 512GB 670P M.2 SSDPEKNU512GZX1 PCIe 3.0 x4 NVMe已驱动 显卡AMD Radeon RX Vega 7已驱动 声卡瑞昱 AMD Hi…

vue antdesgin table 动态表头动态数据示例

以下是一个基于 Vue 和 Ant Design Vue 的示例&#xff0c;可以动态生成表格的表头和数据&#xff1a; <template><div><a-button click"addColumn">添加列</a-button><a-table :columns"columns" :dataSource"dataSource…

HC-05蓝牙模块配置连接和使用

文章目录 1. 前期准备 2. 进入AT模式 3. 电脑串口配置 4. 配置过程 5. 主从机蓝牙连接 6. 蓝牙模块HC-05和电脑连接 1. 前期准备 首先需要准备一个USB转TTL连接器&#xff0c;电脑安装一个串口助手&#xff0c;然后按照下面的连接方式将其相连。 VCCVCCGNDGNDRXDTXDTXD…

ICLR24大模型提示(8) | 退一步思考:在大型语言模型中通过抽象引发推理

【摘要】我们提出了一种简单的提示技术&#xff0c;即后退提示法&#xff0c;它使 LLM 能够进行抽象&#xff0c;从包含特定细节的实例中得出高级概念和第一原理。通过使用概念和原理来指导推理&#xff0c;LLM 显著提高了遵循正确推理路径解决问题的能力。我们使用 PaLM-2L、G…

Facebook企业户 | Facebook公共主页经营

Facebook作为社交媒体巨头&#xff0c;拥有庞大的用户基数&#xff0c;因此&#xff0c;有效经营公共主页是获取持续流量、提升客户信任度和粘性、促进产品或服务销售与转化的关键。要优化Facebook主页&#xff0c;关注以下几点&#xff1a; 1、参与度是关键指标&#xff1a;因…