0 ile 1 arasında hızlı rastgele kayan


82

Swift'de, 0 ile 1 arasında rastgele bir kayan nokta elde etmeye çalışıyorum, ancak tür dönüşümlerinin çalışmasını sağlayamıyorum.

func randomCGFloat() -> CGFloat {
    return CGFloat(arc4random()) / UINT32_MAX
}

Bir 'CGFloat' alıyorum, 'UInt8' hatasına dönüştürülemiyor

Xcode 6 çalıştırılıyor.


Yanıtlar:


103

Böleni bir kayan nokta olarak başlatmayı deneyin, a la:

CGFloat(Float(arc4random()) / Float(UINT32_MAX))

Teşekkürler, paydayı CGFloat ile yayınlamayı denedim ve bu işe yaramadı. Ama önce bir şamandıraya, sonra bir CGFloat'a döküm işe yarıyor gibi görünüyor.
Joe_Schmoe

2
Geçmeye gerek yok Float- sadece CGFloat(arc4random()) / CGFloat(UInt32.max).
Hamish

@Joe_Schmoe Float / CGFloat'tan geçmezseniz, bölme büyük bir tam sayıya bölünen küçük bir tam sayıdır ve her zaman 0 döndürür.
Teng L

4
ama hızlı 4.2'de sadece şunu kullanın: Float.random (içinde: 0 .. <1)
Andrew Paul Simmons

116

Bu için uzantısıdır rastgele sayılar arasında Int, Çift, Float CGFloat

Swift 3 & 4 & 5 sözdizimi

import Foundation
import CoreGraphics

// MARK: Int Extension

public extension Int {

    /// Returns a random Int point number between 0 and Int.max.
    static var random: Int {
        return Int.random(n: Int.max)
    }

    /// Random integer between 0 and n-1.
    ///
    /// - Parameter n:  Interval max
    /// - Returns:      Returns a random Int point number between 0 and n max
    static func random(n: Int) -> Int {
        return Int(arc4random_uniform(UInt32(n)))
    }

    ///  Random integer between min and max
    ///
    /// - Parameters:
    ///   - min:    Interval minimun
    ///   - max:    Interval max
    /// - Returns:  Returns a random Int point number between 0 and n max
    static func random(min: Int, max: Int) -> Int {
        return Int.random(n: max - min + 1) + min

    }
}

// MARK: Double Extension

public extension Double {

    /// Returns a random floating point number between 0.0 and 1.0, inclusive.
    static var random: Double {
        return Double(arc4random()) / 0xFFFFFFFF
    }

    /// Random double between 0 and n-1.
    ///
    /// - Parameter n:  Interval max
    /// - Returns:      Returns a random double point number between 0 and n max
    static func random(min: Double, max: Double) -> Double {
        return Double.random * (max - min) + min
    }
}

// MARK: Float Extension

public extension Float {

    /// Returns a random floating point number between 0.0 and 1.0, inclusive.
    static var random: Float {
        return Float(arc4random()) / 0xFFFFFFFF
    }

    /// Random float between 0 and n-1.
    ///
    /// - Parameter n:  Interval max
    /// - Returns:      Returns a random float point number between 0 and n max
    static func random(min: Float, max: Float) -> Float {
        return Float.random * (max - min) + min
    }
}

// MARK: CGFloat Extension

public extension CGFloat {

    /// Randomly returns either 1.0 or -1.0.
    static var randomSign: CGFloat {
        return (arc4random_uniform(2) == 0) ? 1.0 : -1.0
    }

    /// Returns a random floating point number between 0.0 and 1.0, inclusive.
    static var random: CGFloat {
        return CGFloat(Float.random)
    }

    /// Random CGFloat between 0 and n-1.
    ///
    /// - Parameter n:  Interval max
    /// - Returns:      Returns a random CGFloat point number between 0 and n max
    static func random(min: CGFloat, max: CGFloat) -> CGFloat {
        return CGFloat.random * (max - min) + min
    }
}

Kullanım:

let randomNumDouble  = Double.random(min: 0.00, max: 23.50)
let randomNumInt     = Int.random(min: 56, max: 992)
let randomNumFloat   = Float.random(min: 6.98, max: 923.09)
let randomNumCGFloat = CGFloat.random(min: 6.98, max: 923.09)

1
@DaRk -_- D0G Matematiği anlamaya çalışıyorum. Float ve Double için random () için neden 0xFFFFFFFF ile bölüyorsunuz?
Kristal

2
@EthanMick, Swift 2 görünümü için Güncelleme ekleyin;)
YannSteph

