在Swift中使用Objective-C编写类、继承Objective-C类_Swift

互用性(互操作性)使开发者可以定义融合了 Objective-C 语言特性的Swift类。编写 Swift 类时,不仅可以继承 Objective-C 语言编写的父类,采用 Objective-C 的协议,还可以利用 Objective-C 的一些其它功能。这意味着,开发者可以基于 Objective-C 中已有的熟悉、可靠的类、方法和框架来创建 Swift 类,并结合 Swift 提供的现代化和更有效的语言特点对其进行优化。

继承Objective-C的类

在 Swift 中,开发者可以定义一个子类,该子类继承自使用 Objective-C 编写的类。创建该子类的方法是,在 Swift 的类名后面加上一个冒号(:),冒号后面跟上 Objective-C 的类名。

复制代码 代码如下:

// SWIFT
import UIKit

class MySwiftViewController: UIViewController {
    // 定义类
}

开发者能够从 Objective-C 的父类中继承所有的功能。如果开发者要覆盖父类中的方法,不要忘记使用override关键字。
采用协议

在 Swift 中,开发者可以采用 Objective-C 中定义好的协议。和 Swift 协议一样,所有 Objective-C 协议都写在一个用逗号隔开的列表中,跟在所在类的父类名后面(如果它有父类的话)。

复制代码 代码如下:

// SWIFT
class MySwiftViewController: UIViewController, UITableViewDelegate, UITableViewDataSource {
    // 定义类
}

Objective-C 协议与 Swift 协议使用上是一致的。如果开发者想在 Swift 代码中引用 UITableViewDelegate协议,可以直接使用UITableViewDelegate(跟在 Objective-C 中引用id<UITableViewDelegate\>是等价的)。

编写构造器和析构器

Swift 的编译器确保在初始化时,构造器不允许类里有任何未初始化的属性,这样做能够增加代码的安全性和可预测性。另外,与 Objective-C 语言不同,Swift 不提供单独的内存分配方法供开发者调用。当你使用原生的 Swift 初始化方法时(即使是和 Objective-C 类协作),Swift 会将 Objective-C 的初始化方法转换为 Swift 的初始化方法。关于如何实现开发者自定义构造器的更多信息,请查看构造器。

当开发者希望在类被释放前,执行额外的清理工作时,需要执行一个析构过程来代替dealloc方法。在实例被释放前,Swift 会自动调用析构器来执行析构过程。Swift 调用完子类的析构器后,会自动调用父类的析构器。当开发者使用 Objective-C 类或者是继承自 Objective-C 类的 Swift 类时,Swift 也会自动为开发者调用这个类的父类里的dealloc方法。关于如何实现开发者自定义析构器的更多信息,请查看析构器。

集成Interface Builder

Swift 编译器包含一些属性,使得开发者的 Swift 类集成了 Interface Builder 里的一些特色功能。和 Objective-C 里一样,你能在 Swift 里面使用 OutLets,actions 和实时渲染(live rendering)。

使用Outlets和Action

使用 Outlets 和 Action 可以连接源代码和 Interface Builder 的 UI 对象。在Swift里面使用 Outlets 和 Action,需要在属性和方法声明前插入@IBOutlet或者@IBAction关键字。声明一个 Outlet集合同样是用@IBOutlet属性,即为类型指定一个数组。

当开发者在 Swift 里面声明了一个 Outlet 时,Swift 编译器会自动将该类型转换为弱(weak)、隐式(implicitly)、未包装(unwrapped)的 optional(Object-c里面对应指针类型)数据类型,并为它分配一个初始化的空值nil。实际上,编译器使用@IBOutlet weak var name: Type! = nil来代替 @IBOutlet var name: Type。编译器将该类型转换成了弱(weak)、隐式(implicitly)、未包装(unwrapped)的 optional 类型,因此开发者就不需要在构造器中为该类型分配一个初始值了。当开发者从故事板(storyboard)或者xib文件里面初始化对象 class 后,定义好的 Outlet 和这些对象连接在一起了,所以,这些 Outlet 是隐式的,未包装的。由于创建的 outlets 一般都是弱关系,因此默认 outlets 是弱类型。

例如,下面的 Swift 代码声明了一个拥有 Outlet、Outlets 集合和 Action 的类:

复制代码 代码如下:

// SWIFT
class MyViewController: UIViewController {

    @IBOutlet var button: UIButton

    @IBOutlet var textFields: UITextField[]

    @IBAction func buttonTapped(AnyObject) {
        println("button tapped!")
    }
}

在buttonTapped方法中,消息发送者的信息没有被使用,因此可以省略该方法的参数名。
实时渲染(live rendering)

开发者可以在Interface Builder中用@IBDesignable和@IBInspectable来创建生动、可交互的自定义视图(view)。开发者继承UIView或者NSView来自定义一个视图(view)时,可以在类声明前添加@IBDesignable属性。当你在 Interface Builder 里添加了自定义的视图后(在监视器面板的自定义视图类中进行设置),Interface Builder 将在画布上渲染你自定义的视图。

注意:只能针对框架里对象进行实时渲染

你也可以将@IBInspectable属性添加到和用户定义的运行时属性兼容的类型属性里。这样,当开发者将自定义的视图添加到 Interface Builder 里后,就可以在监视器面板中编辑这些属性。

复制代码 代码如下:

// SWIFT

@IBDesignable

class MyCustomView: UIView {
    @IBInspectable var textColor: UIColor
    @IBInspectable var iconHeight: CGFloat
    /* ... */
}

指明属性特性

在 Objective-C 中,属性通常都有一组特性(Attributes)说明来指明该属性的一些附加信息。在 Swift 中,开发者可以通过不同的方法来指明属性的这些特性。

