-

「Generated by Manus, instructions issued by binbinwang」

本章将深入对比Kotlin和Objective-C这两种语言,帮助iOS开发者快速理解Kotlin的特性并建立知识迁移。我们将聚焦于语法差异、编程范式和各自的优势特性,以便你能够更顺畅地从Objective-C过渡到Kotlin。

2.1 语法基础对比

变量与常量声明

Kotlin:

1
2
3
4
5
6
7
8
9
10
11
12
// 变量声明(可修改)
var name: String = "Android"
var age: Int = 30
// 类型推断
var message = "Hello" // 自动推断为String类型

// 常量声明(不可修改)
val PI: Double = 3.14159
val MAX_COUNT = 100 // 自动推断为Int类型

// 空安全类型
var nullableName: String? = null

Objective-C:

1
2
3
4
5
6
7
8
9
10
// 变量声明
NSString *name = @"iOS";
int age = 30;

// 常量声明
const double PI = 3.14159;
static NSString * const MAX_NAME = @"Max";

// 空值处理
NSString *nullableName = nil;

主要差异:

  1. Kotlin使用varval关键字区分变量和常量,而Objective-C使用const标记常量
  2. Kotlin的类型声明在变量名后,而Objective-C在变量名前
  3. Kotlin有内置的空安全机制(通过?标记),而Objective-C使用nil
  4. Kotlin支持类型推断,而Objective-C通常需要显式声明类型
  5. Objective-C的字符串需要@前缀,而Kotlin直接使用双引号

基本数据类型

Kotlin基本类型:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
// 整数类型
val byte: Byte = 1 // 8位
val short: Short = 1 // 16位
val int: Int = 1 // 32位
val long: Long = 1L // 64位

// 浮点类型
val float: Float = 1.0f // 32位
val double: Double = 1.0 // 64位

// 布尔类型
val isTrue: Boolean = true

// 字符类型
val char: Char = 'A'

// 字符串
val str: String = "Hello"

Objective-C基本类型:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
// 整数类型
char byteValue = 1; // 8位
short shortValue = 1; // 16位
int intValue = 1; // 32位
long longValue = 1L; // 32/64位
long long longLongValue = 1LL; // 64位

// 浮点类型
float floatValue = 1.0f; // 32位
double doubleValue = 1.0; // 64位

// 布尔类型
BOOL isTrue = YES; // Objective-C特有的YES/NO

// 字符类型
char charValue = 'A';

// 字符串
NSString *str = @"Hello";

主要差异:

  1. Kotlin的基本类型都是对象,没有原始类型,而Objective-C区分原始类型和对象类型
  2. Kotlin的布尔值使用true/false,而Objective-C使用YES/NO
  3. Kotlin的字符串是不可变的,而Objective-C有NSString(不可变)和NSMutableString(可变)
  4. Kotlin的数字类型更加明确,而Objective-C的长整型在不同平台上可能有不同的大小

运算符与表达式

Kotlin:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
// 算术运算符
val sum = 5 + 3
val diff = 5 - 3
val product = 5 * 3
val quotient = 5 / 3 // 整数除法,结果为1
val remainder = 5 % 3
val floatQuotient = 5.0 / 3.0 // 浮点除法,结果为1.6666...

// 比较运算符
val isEqual = (5 == 3) // false
val isNotEqual = (5 != 3) // true
val isGreater = (5 > 3) // true

// 逻辑运算符
val andResult = true && false // false
val orResult = true || false // true
val notResult = !true // false

// 位运算符
val bitwiseAnd = 0b1010 and 0b1100 // 8 (0b1000)
val bitwiseOr = 0b1010 or 0b1100 // 14 (0b1110)
val bitwiseXor = 0b1010 xor 0b1100 // 6 (0b0110)
val bitwiseNot = 0b1010.inv() // -11 (取反)

// 字符串模板
val name = "Kotlin"
val greeting = "Hello, $name!" // "Hello, Kotlin!"
val calculation = "Sum: ${5 + 3}" // "Sum: 8"

