Here’s another fun one: the following type checks despite the literal closure; we’ve constrained the return type of filter() to [Int], and not LazyFilterCollection<Int>, so again the type checker picks the overload that takes the non-escaping closure:
func myFilter(array: [Int], predicate: (Int) -> Bool) -> [Int] { return array.lazy.filter { predicate($0) } } Slava > On Mar 26, 2017, at 1:14 AM, Slava Pestov via swift-users > <swift-users@swift.org> wrote: > > Hi Ray, > > There are two overloads of filter() available on ‘array.lazy’; the version > that takes an escaping closure and returns a LazyFilterCollection, and the > version that takes a non-escaping closure and returns [Int]. > > In the first example, we pick the LazyFilterCollection-returning overload, > because the literal closure { predicate($0) } can be coerced to both an > escaping or a non-escaping closure type, and in the absence of additional > constraints we go with the overload from a concrete type over an overload in > a protocol extension. After the overload has been picked we validate the body > of the closure, and notice that it is invalid because whole the closure is > already known to be @escaping, it references the non-@escaping ‘predicate’. > > In the second example, ‘predicate’ is known to be non-@escaping, which rules > out the first overload completely, so we go with the second overload and > perform a non-lazy filter. > > I would argue this is somewhat confusing, but it might be difficult to change > the overload resolution rules in a way where the first overload is always > chosen. > > Slava > >> On Mar 26, 2017, at 12:13 AM, Ray Fix via swift-users <swift-users@swift.org >> <mailto:swift-users@swift.org>> wrote: >> >> >> Hi, >> >> One of the motivating examples for withoutActuallyEscaping looks like the >> following. This predictably doesn’t work because "use of non-escaping >> parameter 'predicate' may allow it to escape" >> >> func myFilter(array: [Int], predicate: (Int) -> Bool) -> [Int] { >> return Array(array.lazy.filter { predicate($0) }) >> } >> >> The solution is to use withoutActuallyEscaping. This works and produces the >> expected results. >> >> func myFilter(array: [Int], predicate: (Int) -> Bool) -> [Int] { >> return withoutActuallyEscaping(predicate) { predicate in >> Array(array.lazy.filter({predicate($0)})) >> } >> } >> >> What I find puzzling is the below example compiles and runs correctly. It >> seems like it should be the same compiler error as in the first example. >> >> func myFilter(array: [Int], predicate: (Int) -> Bool) -> [Int] { >> return Array(array.lazy.filter(predicate)) >> } >> >> If you understand why this is so, it would be very helpful. >> >> Thank you and best wishes, >> Ray >> _______________________________________________ >> swift-users mailing list >> swift-users@swift.org <mailto:swift-users@swift.org> >> https://lists.swift.org/mailman/listinfo/swift-users > > _______________________________________________ > swift-users mailing list > swift-users@swift.org > https://lists.swift.org/mailman/listinfo/swift-users
_______________________________________________ swift-users mailing list swift-users@swift.org https://lists.swift.org/mailman/listinfo/swift-users