IOS开发Swift 与 OC相互调用详解
<div id="navCategory"><h5 class="catalogue">目录</h5><ul class="first_class_ul"><li>1、创建桥接文件</li><li>2、Swift调用OC</li><ul class="second_class_ul"><li>NS_SWIFT_NAME、NS_SWIFT_UNAVAILABLE</li><li>NS_REFINED_FOR_SWIFT</li><ul class="third_class_ul"><li>规则</li></ul></ul><li>3、OC调用Swift</li><ul class="second_class_ul"></ul><li>4、坑点</li><ul class="second_class_ul"></ul></ul></div><p class="maodian"></p><h2>1、创建桥接文件</h2><p>在创建另一种语言的文件时XCode会提示创建项目名-Bridging-Header.h的桥接文件</p>
<p class="maodian"></p><h2>2、Swift调用OC</h2>
<p>1.创建OC文件</p>
<div class="jb51code"><pre class="brush:cpp;">#import "MyViewController.h"
@interface MyViewController ()
@end
@implementation MyViewController
- (void)viewDidLoad {
;
self.title = @"OC";
self.view.backgroundColor = UIColor.cyanColor;
UILabel *lbl = ;
lbl.text = @"OC的标签";
lbl.backgroundColor = UIColor.redColor;
lbl.frame = CGRectMake(100, 100, 150, 50);
;
}
</pre></div>
<p>2.桥接文件:<strong>项目名-Bridging-Header.h</strong> 文件中要将想要使用的 <strong>OC的.h文件导入</strong></p>
<div class="jb51code"><pre class="brush:cpp;">#import "MyViewController.h"
</pre></div>
<p>3.在Swift文件中调用</p>
<div class="jb51code"><pre class="brush:cpp;">import UIKit
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view.
title = "Swift"
view.backgroundColor = .white
let btn = UIButton(type: .custom)
btn.frame = CGRect(x: 150, y: 150, width: 150, height: 100)
btn.setTitle("点击跳转", for: .normal)
btn.backgroundColor = .green
view.addSubview(btn)
btn.addTarget(self, action: #selector(didClickBtn), for: UIControl.Event.touchUpInside)
}
@objc func didClickBtn() {
let myVC = MyViewController()
navigationController?.pushViewController(myVC, animated: true)
}
}
</pre></div>
<p>Button执行的方法要用 @objc 修饰</p>
<p class="maodian"></p><h3>NS_SWIFT_NAME、NS_SWIFT_UNAVAILABLE</h3>
<ul><li>NS_SWIFT_NAME(替换名):重命名在Swift中的名称,<strong>可用来进行方法名隐藏</strong></li><li>NS_SWIFT_UNAVAILABLE(_msg):<strong>Swift中不可见</strong>,不能使用</li></ul>
<div class="jb51code"><pre class="brush:cpp;">// OC的MyViewController.h文件
#import <UIKit/UIKit.h>
NS_ASSUME_NONNULL_BEGIN
@interface MyViewController : UIViewController
// 将method1方法在Swift中替换成swiftMethod()方法
- (void)method1 NS_SWIFT_NAME(swiftMethd());
// 将method2方法再Swift中隐藏
- (void)method2 NS_SWIFT_UNAVAILABLE("Swift中该方法不可调用");
@end
NS_ASSUME_NONNULL_END
</pre></div>
<div class="jb51code"><pre class="brush:cpp;">class ViewController: UIViewController {
override func viewDidLoad() {
......
}
@objc func didClickBtn() {
let myVC = MyViewController()
// 在Swift中找不到OC的method1与method2方法,只有一个改了名的swiftMethod方法
myVC.swiftMethod()
navigationController?.pushViewController(myVC, animated: true)
}
}
</pre></div>
<p class="maodian"></p><h3>NS_REFINED_FOR_SWIFT</h3>
<p>在Swift中调用OC的接口有时发现并不符合Swift的语法规范或者使用起来会比较别扭,这个时候可以使用NS_REFINED_FOR_SWIFT宏定义 来对OC的接口进行升级改造</p>
<p class="maodian"></p><h4>规则</h4>
<p><strong>NS_REFINED_FOR_SWIFT</strong> 可用于方法和属性,添加了 <strong>NS_REFINED_FOR_SWIFT</strong> 的 Objective-C API 在导入到 Swift 时,具体的 API 重命名规则如下:</p>
<p>对于 <strong>初始化方法</strong>,在其第一个参数标签前面加 "__"</p>
<div class="jb51code"><pre class="brush:cpp;">// Objective-C API
- (instancetype)initWithClassName:(NSString *)name NS_REFINED_FOR_SWIFT;
// In Swift
init(__className: String)
</pre></div>
<p>对于 <strong>其它方法</strong>,在其基名前面加 "__"</p>
<div class="jb51code"><pre class="brush:cpp;">// Objective-C API
- (NSString *)displayNameForMode:(DisplayMode)mode NS_REFINED_FOR_SWIFT;
// In Swift
func __displayNameForMode(mode: DisplayMode) -&gt; String
</pre></div>
<p><strong>下标方法将被视为任何其它方法</strong>,在方法名前面加 <strong>"__"</strong>(而不是作为 Swift 下标导入)</p>
<p>其他声明将在其名称前加上 "__",例如属性</p>
<div class="jb51code"><pre class="brush:cpp;">// Objective-C API
@property DisplayMode mode NS_REFINED_FOR_SWIFT;
// In Swift
var __mode: DisplayMode { get set }
</pre></div>
<p>注意:NS_REFINED_FOR_SWIFT 和 NS_SWIFT_NAME 一起用的话,NS_REFINED_FOR_SWIFT 不生效,而是以 NS_SWIFT_NAME 指定的名称重命名 Objective-C API</p>
<p class="maodian"></p><h2>3、OC调用Swift</h2>
<p>创建Swift文件</p>
<div class="jb51code"><pre class="brush:cpp;">import Foundation
// 必须继承于 NSObject
class Person: NSObject {
// 想公开给OC的要使用 @objc 修饰
@objc var name: String
@objc var age : Int
@objc init(name: String, age: Int) {
self.name = name
self.age = age
}
}
</pre></div>
<ul><li>必须继承于 NSObject,类、结构体等才会公开给OC</li><li>必须使用 @objc 修饰,属性、方法等才会公开给OC</li></ul>
<p>在Swift文件中引入项目名-Swift.h文件,然后使用Swift内容</p>
<div class="jb51code"><pre class="brush:cpp;">#import "MyViewController.h"
#import "SwiftAndOC-Swift.h"
@interface MyViewController ()
@end
@implementation MyViewController
- (void)viewDidLoad {
;
Person *p = [ initWithName:@"LZ" age:18];
NSLog(@"%@",p.name);
}
</pre></div>
<p class="maodian"></p><h2>4、坑点</h2>
<ul><li>OC类不能继承于Swift类,但Swift类可以继承于OC类</li><li>Swift中没有宏定义:</li></ul>
<p>常量宏用let参数代替</p>
<p>无参变量宏可以用"只读属性"代替也可用函数代替</p>
<p>变量宏用函数代替</p>
<ul><li>要给OC用的内容不要用Swift独有特性书写(比如元组)</li><li>如果OC通过pod的形式集成Swift,需要在 <strong>Swift的类上也要声明Public</strong>,否则在对应的 <strong>项目名-Swift.h</strong> 上不会有对应的类出现</li></ul>
<p>以上就是IOS开发Swift 与 OC相互调用详解的详细内容,更多关于Swift OC相互调用的资料请关注琼殿技术社区其它相关文章!</p>
<div class="art_xg">
<b>您可能感兴趣的文章:</b><ul><li>iOS Swift利用UICollectionView实现无限轮播功能(原理)详解</li><li>iOS开发中Swift逃逸闭包知识</li><li>iOS Swift控制器转场动画示例代码</li><li>iOS Swift读取本地json文件报错的解决方法</li><li>IOS 开发之swift中手势的实例详解</li><li>iOS Swift 值类型与引用类型使用区别基础详解</li></ul>
</div>
</div>
<!--endmain-->
頁:
[1]