Objective-C:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
// 算术运算符
int sum = 5 + 3;
int diff = 5 - 3;
int product = 5 * 3;
int quotient = 5 / 3; // 整数除法,结果为1
int remainder = 5 % 3;
float floatQuotient = 5.0 / 3.0; // 浮点除法,结果为1.6666...

// 比较运算符
BOOL isEqual = (5 == 3); // NO
BOOL isNotEqual = (5 != 3); // YES
BOOL isGreater = (5 > 3); // YES

// 逻辑运算符
BOOL andResult = (YES && NO); // NO
BOOL orResult = (YES || NO); // YES
BOOL notResult = !YES; // NO

// 位运算符
int bitwiseAnd = 0b1010 & 0b1100; // 8 (0b1000)
int bitwiseOr = 0b1010 | 0b1100; // 14 (0b1110)
int bitwiseXor = 0b1010 ^ 0b1100; // 6 (0b0110)
int bitwiseNot = ~0b1010; // 取反

// 字符串格式化
NSString *name = @"Objective-C";
NSString *greeting = [NSString stringWithFormat:@"Hello, %@!", name];
NSString *calculation = [NSString stringWithFormat:@"Sum: %d", 5 + 3];

主要差异:

  1. Kotlin使用命名的位运算符(and, or, xor, inv等),而Objective-C使用符号(&, |, ^, ~)
  2. Kotlin提供了字符串模板功能,而Objective-C使用格式化字符串
  3. Kotlin的字符串连接使用+运算符,而Objective-C通常使用stringWithFormat:stringByAppendingString:
  4. Kotlin支持中缀表达式和运算符重载,而Objective-C不支持

2.2 函数与方法

Kotlin函数定义与特性

基本函数定义:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
// 基本函数
fun greet(name: String): String {
return "Hello, $name!"
}

// 单表达式函数
fun add(a: Int, b: Int): Int = a + b

// 默认参数
fun greetWithTitle(name: String, title: String = "Mr."): String {
return "Hello, $title $name!"
}

// 命名参数
val greeting = greetWithTitle(title = "Dr.", name = "Smith")

// 可变参数
fun sum(vararg numbers: Int): Int {
return numbers.sum()
}
val total = sum(1, 2, 3, 4, 5) // 15

// 局部函数
fun processData(data: String): String {
fun validate(input: String): Boolean {
return input.isNotEmpty()
}

if (!validate(data)) {
return "Invalid data"
}
return "Processed: $data"
}

Objective-C方法定义与特性

基本方法定义:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
// 实例方法
- (NSString *)greet:(NSString *)name {
return [NSString stringWithFormat:@"Hello, %@!", name];
}

// 类方法
+ (int)add:(int)a to:(int)b {
return a + b;
}

// 多参数方法
- (NSString *)greetWithTitle:(NSString *)title name:(NSString *)name {
return [NSString stringWithFormat:@"Hello, %@ %@!", title, name];
}

// 调用方法
NSString *greeting = [self greetWithTitle:@"Dr." name:@"Smith"];

// 可变参数
- (int)sumNumbers:(int)firstNumber, ... {
va_list args;
va_start(args, firstNumber);

int sum = firstNumber;
int value;
while ((value = va_arg(args, int)) != 0) {
sum += value;
}

va_end(args);
return sum;
}
// 调用可变参数方法(需要以0结尾)
int total = [self sumNumbers:1, 2, 3, 4, 5, 0];

函数式编程特性对比

Kotlin的函数式特性:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
// 函数类型
val adder: (Int, Int) -> Int = { a, b -> a + b }

// 高阶函数(接收函数作为参数)
fun calculate(a: Int, b: Int, operation: (Int, Int) -> Int): Int {
return operation(a, b)
}
val result = calculate(5, 3, adder) // 8

// Lambda表达式
val subtractor = { a: Int, b: Int -> a - b }
val subResult = calculate(5, 3, subtractor) // 2
// 或者直接传递lambda
val mulResult = calculate(5, 3) { a, b -> a * b } // 15

// 函数引用
fun multiply(a: Int, b: Int): Int = a * b
val mulResult2 = calculate(5, 3, ::multiply) // 15

