2015-11-13 10:06:44 lvdezhou 阅读数 1579
  • iOS/Swift/OC/Objective-C/Xcode/0基础/入门

    这是一门快速入门iOS开发的课程,目的是让大家快速学会,iOS开发环境搭建,和iOS一些基础知识,最后完成一个小项目。 项目信息 提供完整的Git提交历史,和每节视频一一对应,目前有41次提交,355行注释,271行代码(不包含可视化布局文件,纯Swift和Objective-C代码)。

    2848 人正在学习 去看看 任苹蜻

1、oc工程调用swift-----自动创建桥接头文件

创建一个oc工程:


设置Product Module Name = 工程名



创建一个swift文件:

点击自动创建桥接头文件:



设置Objective-C Bridging Header 如下图:



至此,我们可以在oc中调用swift了

测试代码:


import Foundation

class Student: NSObject {
    var name : String = "dzl"
    var age : Int = 22
}

在oc中引入头文件  "工程名-swift.h" 就可以使用swift中的类了

注:这个头文件是不可见的

#import "ViewController.h"

#import "OCuseSwift1-swift.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    
    Student * s1 = [[Student alloc] init];
    NSLog(@"%@", s1.name);
}

- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

输出:

2015-11-13 10:01:32.250 OCuseSwift1[1040:20007] dzl

此时swift也是可以引用OC的,只需将swift需要使用的oc类头文件在桥接头文件中引入即可,






2、oc工程调用swift-----手动创建桥接头文件


首先创建一个oc工程,与上面相同

然后创建一个swift文件,但是不选择创建桥接头文件,如图:



然后手动创建桥接头文件,就相当于创建一个普通的头文件一样,只是命名一定要严格按照规则:



工程设置中指定桥接头文件:


经测试,oc引用swift或者swift引用oc都是可以的,测试方法与上面相同,这里不再赘述。



3、swift工程调用-----自动创建桥接头文件 和 手动创建桥接头文件,基本与上面操作一直,不再详述




其实桥接头文件里面引入oc头文件,是为了swift可以访问oc文件,

在oc中引入   工程名-swift.h   文件  是为了oc可以访问swift,但这个文件是不可见的,不过可以通过command+鼠标左键查看该文件



2015-11-14 11:30:12 Xiao_Long_Li 阅读数 1592
  • iOS/Swift/OC/Objective-C/Xcode/0基础/入门

    这是一门快速入门iOS开发的课程,目的是让大家快速学会,iOS开发环境搭建,和iOS一些基础知识,最后完成一个小项目。 项目信息 提供完整的Git提交历史,和每节视频一一对应,目前有41次提交,355行注释,271行代码(不包含可视化布局文件,纯Swift和Objective-C代码)。

    2848 人正在学习 去看看 任苹蜻

OC调用Swift


如果你想在Objective-C工程中,嵌入Swift文件,实现OC调用Swift,那么这篇文章就能帮助你,由于本人最近在开发Swift项目,所以实现过OC调用Swift,在这过程中遇到过一些瓶颈,特此,将此文奉上,供大家参考。

OC调用Swift文件,需要创建桥接头文件,这个文件是当你在OC开发环境中创建Swift文件时由Xcode生成,该桥接头文件的命名是:<工程名>-Bridging-Header.h,它的作用是实现OC和Swift文件混编,并且如果我们想在当前OC类中调用Swift文件,必须在当前OC类中包含头文件,它的命名是:<工程名>-Swift.h,该头文件是由Xcode本身去维护,当我们点击进入到该头文件时,会看到Swift文件被编译成了OC封装的接口。

下面让我们来通过一个小Demo,进一步了解如何实现OC调用Swift。


第一步:创建OC工程

启动Xcode 7.1,然后单击File→New→Project菜单,在打开的Choose atemplate for your new project界面中选择“iOS →Application→Single View Application”工程模板(如下图所示)。


选中Single View Application单击“Next”,会出现如下图所示



这里我们将工程名命名为OCCallSwift,单击“Next”,就会出现下图所示的界面


单击“Create”,这时我们的OC工程就创建好了。