1
CGFloat.random (min: 0.1, maks: 10.1) kullanmaya çalışırken "işlev olmayan 'CGFloat' türünün değeri çağrılamıyor" hatası görünüyor
Andrew

1
Swift 4 için güncelleme gerekmiyor.
RyuX51

1
@YannickSteph ile Swift 5 ve Xcode 10.2, çizgi return Float(arc4random()) / 0xFFFFFFFFartık bir uyarı verir: '4294967295' is not exactly representable as 'Float'; it becomes '4294967296'. Bu uyarının nasıl çözüleceğine dair bir fikriniz var mı? Bunun Float(UInt32.max)yerine bölmeyi düşünüyorum .
barış tipi


26

Sandy Chapman'ın Swift 3 için cevabını güncelleme :

extension ClosedRange where Bound : FloatingPoint {
    public func random() -> Bound {
        let range = self.upperBound - self.lowerBound
        let randomValue = (Bound(arc4random_uniform(UINT32_MAX)) / Bound(UINT32_MAX)) * range + self.lowerBound
        return randomValue
    }
}

Şimdi gibi şeyler söyleyebilirsin (-1.0...1.0).random().

DÜZENLE Bugün düşünüyorum (Swift 4) Şöyle bir şey yazacağım:

extension ClosedRange where Bound : FloatingPoint {
    public func random() -> Bound {
        let max = UInt32.max
        return
            Bound(arc4random_uniform(max)) /
            Bound(max) *
            (upperBound - lowerBound) +
            lowerBound
    }
}

NOT Swift 4.2, yerel rastgele sayı oluşturma özelliğini sunar ve bunların tümü tartışmalı hale gelir.


16

Burada çerçeve, Swift'de rastgele sayı verileri oluşturmada iyi bir iş çıkarıyor: https://github.com/thellimist/SwiftRandom/blob/master/SwiftRandom/Randoms.swift

public extension Int {
    /// SwiftRandom extension
    public static func random(lower: Int = 0, _ upper: Int = 100) -> Int {
        return lower + Int(arc4random_uniform(UInt32(upper - lower + 1)))
    }
}

public extension Double {
    /// SwiftRandom extension
    public static func random(lower: Double = 0, _ upper: Double = 100) -> Double {
        return (Double(arc4random()) / 0xFFFFFFFF) * (upper - lower) + lower
    }
}

public extension Float {
    /// SwiftRandom extension
    public static func random(lower: Float = 0, _ upper: Float = 100) -> Float {
        return (Float(arc4random()) / 0xFFFFFFFF) * (upper - lower) + lower
    }
}

public extension CGFloat {
    /// SwiftRandom extension
    public static func random(lower: CGFloat = 0, _ upper: CGFloat = 1) -> CGFloat {
        return CGFloat(Float(arc4random()) / Float(UINT32_MAX)) * (upper - lower) + lower
    }
}

Bu, için derlerken çalışmaz Generic iOS device. 0xffffffff bir Int içine sığmaz. Bunun yerine bunu deneyin CGFloat (UInt32.max)
neoneye

13

Aşağıda, IntervalTypebunu yapma türüne ilişkin bir uzantı verilmiştir :

extension IntervalType {
    public func random() -> Bound {
        let range = (self.end as! Double) - (self.start as! Double)
        let randomValue = (Double(arc4random_uniform(UINT32_MAX)) / Double(UINT32_MAX)) * range + (self.start as! Double)
        return randomValue as! Bound
    }
}

Bu uzantı ile, bir aralık oluşturmak ve ardından bu aralıkta rastgele bir değer elde etmek için aralık sözdizimini kullanabilirsiniz:

(0.0...1.0).random()

İlave

Aynı şeyi Ints için yapmak istiyorsanız, CollectionTypeprotokolde aşağıdaki uzantıyı kullanabilirsiniz :

extension CollectionType {
    public func random() -> Self._Element {
        if let startIndex = self.startIndex as? Int {
            let start = UInt32(startIndex)
            let end = UInt32(self.endIndex as! Int)
            return self[Int(arc4random_uniform(end - start) + start) as! Self.Index]
        }
        var generator = self.generate()
        var count = arc4random_uniform(UInt32(self.count as! Int))
        while count > 0 {
            generator.next()
            count = count - 1
        }
        return generator.next() as! Self._Element
    }
}

Ints IntervalType. Onlar kullanmak Rangeyerine. Bunu CollectionTypetür üzerinde yapmanın yararı, otomatik olarak Dictionaryve Arraytürlerine taşınmasıdır .

Örnekler:

