推广

Swift 优雅的前缀

iseeyu2年前 (2024-02-21)推广138

朋友的自拍.jpg

从OC转Swift不再需要添加类前缀了,如OC的PMHomeViewController转到Swift可以命名为HomeViewController。因为Swift有命名空间的概念了。不用担心与系统或者第三方命名。如果创建的类和系统的类重名,调用自己的直接调用即可,系统的类则需要加上相应的框架名字,如

import UIKit
class UIView {
    
}
class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        
        let v1 = UIKit.UIView()
        let v2 = Project.UIView() //项目名称或说Target为Project,通常被省略
        print("v1:\(v1)")
        print("v2:\(v2)")
    }
}
//v1:<UIView: 0x7ff24e40a830; frame = (0 0; 0 0); layer = <CALayer: 0x6000027f2a80>>
//v2:Project.UIView

此处Project就有命名空间的意思了

本文借鉴Swift编程从入门到精通-MJ大神精选课程中的一节,将加前缀的推导过程整理并加深自己的理解。看过的朋友可跳过。

假设有个字符串,需要计算包含的数字个数。通常的做法是写个方法传递该字符串,返回数字的个数,如:

//计算字符串里的数字
let str = "123test123" 
 func numberCount(_ str: String) -> Int {
    var count = 0
    for c in str {
        if ("0"..."9").contains(c) {
            count += 1
        }
    }
    return count
 }

优化成Swift 风格

 func numberCount(_ str: String) -> Int {
    var count = 0
    for c in str where ("0"..."9").contains(c) {
        count += 1
    }
    return count
 }

可以直接调用

print("数字count", numberCount(str))

一般情况,写的是字符串相关的方法,会扩充到相应类/结构体的扩展extension

 extension String {
     //给字符串扩展功能
     func numberCount() -> Int {
         var count = 0
         for c in self where ("0"..."9").contains(c) {
             count += 1
         }
         return count
     }
 }
 print(str.numberCount()) //可以使用str直接调用改方法

调方法有小括号,可以改为计算属性,去掉小括号

 extension String {
     //改为计算属性
     var numberCount: Int {
         var count = 0
         for c in self where ("0"..."9").contains(c) {
             count += 1
         }
         return count
     }
 }
 print(str.numberCount)

计算属性可能跟系统自带的产生冲突

解决办法

  1. 仿照OC写前缀
var pm_numberCount: Int {
     var count = 0
     for c in self where ("0"..."9").contains(c) {
        count += 1
     }
     return count
  }
 print(str.pm_numberCount)
  1. 仿照RxSwift,给str添加前缀,实现:str.pm.numberCount类似的效果
 print(str.pm.numberCount)

扩展.pm属性,然后在PM中扩展方法

extension String {
    var pm: PM {
        //返回PM初始化话,传入当前字符串
        return PM(self)
    }
}

struct PM {
    //属性列表
    var string: String = ""
    //初始化
    init(_ string: String) {
        self.string = string
    }
    //给字符串扩充方法
    var numberCount: Int {
        var count = 0
        for c in self.string where ("0"..."9").contains(c) {
            count += 1
        }
        return count
    }
}

不够通用,给字符串扩充功能,给数组呢,都要添加属性列表、初始化一遍?考虑泛型,扩展方法列表都扩展.pm属性。

struct PM<Base> {
    //属性列表
    var base: Base
    //初始化
    init(_ base: Base) {
        self.base = base
    }
}

extension String {
    var pm: PM<String> {
        PM(self)
    }
}

然后给PM扩展方法

 //扩展 PM(self) base就是字符串,Base是泛型String类型
 extension PM where Base == String {
     var numberCount: Int {
         var count = 0
         for c in base where ("0"..."9").contains(c) {
             count += 1
         }
         return count
     }
 }

这样也可以给自定义的类扩充.pm属性

class PMPerson {}
extension PMPerson {
    var pm: PM<PMPerson> {
        //Base就是PMPerson
        PM(self)
    }
}
 extension PM where Base == PMPerson { //==只是Person类,: Person及其子类
     func test() {
         print("test--")
     }
 }

再进一步优化,不仅给成员属性增加,给类属性也增加.pm属性,实现类似String.pm.numberCount

extension String {
    var pm: PM<String> {
        PM(self)
    }
    static var pm: PM<String>.Type {
        PM<String>.self
    }
}
extension PM where Base == String {
     var numberCount: Int {
         var count = 0
         for c in base where ("0"..."9").contains(c) {
             count += 1
         }
         return count
     }
     static func Stringtest() {
         print("Stringtest")
     }
 }

这样也实现了类属性增加.pm属性,实现String.pm.Stringtest()
这样以后自定义的类都可以像上述那样,增加成员属性、类属性.pm,并扩充方法,如下Dog

class Dog{ }
extension Dog {
    var pm: PM<Dog> {
        PM(self)
    }
    static var pm: PM<Dog>.Type {
        PM<Dog>.self
    }
}
extension PM where Base == Dog {
    func dogf1() {
        
    }
    static func dogf2() {
        
    }
}