接下来我们创建一个OC的类,类名为OCViewController,具体怎么创建就不详细说了。


第二步:在OC工程中添加Swift文件


在当前OC工程中,创建Swift文件,如下图所示:


选中Cocoa Touch Class,点击“Next”,此时会跳到下图:


这里我们将Swift文件命名为SwiftViewController,将Language设置为Swift,单击“Next”,跳到下图:


这时单击“Creat”,就会弹出“Would you like to configure an Objective-C bridging header?”,如下图所示


单击“Create Bridging Header”,这时我们就在OC工程中创建好了一个Swift文件,如下图所示



第三步:实现OC调用Swift

这里我们就简单实现一下在OC类中跳转到Swift文件中

1)在OC类中包含头文件

#import <OCCallSwift-Swift.h>,命名规则为:<工程名>-Swift.h,如下图所示


2)在OC类中调用Swift文件,跟OC类调用OC类实现方式一样,如下图所示:



这样我们就实现了OC调用Swift。


友情提示:1.以上的代码是在Xcode7.1中实现,并且是新工程;

2.必须创建桥接头文件(“Create Bridging Header);

3.必须在调用Swift文件的OC类中包含如下头文件:<工程名>-Swift.h

4.如果以上条件全部符合,但是你在一个老的OC应用中调用Swift,有可能出现下面情况'<工程名>.Swift' file not found,这时你就应该在Xcode中的Build Setting中找到"Defines module",将其改为Yes


特别说明:1. 以上Swift文件均指Swift类;

  2. Swift语言为Swift2.0版本。


著作权声明:本文为原创,欢迎转载分享。请尊重作者劳动,转载时保留该声明和作者博客链接,谢谢

2019-03-25 14:27:53 lfdanding 阅读数 81
  • iOS/Swift/OC/Objective-C/Xcode/0基础/入门

    这是一门快速入门iOS开发的课程,目的是让大家快速学会,iOS开发环境搭建,和iOS一些基础知识,最后完成一个小项目。 项目信息 提供完整的Git提交历史,和每节视频一一对应,目前有41次提交,355行注释,271行代码(不包含可视化布局文件,纯Swift和Objective-C代码)。

    2848 人正在学习 去看看 任苹蜻

最近在oc工程里要引用swiftMonkeypaws,需要使用oc调用swift技术,里面踩了很多坑,这里简单记录一下。

1、oc调用swift

oc调用swift其实很简单,google一下就有很多的教程,可以参考文章1、2、3、4。总体上就是在要调用swift代码的时候,引入头文件 “模块名-Swift.h”,这个头文件对外不可显示,需要手动导入。可以在 Build setting 中查看:
在这里插入图片描述
现在我们在 OC 中添加这个头文件:
在这里插入图片描述
我们需要查看一下swift到底可以调用那些方法和类。这个时候,我们可以点进这个头文件中去。进入头文件往下拉,可以看到暴露出来的 Swift 的类名和方法:
在这里插入图片描述

2、oc调用swift ,涉及cocoapods

上面完全是主工程里使用当然毫无难度,但是我要引入的swiftMonkeypaws是要以cocoapods方式进行引入,这里面就有很多的问题。

2.1 引入头文件名字不一样

头文件要引入的格式为 #import <产品名/模块名-Swift.h>,参考文章5,我们可以在 Build Settings 中查看:
在这里插入图片描述
2.2、头文件引入的位置
头文件如果引用在pods工程的起点文件中,还是会报找不到头文件的错误,这里面我没有深入研究pods工程的编译原则,但是不难猜出因为是在pods工程的起点文件中,还没有开始编译pod工程里的swift文件,所以就没有生成oc调用swift的接口文件,导致头文件找不到。我这边的临时解决办法就是把头文件引用移动到了其他的位置
在这里插入图片描述

3、oc引用swiftMonkeypaws

oc引用swiftMonkeypaws,网上的文章也很多,可以参考文章6、7、8、9、10。对于参考文章8,矫正一下

var paws: MonkeyPaws?

class MonkeyUIPaws: NSObject 
{        
    @objc public func showMonkeyPawsINUITest ( window: UIWindow ) -> () {
        paws = MonkeyPaws(view: window)    
    }

