[swift-users] Subclass instance passed to closure expecting parent class “cannot convert value of type to expected argument type”

Mark Lacey mark.lacey at apple.com
Wed Sep 28 22:12:32 CDT 2016


> On Sep 28, 2016, at 5:43 PM, Eric Conner via swift-users <swift-users at swift.org> wrote:
> 
> Hello all!
> 
> I hope it's not frowned upon to repost a question here from Stack Overflow, but I have not found help there.  In any case, the email gets some nice formatting out of it :-D.
> 
> I am trying to create a subclass with a generic type that can be passed into a completion block, but am getting the following error:
> 
> Cannot convert value of type '(ModelListResponse) -> ()' to expected argument type '(APIResponse) -> ()'
> Parent class:
> 
> import ObjectMapper
> 
> class APIResponse: Mappable {
>     var status: String!
>     var message: String?
> 
>     required init?(map: Map) {}
> 
>     func mapping(map: Map) {
>         status <- map["status"]
>         message <- map["message"]
>     }
> }
> Subclass:
> 
> import ObjectMapper
> 
> class ModelListResponse<T: Mappable>: APIResponse {
>     var data: ModelList<T>?
> 
>     required init?(map: Map) {
>         super.init(map: map)
>     }
> 
>     override func mapping(map: Map) {
>         super.mapping(map: map)
>         data <- map["data"]
>     }
> }
> Then I have a function definition:
> 
>  func _GET(path: String, parameters: Parameters, completion:@escaping (APIResponse) -> ())
> And I'm trying to call it like so:
> 
> // error is on this line
> self._GET(path: "/rest/posts", parameters: parameters) { (response: ModelListResponse<Post>) in
>     // ...
> }
> But here I get the error about how ModelListResponse cannot be converted. Isn't a ModelListResponse an APIResponse?  I don't understand why this does not work.
> 
> 

Below is a simplified version of what you’re attempting to do, which will hopefully shed some light. 

In your example, you’ve got a function, _GET() which expects to be passed a function that can take an APIReponse (which it could then invoke methods on). You’re instead passing in a function that can take a ModelListResponse<Post>, and that function would expect to be able to invoke methods on a ModelListResponse<Post>, but the caller (_GET()), might pass in an APIRespons that is not a ModelListResponse<Post>.

Mark


class A {}
class B : A {
  func gotcha() {}
}

func apply(_ f: (A) -> (), _ a: A) { f(a) }

func processB(_ b: B) -> () { b.gotcha() }

apply(processB, A()) // cannot convert value of type '(B) -> ()' to expected argument type '(A) -> ()'





> I managed to get around the issue by using a generic. Though I'm not sure why this works and the above code does not. Here is my working declaration:
> 
> func _GET<T: Mappable>(path: String, parameters: Parameters, completion:@escaping (T) -> ()) { }
> Thanks!
> 
> _______________________________________________
> swift-users mailing list
> swift-users at swift.org
> https://lists.swift.org/mailman/listinfo/swift-users

-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://lists.swift.org/pipermail/swift-users/attachments/20160928/a0cff293/attachment.html>


More information about the swift-users mailing list