从Objective-C向Swift转换学习到的经验(转)

首先贴上原文地址 如有侵权 告知删除,在这里仅仅是个人学习和交流
http://www.pig66.com/weixintoutiao/kejika/2016-02-29/688907.html

一、初始化UIView的子类

在iOS应用上实现UI就需要子类化UIView,也就是要重写UIView的init方法。注意:两种语言有所区别。

Objective-C只需在UIView子类中重写必要的init方法。要初始化一个UIView框架,就要重写initWithFrame:框架,如下所示:

@implementation SubUIView

  - (id) initWithFrame:(CGRect)frame{
    self = ;
    if (self != nil) {
        // ...
    }
    return self;
  }
  @end

然而Swift需要多一些步骤来重写同一个init方法。首先,重写使用CGRect框架作为其参数的init方法。根据UIView文档,用Swift语言构建时,须重写init(coder:),但我们不需要这种方法,就用如下代码处理。类属性初始化所需的代码可以在init(frame:)中执行。

class SubUIView: UIView {
  override init(frame: CGRect) {

          super.init(frame: frame)
          // ...
}
    required init?(coder aDecoder: NSCoder) {
          fatalError("init(coder:) has not been implemented")
  }
}

初始化UIViewController的子类

子类化UIViewController是iOS开发的重要步骤。使用Interface Builder的开发者需要重写initWithNibName:bundle:,但既然我们用代码来构建UI,就不需要执行这一方法了。只需重写init方法,在其中初始化类属性即可。

@implementation SubUIViewController

- (id) init{
self = ;
    if (self != nil) {
        // ...
  }
    return self;
}

  @end

Swift也一样要重写init()方法。实现指定的初始化init(nibName:bundle:)来子类化UIViewController。重申:此过程不适用Interface Builder,所以无需定义nibName和bundle的值,而是调用比指定初始化更简单的convenience初始化,将指定初始化init(nibName:bundle:)设为零。现在调用init()来初始化类,用重写的(nibName:bundle:)执行类属性。

class SubUIViewController: UIViewController {

  convenience init() {
      self.init(nibName: nil, bundle: nil)
}

override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: NSBundle?) {

    super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil)
    // Initialize properties of class
}   

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
  }
}

现在可以创建和调用UIViewController的子类,如下所示:

    let viewController: SubUIViewController = SubUIViewController()
    self.navigationController?.pushViewController(viewController, animated: false)

二、使用Auto Layout来实现View

没有Interface Builder的情况下,就用Auto Layout中的NSLayoutConstraint类来设置View的大小和位置——注意Objective-C和Swift在这里有微妙差别。

Objective-C使用NSLayoutConstraint类中的constraintWithItem方法。

+ (instancetype)constraintWithItem:(id)view1
                     attribute:(NSLayoutAttribute)attire
                     relatedBy:(NSLayoutRelation)relation
                        toItem:(id)view2
                     attribute:(NSLayoutAttribute)attr2
                    multiplier:(CGFloat)multiplier
                      constant:(CGFloat)c

Swift使用同一个类中的init方法。

convenience init(item view1: AnyObject,
   attribute attr1: NSLayoutAttribute,
   relatedBy relation: NSLayoutRelation,
      toItem view2: AnyObject?,
   attribute attr2: NSLayoutAttribute,
  multiplier multiplier: CGFloat,
    constant c: CGFloat)


如果是Objective-C,则执行以下代码。这段代码将创建NSLayoutConstraint(定义self.profileImageView和self之间的位置),然后添加到self上。

使用Swift也可以创建NSLayoutConstraint,具体如下:

self.addConstraint(NSLayoutConstraint.init(item: self.profileImageView!, attribute: NSLayoutAttribute.Leading, relatedBy: NSLayoutRelation.Equal, toItem: self, attribute: NSLayoutAttribute.Leading, multiplier: 1, constant: kMessageCellLeftMargin))

比较两种语言版本你会发现,不同于Objective-C,Swift是从NSLayoutConstraint调用init方法的,而且属性和relatedBy的枚举值也有差别。

两种语言NSLayoutConstraint中的枚举值分别是:

   NSLayoutAttribute:

Objective-C

typedef enum: NSInteger {
     NSLayoutAttributeLeft = 1,
     NSLayoutAttributeRight,
     NSLayoutAttributeTop,
     NSLayoutAttributeBottom,
     NSLayoutAttributeLeading,
     NSLayoutAttributeTrailing,     
     NSLayoutAttributeWidth,
     NSLayoutAttributeHeight,
     NSLayoutAttributeCenterX,
     NSLayoutAttributeCenterY,
     NSLayoutAttributeBaseline,
     NSLayoutAttributeLastBaseline = NSLayoutAttributeBaseline,
     NSLayoutAttributeFirstBaseline,
     NSLayoutAttributeLeftMargin,
     NSLayoutAttributeRightMargin,
     NSLayoutAttributeTopMargin,
     NSLayoutAttributeBottomMargin,
     NSLayoutAttributeLeadingMargin,
     NSLayoutAttributeTrailingMargin,
     NSLayoutAttributeCenterXWithinMargins,
     NSLayoutAttributeCenterYWithinMargins,
     NSLayoutAttributeNotAnAttribute = 0
} NSLayoutAttribute;

