Correct way to find max in an Array in Swift

前端 未结 12 1697
一整个雨季
一整个雨季 2020-11-29 18:05

I\'ve so far got a simple (but potentially expensive) way:

var myMax = sort(myArray,>)[0]

And how I was taught to do it at school:

相关标签:
12条回答
  • 2020-11-29 18:34

    The other answers are all correct, but don't forget you could also use collection operators, as follows:

    var list = [1, 2, 3, 4]
    var max: Int = (list as AnyObject).valueForKeyPath("@max.self") as Int
    

    you can also find the average in the same way:

    var avg: Double = (list as AnyObject).valueForKeyPath("@avg.self") as Double
    

    This syntax might be less clear than some of the other solutions, but it's interesting to see that -valueForKeyPath: can still be used :)

    0 讨论(0)
  • 2020-11-29 18:35

    Update: This should probably be the accepted answer since maxElement appeared in Swift.


    Use the almighty reduce:

    let nums = [1, 6, 3, 9, 4, 6];
    let numMax = nums.reduce(Int.min, { max($0, $1) })
    

    Similarly:

    let numMin = nums.reduce(Int.max, { min($0, $1) })
    

    reduce takes a first value that is the initial value for an internal accumulator variable, then applies the passed function (here, it's anonymous) to the accumulator and each element of the array successively, and stores the new value in the accumulator. The last accumulator value is then returned.

    0 讨论(0)
  • 2020-11-29 18:36

    With Swift 1.2 (and maybe earlier) you now need to use:

    let nums = [1, 6, 3, 9, 4, 6];
    let numMax = nums.reduce(Int.min, combine: { max($0, $1) })
    

    For working with Double values I used something like this:

    let nums = [1.3, 6.2, 3.6, 9.7, 4.9, 6.3];
    let numMax = nums.reduce(-Double.infinity, combine: { max($0, $1) })
    
    0 讨论(0)
  • 2020-11-29 18:37

    With Swift 5, Array, like other Sequence Protocol conforming objects (Dictionary, Set, etc), has two methods called max() and max(by:) that return the maximum element in the sequence or nil if the sequence is empty.


    #1. Using Array's max() method

    If the element type inside your sequence conforms to Comparable protocol (may it be String, Float, Character or one of your custom class or struct), you will be able to use max() that has the following declaration:

    @warn_unqualified_access func max() -> Element?
    

    Returns the maximum element in the sequence.

    The following Playground codes show to use max():

    let intMax = [12, 15, 6].max()
    let stringMax = ["bike", "car", "boat"].max()
    
    print(String(describing: intMax)) // prints: Optional(15)
    print(String(describing: stringMax)) // prints: Optional("car")
    
    class Route: Comparable, CustomStringConvertible {
    
        let distance: Int
        var description: String { return "Route with distance: \(distance)" }
    
        init(distance: Int) {
            self.distance = distance
        }
    
        static func ==(lhs: Route, rhs: Route) -> Bool {
            return lhs.distance == rhs.distance
        }
    
        static func <(lhs: Route, rhs: Route) -> Bool {
            return lhs.distance < rhs.distance
        }
    
    }
    
    let routes = [
        Route(distance: 20),
        Route(distance: 30),
        Route(distance: 10)
    ]
    
    let maxRoute = routes.max()
    print(String(describing: maxRoute)) // prints: Optional(Route with distance: 30)
    

    #2. Using Array's max(by:) method

    If the element type inside your sequence does not conform to Comparable protocol, you will have to use max(by:) that has the following declaration:

    @warn_unqualified_access func max(by areInIncreasingOrder: (Element, Element) throws -> Bool) rethrows -> Element?
    

    Returns the maximum element in the sequence, using the given predicate as the comparison between elements.

    The following Playground codes show to use max(by:):

    let dictionary = ["Boat" : 15, "Car" : 20, "Bike" : 40]
    
    let keyMaxElement = dictionary.max(by: { (a, b) -> Bool in
        return a.key < b.key
    })
    
    let valueMaxElement = dictionary.max(by: { (a, b) -> Bool in
        return a.value < b.value
    })
    
    print(String(describing: keyMaxElement)) // prints: Optional(("Car", 20))
    print(String(describing: valueMaxElement)) // prints: Optional(("Bike", 40))
    
    class Route: CustomStringConvertible {
    
        let distance: Int
        var description: String { return "Route with distance: \(distance)" }
    
        init(distance: Int) {
            self.distance = distance
        }
    
    }
    
    let routes = [
        Route(distance: 20),
        Route(distance: 30),
        Route(distance: 10)
    ]
    
    let maxRoute = routes.max(by: { (a, b) -> Bool in
        return a.distance < b.distance
    })
    
    print(String(describing: maxRoute)) // prints: Optional(Route with distance: 30)
    
    0 讨论(0)
  • 2020-11-29 18:37

    You can also sort your array and then use array.first or array.last

    0 讨论(0)
  • 2020-11-29 18:39

    Swift 3.0

    You can try this code programmatically.

    func getSmallAndGreatestNumber() -> Void {
    
        let numbers = [145, 206, 116, 809, 540, 176]
        var i = 0
        var largest = numbers[0]
        var small = numbers[0]
        while i < numbers.count{
    
            if (numbers[i] > largest) {
                largest = numbers[i]
            }
            if (numbers[i] < small) {
                small = numbers[i]
            }
            i = i + 1
        }
        print("Maximum Number ====================\(largest)")// 809
        print("Minimum Number ====================\(small)")// 116
    }
    
    0 讨论(0)
提交回复
热议问题