美文网首页
[Swift5.1] 17-高级运算符

[Swift5.1] 17-高级运算符

作者: codeTao | 来源:发表于2020-06-02 14:44 被阅读0次

溢出运算符(Overflow Operator)

  • Swift的算数运算符出现溢出时会抛出运行时错误
  • Swift有溢出运算符(&+、&-、&*),用来支持溢出运算
    溢出时仍在范围内,超出范围从头开始
var min = UInt8.min
print(min &- 1) // 255, Int8.max

var max = UInt8.max
print(max &+ 1) // 0, Int8.min
print(max &* 2) // 254, 等价于 max &+ max

运算符重载(Operator Overload)

1)运算符重载定义

  • 类、结构体、枚举可以为现有的运算符提供自定义的实现,这个操作叫做:运算符重载
struct Point {
    var x: Int, y: Int
}

func + (p1: Point, p2: Point) -> Point {
    Point(x: p1.x + p2.x, y: p1.y + p2.y)
}

let p = Point(x: 10, y: 20) + Point(x: 11, y: 22)
print(p) // Point(x: 21, y: 42)
struct Point {
    var x: Int, y: Int
    static func + (p1: Point, p2: Point) -> Point {
        Point(x: p1.x + p2.x, y: p1.y + p2.y)
    }
}

2)运算符重载前缀和后缀

运算符重载默认中缀, 可以用prefix 设置前缀运算符重载, 用postfix设置后缀运算符重载

static func + (p1: Point, p2: Point) -> Point {
    Point(x: p1.x + p2.x, y: p1.y + p2.y)
}
static func - (p1: Point, p2: Point) -> Point {
    Point(x: p1.x - p2.x, y: p1.y - p2.y)
}

运算符重载:取反

static prefix func - (p: Point) -> Point {
    Point(x: -p.x, y: -p.y)
}

运算符重载:+=

static func += (p1: inout Point, p2: Point) {
    p1 = p1 + p2
}

运算符重载:==

static func == (p1: Point, p2: Point) -> Bool {
    (p1.x == p2.x) && (p1.y == p2.y)
}

运算符重载:++

static prefix func ++ (p: inout Point) -> Point {
    p += Point(x: 1, y: 1)
    return p
}

static postfix func ++ (p: inout Point) -> Point {
    let tmp = p
    p += Point(x: 1, y: 1)
    return tmp
}

Equatable

  • 要想得知2个实例是否等价,一般做法是遵守 Equatable 协议,重载 == 运算符
  • 与此同时,等价于重载了 != 运算符
struct Point : Equatable {
    var x: Int, y: Int
}
var p1 = Point(x: 10, y: 20)
var p2 = Point(x: 11, y: 22)
print(p1 == p2) // false
print(p1 != p2) // true

Swift为以下类型提供默认的 Equatable 实现:

  • 没有关联类型的枚举 (示例一)
  • 只拥有遵守 Equatable 协议关联类型的枚举 (示例二)
  • 只拥有遵守 Equatable 协议存储属性的结构体
//示例一
enum Answer {
    case wrong
    case right
}
var s1 = Answer.wrong
var s2 = Answer.right
print(s1 == s2) // false
//示例二
enum Answer : Equatable{
    case wrong(Int)
    case right
}
var s1 = Answer.wrong(11)
var s2 = Answer.wrong(22)
print(s1 == s2)  //false
  • 引用类型比较存储的地址值是否相等(是否引用着同一个对象),使用恒等运算符 === 、!==

Comparable

要想比较2个实例的大小,一般做法是:

  • 遵守 Comparable 协议
  • 重载相应的运算符