Swift版:

enum NSLayoutAttribute : Int {
      case Left
      case Right
      case Top
      case Bottom
      case Leading
      case Trailing
      case Width
      case Height
      case CenterX
      case CenterY
      case Baseline
      static var LastBaseline: NSLayoutAttribute { get }
      case FirstBaseline
      case LeftMargin
      case RightMargin
      case TopMargin
      case BottomMargin
      case LeadingMargin
      case TrailingMargin
      case CenterXWithinMargins
      case CenterYWithinMargins
      case NotAnAttribute
}

NSLayoutRelation:

Objective-C版:

enum {
     NSLayoutRelationLessThanOrEqual = -1,
     NSLayoutRelationEqual = 0,
     NSLayoutRelationGreaterThanOrEqual = 1,
};
typedef NSInteger NSLayoutRelation;

Swift版:

enum NSLayoutRelation : Int {
    case LessThanOrEqual
    case Equal
    case GreaterThanOrEqual
}

三、选择器:

使用UIButton、NSNotificationCenter、NSTimer等时,使用选择器来分配要执行的方法。在Objective-C中,@selector指令代表使用选择器。

- (void)test{
      // ...
      mTimer = ;
  }

  - (void)timerCallback:(NSTimer *)timer   {    
      // ...
  }

在Swift中,不需要使用指令或字符串来分配方法。

func test() {
    self.mTimer = NSTimer.scheduledTimerWithTimeInterval(1, target: self, selector: "timerCallback:", userInfo: nil, repeats: true)
}

func timerCallback(timer: NSTimer) {
    // ...
}

四、字符串:

尽管在Swift代码中也可以用Objective-C专门处理字符串的NSString,但要使用以String对象为属性的UITextField上的文本或其他的话,就要清楚NSString和String的区别。

在Objective-C中,UITextField上的文本为NSString,所以属性的长度就是字符串的长度。

- (BOOL)textFieldShouldReturn:(UITextField *)textField{
    NSString *message = ;
      if ( > 0) {
       // ...
      }
      return YES;
  }

Swift是没有长度属性的,所以要用characters属性的count属性。

func textFieldShouldReturn(textField: UITextField) -> Bool {
    let message: String = textField.text!
    if message.characters.count > 0 {
        // ...
    }
   return true
}

在Objective-C中,我们用 stringWithFormat: 来创建一个格式化字符串。

但在Swift中,String里没有 stringWithFormat 方法,所以用init(format:_ arguments:)代之。我们可以分配一个与NSString格式化结构相同的格式化字符串来创建一个新字符串,然后给arguments赋以相关的值。

    self.typingLabel?.text = String.init(format: "%d Typingsomething cool...", count)

五、从数据类型得到最小&最大值

就从数字格式上得到最小和最大值而言,Objective-C和Swift也有差别。Objective-C使用一个预定义宏来得到最小和最大值,但Swift则可以直接从数据类型上得到这些值。Objective-C使用的是如下的宏:

  CGFLOAT_MAX
  CGFLOAT_MIN
  INT32_MAX
  INT32_MIN
  LLONG_MAX
  LLONG_MIN

而Swift则从数据类型上得到最小和最大值,如下:

CGFloat.max
CGFloat.min
Int32.max
Int32.min
Int64.max
Int64.min

六、字典和枚举值:

Objective-C用NSDictionary来定义NSAttributedString的属性。Swift则用Dictionary而不是NSDictionary,但想为Dictionary分配枚举值的时候,做法稍有不同。

Objective-C直接为NSDictionary分配键值,如下所示:称为NSUnderlineStyleSingle的枚举值不能作为NSDictionary值直接分配,所以要先用@()将它转换成一个对象。

  NSDictionary *underlineAttribute = @{NSUnderlineStyleAttributeName: @(NSUnderlineStyleSingle)};

Swift可以直接为Dictionary分配键值(如下所示)。如果该值定义为AnyObject,那么Swift就跟Objective-C一样不能直接使用枚举值,而是使用rawValue属性代之。

let underlineAttribute: =

即使使用相同名称的类,也会在两种语言中遇到针对同一功能的不同方法名称,所以以类参考文件为准比较保险。

如果决定使用Swift,建议先学习其基本知识,并试着将手头现有的Objective-C项目转化为Swift版本练练手。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容