强类型和弱类型

Swift 里属性默认都是强类型的。使用weak关键字修饰一个属性,能指明其对象存储时是一个弱引用。该关键字仅能修饰 optional 对象类型。更多的信息,请查阅特性。

读/写和只读

在 Swift 中,没有readwrite和readonly特性。当声明一个存储型属性时,使用let修饰其为只读;使用var修饰其为可读/写。当声明一个计算型属性时,为其提供一个 getter 方法,使其成为只读的;提供 getter 方法和 setter 方法,使其成为可读/写的。更多信息,请查阅属性。

拷贝

在 Swift 中,Objective-C 的copy特性被转换为@NSCopying属性。这一类的属性必须遵守 NSCopying协议。更多信息,请查阅特性。

实现Core Data Managed Object子类

Core Data 提供了基本存储和实现NSManagedObject子类的一组属性。在 Core Data 模型中,与管理对象子类相关的特性或者关系的每个属性定义之前,将@NSmanaged特性加入。与 Objective-C 里面的 @dynamic特性类似,@NSManaged特性告知 Swift 编译器,这个属性的存储和实现将在运行时完成。但是,与@dynamic不同的是,@NSManaged特性仅在 Core Data 支持中可用。

时间: 2024-10-24 12:38:05

在Swift中使用Objective-C编写类、继承Objective-C类_Swift的相关文章

MFC中使用Tab Control从CTabCtrl控件类继承自己的类(CTabSheet)来处理。

使用Tab Control,并且从CTabCtrl控件类继承自己的类(CTabSheet)来处理. 首先我先介绍一下如何使用CTabSheet. 先要制作子对话框类,这次的子对话框类不要从CPropertyPage继承,而是直接从CDialog继承.并且各个子对话框资源的属性应设置为:Style为Child, Border为None. 在主对话框资源中,加入一个Tab Control,并且适当调整位置和大小.利用ClassWizard来为这个Tab Control创建一个CTabSheet的控件

深入解析Swift中switch语句对case的数据类型匹配的支持_Swift

Swift可以对switch中不同数据类型的值作匹配判断: var things = Any[]() things.append(0) things.append(0.0) things.append(42) things.append(3.14159) things.append("hello") things.append((3.0, 5.0)) things.append(Movie(name:"Ghostbusters", director:"Iv

Swift中通过叠加UILabel实现混合进度条的方法_Swift

先给大家展示下效果图,如果大家感觉还不错,请参考实现代码. 效果图如下所示: 源码 https://github.com/YouXianMing/Swift-Animations // // MixedColorProgressViewController.swift // Swift-Animations // // Created by YouXianMing on 16/8/21. // Copyright 2016年 YouXianMing. All rights reserved. /

Microsoft .NET 中的基类继承

接口继承 创建抽象类时,请使用关键字 Interface 而不是 Class.为接口命名,然后定义需要子类实现的所有属性和方法.这是因为基类中没有可以实现的属性和方法,它只包含一般数据,而不包含方法.您所创建的只是一个合约,它规定所有使用此接口的子类都必须遵循一定的规则. 1.        现在,请在已创建的项目中添加一个新类. 2.        从 Visual Studio 菜单中,单击 Project(项目),然后单击 Add Class(添加类). 3.        在类中添加以下

Microsoft .NET 中的基类继承(2)

现在,请在已创建的项目中添加一个新类. 从 Visual Studio 菜单中,单击 Project(项目),然后单击 Add Class(添加类). 在类中添加以下代码:Interface Person   Property FirstName() As String   Property LastName() As StringSub Print()   Sub Talk()End Interface您会发现,您定义属性和子过程的方法与您通常定义这些属性和过程的方法一样.唯一的差别在于,您没

一方面:int是值类型,int a 定义在栈上;另一方面int又继承自Object类,是个类,类的对象应该在堆上

问题描述 ①inta=newint();这种写法,a中存储的是地址吗?②有人说,int是个静态类,所以可以inta:这个说法对吗?③版主说,int是值类型,在编译的时候会自动进行装箱拆箱,请问inta=10在编译时会被装箱成对象吗(如果这样岂不是更复杂了)?④一方面:int是值类型,inta定义在栈上:另一方面int又继承自Object类,是个类,类的对象应该在堆上.这不是矛盾嘛? 解决方案 解决方案二:引用楼主sshziliao3的回复: ①inta=newint();这种写法,a中存储的是地

Ruby简洁学习笔记(二):类继承、属性、类变量_ruby专题

1.如何声明一个子类 复制代码 代码如下: class Treasure < Thing 这样Thing类中的属性name,description都被Treasure继承 2.以下三种方式传入父类initialize方法的参数分别是什么? 复制代码 代码如下: # This passes a, b, c to the superclass def initialize( a, b, c, d, e, f ) super( a, b, c ) end # This passes a, b, c t

javascript类继承的一些实验

其实一开始编js没怎么用过对象,一般都用func,func,func···但是用多了,感觉代码一点都不美观,还要这里包一个函数,那里包一个 函数,或者一直都是函数调用,不好看,而且一些重用的都要重写的话就很麻烦(不好意思,对于我这种新手,开始还是一般用func比较高效率···).所以 就决定开始要用object来编程才能更省事,下面就是我看了一些博客文章关于类的见解,有什么不对的希望各位可以多多给些指点: 对于类的编程,声明的方法有如下几种:1.var test = function(){};2

Python类继承的实现

代码: #! /usr/bin/python # Filename: inherit.py # Author: yanggang class SchoolMember: def __init__(self,name,age): self.name = name self.age = age print 'init SchoolMember: ', self.name def tell(self): print 'name:%s; age:%s' % (self.name, self.age) c