// score大的比较大,若score相等,age小的比较大
struct Student : Comparable {
    var age: Int
    var score: Int
    init(score: Int, age: Int) {
        self.score = score
        self.age = age
    }
    static func < (lhs: Student, rhs: Student) -> Bool {
        (lhs.score < rhs.score)
            || (lhs.score == rhs.score && lhs.age > rhs.age)
    }
    static func > (lhs: Student, rhs: Student) -> Bool {
        (lhs.score > rhs.score)
            || (lhs.score == rhs.score && lhs.age < rhs.age)
    }
    static func <= (lhs: Student, rhs: Student) -> Bool {
        !(lhs > rhs)
    }
    static func >= (lhs: Student, rhs: Student) -> Bool {
        !(lhs < rhs)
    }
}
var stu3 = Student(score: 100, age: 20)
print(stu1 > stu2) // true
print(stu1 >= stu2) // true
print(stu1 >= stu3) // true
print(stu1 <= stu3) // true
print(stu2 < stu1) // true
print(stu2 <= stu1) // true

自定义运算符(Custom Operator)

可以自定义新的运算符:在全局作用域使用operator进行声明.

  • prefix operator 前缀运算符
  • postfix operator后缀运算符
  • infix operator 中缀运算符 : 优先级组
precedencegroup 优先级组 {
    associativity: 结合性(left\right\none)
    higherThan: 比谁的优先级高
    lowerThan: 比谁的优先级低
    assignment: true  代表在可选链操作中拥有跟赋值运算符一样的优先级
}
prefix operator +++
infix operator +- : PlusMinusPrecedence
precedencegroup PlusMinusPrecedence {
    associativity: none
    higherThan: AdditionPrecedence
    lowerThan: MultiplicationPrecedence
    assignment: true
}

Apple文档参考:
Operator Declarations-官方文档
Operator Declaration-Swift文档说明

自定义运算符示例

struct Point {
    var x: Int, y: Int
    static prefix func +++ (point: inout Point) -> Point {
        point = Point(x: point.x + point.x, y: point.y + point.y)
        return point
    }
    static func +- (left: Point, right: Point) -> Point {
        return Point(x: left.x + right.x, y: left.y - right.y)
    }
    static func +- (left: Point?, right: Point) -> Point {
        print("+-")
        return Point(x: left?.x ?? 0 + right.x, y: left?.y ?? 0 - right.y)
    }
}
struct Person {
    var point: Point
}
var person: Person? = nil
person?.point +- Point(x: 10, y: 20)

相关文章

  • [Swift5.1] 17-高级运算符

    溢出运算符(Overflow Operator) Swift的算数运算符出现溢出时会抛出运行时错误 Swift有溢...

  • 17-高级运算符

  • Swift5.1学习随笔之高级运算符

    溢出运算符(Overflow Operator) Swift的算数运算符出现溢出时会抛出运行时错误 首先Int8的...

  • 第二十四节 Swift 高级运算符

    除了加减乘除运算符,常用的位运算符在 Swift里属于高级运算符。 其它的高级运算符还有移位运算符和溢出加法运算符...

  • 26_高级运算符

    除了基本运算符,Swift 中还有许多可以对数值进行复杂运算的高级运算符。这些高级运算符包含了在位运算符和移位运算...

  • Swift编程二十八(高级运算符)

    案例代码下载 高级运算符 除了基本运算符中描述的运算符之外,Swift还提供了几个执行更复杂值操作的高级运算符。这...

  • swift2.2 - 高级运算符

    高级运算符 文档地址 作为 基本运算符 的补充,Swift 提供了几个高级运算符执行对数传值进行更加复杂的操作。这...

  • 26 高级运算符

    高级运算符 除了Basic Operators中描述的基本运算符之外,Swift还提供了几个执行更复杂值操作的高级...

  • 27、【Swift】高级运算符 - Advanced Opera

    Swift 运算符基本运算符高级运算符(包括 C 或 Objective-C 所有按位和移位运算符。) 与 C 的...

  • Swift5.1基本运算符

    2.基本运算符 求余运算符:在对负数 b 求余时, b 的符号会被忽略略。这意味着 a % b 和 a % -b ...

网友评论

      本文标题:[Swift5.1] 17-高级运算符

      本文链接:https://www.haomeiwen.com/subject/zdidzhtx.html