// 闭包
fun createCounter(): () -> Int {
var count = 0
return { count++ }
}
val counter = createCounter()
println(counter()) // 0
println(counter()) // 1

Objective-C的函数式特性:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
// 块(Block)
typedef int (^MathOperation)(int, int);
MathOperation adder = ^(int a, int b) {
return a + b;
};

// 接收块作为参数的方法
- (int)calculate:(int)a with:(int)b using:(MathOperation)operation {
return operation(a, b);
}
int result = [self calculate:5 with:3 using:adder]; // 8

// 内联块
MathOperation subtractor = ^(int a, int b) {
return a - b;
};
int subResult = [self calculate:5 with:3 using:subtractor]; // 2

// 闭包
- (void (^)(void))createCounter {
__block int count = 0;
return ^{
NSLog(@"%d", count++);
};
}
void (^counter)(void) = [self createCounter];
counter(); // 输出: 0
counter(); // 输出: 1

主要差异:

  1. Kotlin的函数是一等公民,可以作为变量、参数和返回值,而Objective-C使用块(Block)实现类似功能
  2. Kotlin支持命名参数和默认参数,而Objective-C通过方法名的一部分来命名参数
  3. Kotlin的Lambda表达式语法更简洁,而Objective-C的块语法较为复杂
  4. Kotlin支持函数引用(::),而Objective-C没有直接等价物
  5. Kotlin的闭包自动捕获外部变量,而Objective-C需要使用__block修饰符
  6. Kotlin支持局部函数,而Objective-C不支持嵌套方法定义

2.3 面向对象编程

类与对象

Kotlin:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
// 类定义
class Person(val name: String, var age: Int) {
// 属性
var email: String? = null

// 次构造函数
constructor(name: String, age: Int, email: String) : this(name, age) {
this.email = email
}

// 方法
fun introduce(): String {
return "I'm $name, $age years old"
}

// 伴生对象(类似于静态成员)
companion object {
const val SPECIES = "Human"

fun createAnonymous(): Person {
return Person("Anonymous", 0)
}
}
}

// 创建对象
val person = Person("Alice", 30)
person.age = 31 // 可变属性
val intro = person.introduce()
val anonymous = Person.createAnonymous()

Objective-C:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
// 类声明 (.h文件)
@interface Person : NSObject

@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@property (nonatomic, copy, nullable) NSString *email;

- (instancetype)initWithName:(NSString *)name age:(int)age;
- (instancetype)initWithName:(NSString *)name age:(int)age email:(NSString *)email;
- (NSString *)introduce;

+ (NSString *)species;
+ (instancetype)createAnonymous;

@end

// 类实现 (.m文件)
@implementation Person

- (instancetype)initWithName:(NSString *)name age:(int)age {
self = [super init];
if (self) {
_name = [name copy];
_age = age;
}
return self;
}

- (instancetype)initWithName:(NSString *)name age:(int)age email:(NSString *)email {
self = [self initWithName:name age:age];
if (self) {
_email = [email copy];
}
return self;
}

- (NSString *)introduce {
return [NSString stringWithFormat:@"I'm %@, %d years old", _name, _age];
}

+ (NSString *)species {
return @"Human";
}

+ (instancetype)createAnonymous {
return [[Person alloc] initWithName:@"Anonymous" age:0];
}

@end

// 使用类
Person *person = [[Person alloc] initWithName:@"Alice" age:30];
person.age = 31; // 使用属性
NSString *intro = [person introduce];
Person *anonymous = [Person createAnonymous];

主要差异:

  1. Kotlin的主构造函数直接在类头部定义,而Objective-C使用init方法
  2. Kotlin的属性直接在类中声明,而Objective-C使用@property
  3. Kotlin使用伴生对象(companion object)实现类似静态成员的功能,而Objective-C使用类方法(+)
  4. Kotlin的构造过程更简洁,而Objective-C需要显式调用[super init]和设置实例变量
  5. Kotlin自动生成getter和setter,而Objective-C通过@property自动生成或手动实现

继承与构造函数