     @objc class func test() -> () {
        print("test")
    }
}

如果不这么修正,你会收到 unrecognized selector sent to instance,主要原因是因为把MonkeyPaws设置成局部变量以后,这个界面释放以后,MonkeyPaws也被释放,再调用MonkeyPaws的方法就会引发这个错误。所以要把MonkeyPaws设置到类外面。对于unrecognized selector sent to instance这个错误的详细描述可以参考文章11、12。

参考文章:

1、https://blog.csdn.net/studying_ios/article/details/70146879
2、https://www.jianshu.com/p/754396e7e1bd
3、https://www.jianshu.com/p/9f757a09eacd
4、https://www.jianshu.com/p/979ff6350cb6
5、https://zhang759740844.github.io/2017/07/05/Swift%E5%92%8COC%E7%9A%84%E6%B7%B7%E7%BC%96%E5%AE%9E%E8%B7%B5/
6、http://www.php-master.com/post/151913.html
7、https://www.jianshu.com/p/34bf481d5c13
8、https://www.jianshu.com/p/b763b65005bd
9、https://gysxl.github.io/2017/11/03/%E5%9F%BA%E4%BA%8EXCUITest%E8%BF%9B%E8%A1%8C%E7%9A%84iOSMonkey%E6%B5%8B%E8%AF%95/
10、http://www.php-master.com/post/151913.html
11、https://moto0421.iteye.com/blog/1570876
12、https://juejin.im/entry/58eddde3570c350057f2b975

2019-02-26 14:16:29 wei78008023 阅读数 1518
  • iOS/Swift/OC/Objective-C/Xcode/0基础/入门

    这是一门快速入门iOS开发的课程,目的是让大家快速学会,iOS开发环境搭建,和iOS一些基础知识,最后完成一个小项目。 项目信息 提供完整的Git提交历史,和每节视频一一对应,目前有41次提交,355行注释,271行代码(不包含可视化布局文件,纯Swift和Objective-C代码)。

    2848 人正在学习 去看看 任苹蜻

之前在开发是遇到了在OC文件中调用Swift对象的变量调用不到的问题,这里总结一下:

1.首先我们一个Swift项目TestProject。

2.建立一个Swift的文件继承自NSObject名字为SwiftFile.swift,并声明一个字符串变量swiftName。

import UIKit

class SwiftFile: NSObject {

    var swiftName = ""
}

3.建立一个OC的文件ObjectFile,并导入“HFSTeacher-Swift.h”(OC想要调用Swift需要导入此文件)如何配置以及导入原因可以自行查询。

#import "ObjectFile.h"
#import "HFSTeacher-Swift.h"

@implementation ObjectFile

-(void)oneMethod
{
    SwiftFile *swiftFile = [[SwiftFile alloc]init];
    NSLog(@"%@",swiftFile.swiftName);
}

@end

这个时候会报错,说在SwiftFile中找不到swiftName成员。

4.这个时候需要在SwiftFile.swift中加入一个声明@objcMembers以表示文件中成员可以被OC调用:

import UIKit

@objcMembers
class SwiftFile: NSObject {

    var swiftName = ""
}

5.如此则大功告成!

2018-08-10 20:02:33 weixin_38633659 阅读数 677
  • iOS/Swift/OC/Objective-C/Xcode/0基础/入门

    这是一门快速入门iOS开发的课程,目的是让大家快速学会,iOS开发环境搭建,和iOS一些基础知识,最后完成一个小项目。 项目信息 提供完整的Git提交历史,和每节视频一一对应,目前有41次提交,355行注释,271行代码(不包含可视化布局文件,纯Swift和Objective-C代码)。

    2848 人正在学习 去看看 任苹蜻

简介

该项目主要介绍了oc与swift之间、swift内部几种常见的传值方式(属性传值、代码块传值、代理传值、通知)

如果oc与swift之间的桥接有疑问请参考:OC与swift桥接互调