(0...10).random()               // Ex: 6
["A", "B", "C"].random()        // Ex: "B"
["X":1, "Y":2, "Z":3].random()  // Ex: (.0: "Y", .1: 2)

bunun Ints için de çalışmasını nasıl sağlayacağız, orijinal soru değil, ancak IntervalType
Joe_Schmoe

1
@Joe_Schmoe: Yanıtımı s, s ve s kullanımıyla .random()ilgili ayrıntılarla güncelledimIntDictionaryArray
Sandy Chapman

Bu çözümü gerçekten beğendim, ancak daha iyi genel uygulamaları dahil etmek için güncelleme
yapıyorum

1
Broken in Swift 3.
matt


6

Jmduke'nin önerdiği, işlevde küçük bir değişiklikle Playground'da çalışıyor gibi görünüyor:

func randomCGFloat() -> Float {
    return Float(arc4random()) /  Float(UInt32.max)
}

ve swift belgeden ve drewag: tür dönüştürmenin açık olması gerektiğinin nedeni, belgedeki örnek şöyledir:

let three = 3
let pointOneFourOneFiveNine = 0.14159
let pi = Double(three) + pointOneFourOneFiveNine
// pi equals 3.14159, and is inferred to be of type Double

2
drand48()

[Çift] 'e ihtiyacınız olması durumunda. 0 ile 1 arasındadır. Hepsi bu.


1

YannickSteph'in cevabına göre

Herhangi kayan nokta türü için o işi yapmak için, gibi Double, Float, CGFloat, vb sizin için bir uzantı yapabilir BinaryFloatingPointtürü:

extension BinaryFloatingPoint {

    /// Returns a random floating point number between 0.0 and 1.0, inclusive.
    public static var random: Self {
        return Self(arc4random()) / 0xFFFFFFFF
    }

    /// Random double between 0 and n-1.
    ///
    /// - Parameter n:  Interval max
    /// - Returns:      Returns a random double point number between 0 and n max
    public static func random(min: Self, max: Self) -> Self {
        return Self.random * (max - min) + min
    }
}

1

Detaylar

Xcode: 9.2, Swift 4

Çözüm

extension BinaryInteger {

    static func rand(_ min: Self, _ max: Self) -> Self {
        let _min = min
        let difference = max+1 - _min
        return Self(arc4random_uniform(UInt32(difference))) + _min
    }
}

extension BinaryFloatingPoint {

    private func toInt() -> Int {
        // https://stackoverflow.com/q/49325962/4488252
        if let value = self as? CGFloat {
            return Int(value)
        }
        return Int(self)
    }

    static func rand(_ min: Self, _ max: Self, precision: Int) -> Self {

        if precision == 0 {
            let min = min.rounded(.down).toInt()
            let max = max.rounded(.down).toInt()
            return Self(Int.rand(min, max))
        }

        let delta = max - min
        let maxFloatPart = Self(pow(10.0, Double(precision)))
        let maxIntegerPart = (delta * maxFloatPart).rounded(.down).toInt()
        let randomValue = Int.rand(0, maxIntegerPart)
        let result = min + Self(randomValue)/maxFloatPart
        return Self((result*maxFloatPart).toInt())/maxFloatPart
    }
}

Kullanım

print("\(Int.rand(1, 20))")
print("\(Float.rand(5.231233, 44.5, precision: 3))")
print("\(Double.rand(5.231233, 44.5, precision: 4))")
print("\(CGFloat.rand(5.231233, 44.5, precision: 6))")

Tam Örnek

import Foundation
import CoreGraphics

func run() {
    let min = 2.38945
    let max = 2.39865
    for _ in 0...100 {
        let precision = Int.rand(0, 5)
        print("Precision: \(precision)")
        floatSample(min: Float(min), max: Float(max), precision: precision)
        floatSample(min: Double(min), max: Double(max), precision: precision)
        floatSample(min: CGFloat(min), max: CGFloat(max), precision: precision)
        intSample(min: Int(1), max: Int(10000))
        print("")
    }
}

private func printResult<T: Comparable>(min: T, max: T, random: T) {
    let result = "\(T.self) rand[\(min), \(max)] = \(random)"
    print(result)
}

func floatSample<T: BinaryFloatingPoint>(min: T, max: T, precision: Int) {
    printResult(min: min, max: max, random: T.rand(min, max, precision: precision))
}

func intSample<T: BinaryInteger>(min: T, max: T) {
    printResult(min: min, max: max, random: T.rand(min, max))
}

Sonuçlar

görüntü açıklamasını buraya girin

Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.