Kotlin:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
// 基类(默认是final的,需要open才能被继承)
open class Animal(open val name: String) {
open fun makeSound(): String {
return "Some sound"
}
}

// 子类
class Dog(override val name: String, val breed: String) : Animal(name) {
override fun makeSound(): String {
return "Woof!"
}

fun fetch(): String {
return "$name is fetching"
}
}

// 抽象类
abstract class Shape {
abstract fun area(): Double

// 非抽象方法
fun describe(): String {
return "A shape with area ${area()}"
}
}

// 实现抽象类
class Circle(val radius: Double) : Shape() {
override fun area(): Double {
return Math.PI * radius * radius
}
}

Objective-C:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
// 基类
@interface Animal : NSObject

@property (nonatomic, copy) NSString *name;

- (instancetype)initWithName:(NSString *)name;
- (NSString *)makeSound;

@end

@implementation Animal

- (instancetype)initWithName:(NSString *)name {
self = [super init];
if (self) {
_name = [name copy];
}
return self;
}

- (NSString *)makeSound {
return @"Some sound";
}

@end

// 子类
@interface Dog : Animal

@property (nonatomic, copy) NSString *breed;

- (instancetype)initWithName:(NSString *)name breed:(NSString *)breed;
- (NSString *)fetch;

@end

@implementation Dog

- (instancetype)initWithName:(NSString *)name breed:(NSString *)breed {
self = [super initWithName:name];
if (self) {
_breed = [breed copy];
}
return self;
}

- (NSString *)makeSound {
return @"Woof!";
}

- (NSString *)fetch {
return [NSString stringWithFormat:@"%@ is fetching", self.name];
}

@end

// 抽象类(Objective-C没有直接的抽象类语法,通常使用协议或运行时检查)
@interface Shape : NSObject

- (double)area; // 子类应该重写此方法
- (NSString *)describe;

@end

@implementation Shape

- (double)area {
[NSException raise:@"SubclassMustImplement"
format:@"Subclasses must implement area method"];
return 0;
}

- (NSString *)describe {
return [NSString stringWithFormat:@"A shape with area %f", [self area]];
}

@end

// 实现"抽象"类
@interface Circle : Shape

@property (nonatomic, assign) double radius;

- (instancetype)initWithRadius:(double)radius;

@end

@implementation Circle

- (instancetype)initWithRadius:(double)radius {
self = [super init];
if (self) {
_radius = radius;
}
return self;
}

- (double)area {
return M_PI * _radius * _radius;
}

@end

主要差异:

  1. Kotlin默认类是final的,需要使用open关键字允许继承,而Objective-C默认允许继承
  2. Kotlin使用override关键字明确标记重写的方法和属性,而Objective-C没有这种要求
  3. Kotlin有内置的抽象类和方法语法,而Objective-C通常使用运行时异常或协议模拟抽象类
  4. Kotlin的继承使用冒号(:),而Objective-C在类声明中指定父类
  5. Kotlin的构造函数调用更直接,而Objective-C需要显式调用[super init...]

接口

Kotlin:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
// 接口定义
interface Movable {
// 抽象属性
val maxSpeed: Double

// 抽象方法
fun move(distance: Double): String

// 带默认实现的方法
fun stop(): String {
return "Stopped"
}
}

// 实现接口
class Car(override val maxSpeed: Double) : Movable {
override fun move(distance: Double): String {
return "Car moved $distance km"
}

// 可以选择性地重写默认方法
override fun stop(): String {
return "Car stopped with screeching tires"
}
}

// 多接口实现
interface Flyable {
fun fly(): String
}

class Airplane : Movable, Flyable {
override val maxSpeed = 900.0

override fun move(distance: Double): String {
return "Airplane moved $distance km"
}

override fun fly(): String {
return "Airplane is flying"
}
}

Objective-C:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
// 协议定义
@protocol Movable <NSObject>

@required
@property (nonatomic, readonly) double maxSpeed;
- (NSString *)moveWithDistance:(double)distance;

@optional
- (NSString *)stop;

@end

// 实现协议
@interface Car : NSObject <Movable>

@property (nonatomic, readonly) double maxSpeed;