后续有想添加.pm属性的,都可以这样扩展。但相同的部分,可用协议优化

protocol PMCompatible {}
extension PMCompatible {
    var pm: PM<Self> {
        set{} //mutating能编译过
        get{PM(self)}
    }
    static var pm: PM<Self>.Type {
        set{}
        get{PM<Self>.self}
    }
}

只要让类遵守该PMCompatible协议,就有.pm属性,继续优化

class Dog: PMCompatible { }
extension PM where Base == Dog {
    mutating func eat() {
        print("eat")
    }
}

不管字符串是String还是NSString/NSMutableString,都遵守该协议

extension String: PMCompatible {}
extension NSString: PMCompatible {}
//扩展PM遵守字面量协议
extension PM where Base: ExpressibleByStringLiteral {
   static var random10String: String {
        let letters = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
        var randomString = ""
        for _ in 0..<10 {
            let ch = letters.randomElement()
            randomString.append(ch!)
        }
        return randomString
    }
}

以上优雅的实现了
String.pm.random10String:随机产生10个字符串
str.pm.numberCount:计算字符串里数字的个数

核心代码

struct PM<Base> {
    var base: Base
    init(_ base: Base) {
        self.base = base
    }
}

protocol PMCompatible {}
extension PMCompatible {
    var pm: PM<Self> {
        set{} //mutating能编译过
        get{PM(self)}
    }
    static var pm: PM<Self>.Type {
        set{}
        get{PM<Self>.self}
    }
}

最后,要给哪个类/结构体/枚举如XXX扩展.pm属性,只需遵守PMCompatible协议并扩充extension PM where Base == XXX即可

struct XXX { }
extension XXX: PMCompatible { }
extension PM where Base == XXX {
  //扩充方法或属性
  func test1() {
  }
  static func test1 {
  }
  
}
//XXX.pm.test1()
//XXX().pm.test1()

扫描二维码推送至手机访问。

版权声明:本文由西安泽虎代运营发布,如需转载请注明出处。

转载请注明出处https://www.0291.com.cn/post/56861.html

相关文章

财富观影响人生幸福吗?

财富观影响人生幸福吗?

我们无论找什么理由都很难否定一个基本的人生道理,即追求“财富自由”是我们一生中的基本需求,我们绝大多数人一生中的主要时间都在做这件事。人生教育的关键——财商教育,它的意义远远超出了怎样挣钱或花钱的范畴。要特别强调的是财商教育不是理财教育,财商教育包含了理财教育。正确的培训和...

小编教你移动互联网迅速发展,pc端还重要吗(近年来随着移动互联网的迅速发展)

小编教你移动互联网迅速发展,pc端还重要吗(近年来随着移动互联网的迅速发展)

尽管移动兴起之后,PC端的使用频次、关注度大打折扣,但是并不代表我们在移动互联网时代就应该放弃PC端渠道的价值。在PC上还是有很多有价值的营销渠道可以为我们所用。如搜索引擎优化,除了投放百度等搜索广告之外,我们将自己的官网进行优化,大量的生产和投放内容在互联网上都是可以带来搜索流量;如邮件营销(E...

信息流广告着陆页策划技巧!

信息流广告着陆页策划技巧!

着陆页策划对于优化师而言,一直都是令人头疼的事情,而且特别考验我们对于用户需求的把握。 所谓成也着陆页,败也着陆页。没转化、对话成本高等等,大部分皆由着陆页而起。 同样身为优化师,为什么别人家的着陆页转化效果那么好,而我的就那么差呢? 所以,今天我们就来着重探讨下:如何通过着陆页...

微信公众号又有重大改版了

微信公众号又有重大改版了

公众号又有重大改版了,与过往微信改版的气氛不太相符的是,这两天各种新媒体群,有些鸦雀无声。 一位多年活跃在公众号头部的朋友说,如果不是我问起,他还不知道这回事,“怎么没看到人讨论呢”?还有同行看到消息,又开始担心打开率了…… 按理说,公众号的改版总能在圈内引起热烈讨论,微信一...

了解客户真实需求,是全网营销推广的第一步

了解客户真实需求,是全网营销推广的第一步

我们做也好,做也好,最终的目的都是为了让消费者买单,也就是我们常说的获取线索。那么,“为什么消费者会购买我们的产品”、“为什么消费者会在众多产品中选择我们的产品”、“为什么消费者在购买过程中会犹豫”……我们今天就来学习一下“影响消费者购买决策的内外部因素”,从根本找到原因,...

餐饮小店不花一分钱,只用了3个绝妙的营销手段,年入百万

餐饮小店不花一分钱,只用了3个绝妙的营销手段,年入百万

做生意,不管是开实体店,还是摆小摊,总会面临竞争大、难的问题,想要生意过得去,少不了手段!下面通过介绍一个实际案例,教大家几个实用的营销方法。何老板开了一家串串店,店铺位于重庆某大学附近,租金成本、装修成本、材料成本等各种开业成本总共花了10万块钱。不到一个月,她就把成本都...

现在,非常期待与您的又一次邂逅

我们努力让每一部企业宣传片和抖音短视频成为商业大片