Diferença entre receptor de ponteiro e receptor de valor na implementação com a estrutura Iris

Estou estudando o framework Iris recentemente. Encontrei uma pergunta quando estava implementando o manipulador. Como a seguir:

package controller

import "github.com/kataras/iris"    

type Pages struct{
}

func (p *Pages) Serve(c *iris.Context) {
}

Para usar este controlador, implementei o seguinte script de entrada:

package main

import (
     "github.com/kataras/iris"
     "web/controller"
)

func main(){
      ctrl := controller.Pages{}
      iris.Handle("GET", "/", ctrl)
      iris.Listen(":8080")
}

Mas quando compilei o código, recebi a seguinte mensagem de erro:

cannot use ctrl (type controllers.Pages) as type iris.Handler in argument to iris.Handle:
controllers.Pages does not implement iris.Handler (Serve method has pointer receiver)

Depois de alterar a declaração para:

ctrl := &controller.Pages{}

Então o compilador passou sem nenhuma reclamação.

A pergunta é: eu pensei que as seguintes instruções são iguais, já que o compilador GO fará a conversão sob a tabela:

type Pages struct {
}

func (p *Pages) goWithPointer() {
    fmt.Println("goWithPointer")
}

func (p Pages) goWithValue() {
    fmt.Println("goWithValue")
}

func main() {
    p1 := Pages{}
    p2 := &Pages{}

    p1.goWithPointer()
    p1.goWithValue()

    p2.goWithPointer()
    p2.goWithValue()
}

Por que não posso usarctrl := controller.Pages{} como o parâmetro parairis.Handle(), ao invés dectrl := &controller.Pages{} como o parâmetro parairis.Handle()?

Obrigado pelo seu tempo e compartilhamento.

questionAnswers(2)

yourAnswerToTheQuestion