- (instancetype)initWithMaxSpeed:(double)maxSpeed;

@end

@implementation Car

- (instancetype)initWithMaxSpeed:(double)maxSpeed {
self = [super init];
if (self) {
_maxSpeed = maxSpeed;
}
return self;
}

- (NSString *)moveWithDistance:(double)distance {
return [NSString stringWithFormat:@"Car moved %f km", distance];
}

- (NSString *)stop {
return @"Car stopped with screeching tires";
}

@end

// 多协议实现
@protocol Flyable <NSObject>

- (NSString *)fly;

@end

@interface Airplane : NSObject <Movable, Flyable>

@end

@implementation Airplane

- (double)maxSpeed {
return 900.0;
}

- (NSString *)moveWithDistance:(double)distance {
return [NSString stringWithFormat:@"Airplane moved %f km", distance];
}

- (NSString *)fly {
return @"Airplane is flying";
}

@end

主要差异:

  1. Kotlin使用interface关键字,而Objective-C使用@protocol
  2. Kotlin的接口可以有默认实现,而Objective-C的协议使用@optional标记可选方法
  3. Kotlin的接口可以包含属性声明,而Objective-C的协议通常只声明方法(虽然也可以声明属性)
  4. Kotlin的接口实现使用冒号(:),而Objective-C使用尖括号(<>)
  5. Objective-C需要显式标记@required@optional方法,而Kotlin默认所有方法都是必须实现的

扩展与分类

Kotlin扩展:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
// 扩展函数
fun String.addExclamation(): String {
return this + "!"
}

// 使用扩展函数
val message = "Hello".addExclamation() // "Hello!"

// 扩展属性
val String.lastIndex: Int
get() = this.length - 1

// 使用扩展属性
val index = "Kotlin".lastIndex // 5

// 可空接收者的扩展
fun String?.orEmpty(): String {
return this ?: ""
}

// 伴生对象扩展
class MyClass {
companion object {}
}

fun MyClass.Companion.create(): MyClass {
return MyClass()
}

// 使用伴生对象扩展
val instance = MyClass.create()

Objective-C分类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
// NSString的分类声明 (.h文件)
@interface NSString (Additions)

- (NSString *)addExclamation;
- (NSInteger)lastIndex;

@end

// 分类实现 (.m文件)
@implementation NSString (Additions)

- (NSString *)addExclamation {
return [self stringByAppendingString:@"!"];
}

- (NSInteger)lastIndex {
return self.length - 1;
}

@end

// 使用分类
NSString *message = [@"Hello" addExclamation]; // "Hello!"
NSInteger index = [@"Objective-C" lastIndex]; // 11

// 类方法的分类
@interface MyClass : NSObject
@end

@implementation MyClass
@end

@interface MyClass (Creation)
+ (instancetype)create;
@end

@implementation MyClass (Creation)
+ (instancetype)create {
return [[MyClass alloc] init];
}
@end

// 使用类方法分类
MyClass *instance = [MyClass create];

主要差异:

  1. Kotlin的扩展是在编译时解析的,不修改原类,而Objective-C的分类在运行时修改类
  2. Kotlin可以扩展任何类,包括无法修改源码的类,而Objective-C的分类只能添加方法,不能添加实例变量
  3. Kotlin的扩展可以是函数或属性,而Objective-C的分类主要添加方法
  4. Kotlin的扩展可以有可空接收者,而Objective-C需要额外的空检查
  5. Kotlin的扩展在使用时与普通方法无区别,而Objective-C的分类需要导入相应的头文件

2.4 Kotlin特有功能

空安全机制

Kotlin的空安全系统是其最显著的特性之一,它通过编译时检查帮助开发者避免空指针异常。

Kotlin的空安全:

```
// 可空类型声明
var nullableName: String? = “John”
nullableName = null // 合法

// 非空类型声明
var nonNullName: String = “John”
// nonNullName = null // 编译错误

// 安全调用操作符
val length = nullableName?.lTo save on context only part of this file has been shown to you. You should retry this tool after you have searched inside the file with grep -n in order to find the line numbers of what you are looking for.