Dans Swift, j'essaie d'obtenir un float aléatoire compris entre 0 et 1, mais je n'arrive pas à faire en sorte que les conversions de types fonctionnent.
func randomCGFloat() -> CGFloat {
return CGFloat(arc4random()) / UINT32_MAX
}
Je reçois un message 'CGFloat' qui n'est pas convertible en erreur 'UInt8'
Exécution de Xcode 6.
Essayez également d’initialiser le diviseur en tant que float, à la:
CGFloat(Float(arc4random()) / Float(UINT32_MAX))
C'est une extension pour nombres aléatoires de Int, Double, Float, CGFloat
syntaxe Swift 3 & 4
import Foundation
import CoreGraphics
// MARK: Int Extension
public extension Int {
/// Returns a random Int point number between 0 and Int.max.
public 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
public 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
public 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.
public 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
public 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.
public 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
public 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.
public 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.
public 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
public static func random(min: CGFloat, max: CGFloat) -> CGFloat {
return CGFloat.random * (max - min) + min
}
}
tilisez:
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)
Mise à jour réponse de Sandy Chapman pour Swift 3:
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
}
}
Maintenant, vous pouvez dire des choses comme (-1.0...1.0).random()
.
[~ # ~] éditer [~ # ~] Je pense qu'aujourd'hui (Swift 4), j'écrirais quelque chose comme ça:
extension ClosedRange where Bound : FloatingPoint {
public func random() -> Bound {
let max = UInt32.max
return
Bound(arc4random_uniform(max)) /
Bound(max) *
(upperBound - lowerBound) +
lowerBound
}
}
[~ # ~] note [~ # ~] Swift 4.2 introduit la génération de nombres aléatoires natifs et tout cela devient discutable.
Swift 4.2:
let randomFloat = Float.random(in: 0..<1)
Ici, le cadre fait un bon travail pour générer des données de nombres aléatoires dans Swift: 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
}
}
Vous trouverez ci-dessous une extension du type IntervalType
:
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
}
}
Avec cette extension, vous pouvez utiliser la syntaxe d'intervalle pour générer un intervalle, puis obtenir une valeur aléatoire dans cet intervalle:
(0.0...1.0).random()
Ajout
Si vous souhaitez faire de même pour Int
s, vous pouvez utiliser l'extension suivante sur le protocole CollectionType
:
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
}
}
Int
s n'utilise pas le IntervalType
. Ils utilisent Range
à la place. L'avantage de faire cela sur le type CollectionType
est qu'il est automatiquement reporté sur les types Dictionary
et Array
.
Exemples:
(0...10).random() // Ex: 6
["A", "B", "C"].random() // Ex: "B"
["X":1, "Y":2, "Z":3].random() // Ex: (.0: "Y", .1: 2)
Ce que jmduke a suggéré semble fonctionner dans Playground avec un léger changement dans la fonction:
func randomCGFloat() -> Float {
return Float(arc4random()) / Float(UInt32.max)
}
et la raison pour laquelle à partir de la doc Swift et comme indiqué par drewag: type, la conversion doit être explicite, l'exemple dans la doc est:
let three = 3
let pointOneFourOneFiveNine = 0.14159
let pi = Double(three) + pointOneFourOneFiveNine
// pi equals 3.14159, and is inferred to be of type Double
drand48()
Au cas où vous auriez besoin de [Double]. Entre 0 et 1. C'est tout.
Swift 5
let randomFloat = CGFloat.random(in: 0...1)
Basé sur la réponse de YannickSteph
Pour que cela fonctionne pour n'importe quel type à virgule flottante, comme Double
, Float
, CGFloat
, etc., vous pouvez créer une extension pour le type BinaryFloatingPoint
:
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
}
}
Xcode: 9.2, Swift 4
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
}
}
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))")
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))
}