项目介绍

  • 1、swift之间的页面跳转与返回
  • 2、oc监听swift推送的通知
  • 3、swift内部推送接收的通知
  • 4、oc调用swift的代理方法
  • 5、swift调用oc代码块
  • 6、oc对swift的属性传值、方法调用、代码块调用
  • 7、swift对oc的属性传值、方法调用、代码块调用
  • 8、swift之间的传值

1、swift之间的页面跳转与返回

1、present方式跳转

 @objc func btnAction(_ sender:UIButton){
        let subVC  = SubVC();
        self.present(subVC, animated: true) {
            NSLog("������:%@", subVC);
        };
        NSLog("������:%@____%@",sender,self);
    }

2、返回

self.dismiss(animated: true) {
                NSLog("返回");
            };

2、oc监听swift推送的通知:

  • swift发送通知
 func postNotifition(){
        NotificationCenter.default.post(name: NSNotification.Name(rawValue: "XMNotification"), object: "通知方法");
    }
  • oc接收通知
-(void)addNotification{
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notiAction:) name:@"XMNotification" object:nil];
}

-(void)notiAction:(NSNotification *)sender{
    NSLog(@"oc:%@",sender.object);
}

-(void)removeNotification{
    [[NSNotificationCenter defaultCenter] removeObserver:self name:@"XMNotification" object:nil];
}

3、swift内部推送接收的通知:

 func postNotifition(){
        NotificationCenter.default.post(name: NSNotification.Name(rawValue: "XMNotification"), object: "通知方法");
    }

    @objc func addNotification(){
        NotificationCenter.default.addObserver(self, selector: #selector(notificationAction(_:)), name: NSNotification.Name(rawValue: "XMNotification"), object: nil);
    }
    @objc func removeNotifition() {
        NotificationCenter.default.removeObserver(self);
    }

    @objc private func notificationAction(_ noti : Notification){
        let str:String = noti.object as! String;
        print("swift:" + str);
    }

通知打印结果

4、oc调用swift的代理方法:

  • 编辑协议方
///1、编辑协议
@objc(FourVCDelegate)
protocol FourVCDelegate {
    func backSuperVC(str:String)
}

///2、定义协议对象
@objc var myDelegate:FourVCDelegate?;

///3、调用协议
let str:String = "代理方法"
self.myDelegate?.backSuperVC(str: str);
  • 签署协议方
///1、调用协议
@interface FirVC ()<FourVCDelegate>

///2、签署协议
-(SecVC *)seVc{
    if (!_seVc) {
        _seVc = [SecVC new];
        _seVc.myDelegate = self;
    }
    return _seVc;
}

///3、协议实现
-(void)backSuperVCWithStr:(NSString *)str{
    NSLog(@"%@", str);
}

代理协议打印结果

5、swift调用oc代码块

  • oc文件类
///代码块定义
@property(nonatomic,strong)void (^thiBlock)(NSString *str);

///代码块实现
!self.thiBlock?:self.thiBlock(@"swift调用oc代码块");
  • swift文件类

let vc  = ThiVC();
///代码块调用
vc.thiBlock = {(str) -> () in
    print("代码块方法");
}

self.present(vc, animated: true, completion: {

});

提示

6、oc对swift的属性传值、方法调用、代码块调用

  • swift方法类
import UIKit

class FourVC: UIViewController {

    ///数组
    @objc var arr0:NSArray = NSArray()

    ///无参数无返回值
    @objc func swMethod0(){
        print("无参数无返回值");
    }
    ///有参数无返回值
    @objc func swMethod1(str: String)  {
        print("有参数无返回值:\(str)");
    }
    ///有参数有返回值
    @objc func swMethod2(str: String) -> (String){
        return "有参数有返回值:" + str;
    }

    /// mark - 代码块
    ///无返回参数
    @objc func bkBlock0(response:() -> ()){
        response();
    }
    ///返回一个字符串参数
    @objc func bkBlock1(response:(_ res : String) -> ()) {
        response("返回一个字符串参数");
    }
    ///返回多个任意类型参数
    @objc func bkBlock_2_(response:(_ res : Any,_ res1 : Any) -> ()) {
        response(self.arr0, "无参数,返回多个任意类型参数:");
    }
    ///传入多个参数 并返回多个任意类型的参数
    @objc func bkBlock3(p1:Any,p2:Any,p3:Any,res:(_ res0 : Any ,_ res1 : Any ,_ res2:Any,_ res3:Any) -> ()) {
        res("传入多个参数 并返回多个任意类型的参数:\n",p1, p2, p3);
    }
    ///Block带返回值
    @objc func bkBlock4(res:() -> ()) -> (String){
        res();
        return "Block带返回值"
    }


    //定义block
    typealias fucBlock = (_ backMsg : String?) ->()
    func bkBlock4(cmdStr:String?,blockProperty:fucBlock){
        blockProperty("backMsg---by block inside func")
    }

    override func viewDidLoad() {
        super.viewDidLoad()
        self.view.backgroundColor = UIColor.yellow;
        print("属性传值结果:",self.arr0[0],self.arr0[1],arr0[2]);
    }

    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        self.dismiss(animated: true) {

        };
    }
}
  • oc方法类:
    FourVC *vc = [FourVC new];
    // --- 属性传值 ---
    vc.arr0 = @[@"可以",@"可以",@"可以"];

    /// ------ 方法调用 -----

    //无参数无返回值
    [vc swMethod0];
    //有参数无返回值
    [vc swMethod1WithStr:@"dasda"];
    //有参数有返回值
    NSString *str = [vc swMethod2WithStr:@"sdfasda"];
    NSLog(@"%@",str);

    /// ------ 代码块调用 -----

    [vc bkBlock0WithResponse:^{
        NSLog(@"无返回数据代码块");
    }];

    //有一个返回值
    [vc bkBlock1WithResponse:^(NSString * str) {
        NSLog(@"%@", str);
    }];

    //有两个返回值
    [vc bkBlock_2_WithResponse:^(id str, id str1) {
        NSLog(@"%@\n%@",str1,str);
    }];

    //有参数有返回值
    [vc bkBlock3WithP1:@"ds" p2:@"dsa" p3:@"das" res:^(id p0,id p1, id p2, id p3) {
        NSLog(@"%@\n%@\n%@\n%@",p0,p1,p2,p3);
    }];

    [self presentViewController:vc animated:YES completion:^{

    }];

此处输入图片的描述

7、swift对oc的属性传值、方法调用、代码块调用

  • xxx.h文件
@interface OcVC : UIViewController
///代码块
@property(nonatomic,strong)void (^clickBlock)(void);
///字符串
@property(nonatomic,copy)NSString *titStr;
///方法
-(void)testAction;
@end
  • xxx.m文件
- (void)viewDidLoad {
    [super viewDidLoad];
    NSLog(@"%@",self.titStr);
    !self.clickBlock?:self.clickBlock();
}

-(void)testAction{
    NSLog(@"调用方法");
}
  • xxx.swift文件

let sub  = OcVC();
//属性传值
sub.titStr = "属性传值";
//方法调用
sub.testAction();

//代码块调用
sub.clickBlock = { () -> () in
     print("不带参数代码块")
};

sub.clickBlock();
self.present(sub, animated: true, completion: {

});

打印结果

8、swift之间的传值

    func jumpToNextVC(){
        ///mark - =============
        let foVC =  FourVC()

        let dic:NSDictionary = ["key":1,"key1":2];
        let arr0:NSMutableArray = ["你好","好不好"];

        // mark -  属性传值
        foVC.arr0 = arr0;

        //方法调用
        foVC.swMethod0();
        foVC.swMethod1(str: "da");
        let str = foVC.swMethod2(str: "ddd");
        print(str);

        // mark -  代码块回调
        foVC.bkBlock0 {
            print("你可以的");
        };
        foVC.bkBlock1 { (res) in
            print(res);
        }
        foVC.bkBlock_2_{ (res, str) in
            print(res, "+", str);
        }
        foVC.bkBlock3(p1: "1", p2: "2", p3: "3") { (res0, res1, res2, res3) in
            print(res0,res1,res2,res3);
        }
        self.present(foVC, animated: true) {

        };
    }

OC调用Swift方法

阅读数 5625

oc 调用 swift Demo

阅读数 311

没有更多推荐了,返回首页