Como se menciona en yo.ian.g
la respuesta de. Go 1.8 ha incluido esta funcionalidad en la biblioteca estándar.
Ejemplo mínimo para para Go 1.8+
:
server := &http.Server{Addr: ":8080", Handler: handler}
go func() {
if err := server.ListenAndServe(); err != nil {
}
}()
stop := make(chan os.Signal, 1)
signal.Notify(stop, os.Interrupt)
<-stop
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
if err := server.Shutdown(ctx); err != nil {
}
Respuesta original - Pre Go 1.8:
Sobre la base de la respuesta de Uvelichitel .
Puede crear su propia versión de la ListenAndServe
que devuelve un io.Closer
y no bloquea.
func ListenAndServeWithClose(addr string, handler http.Handler) (io.Closer,error) {
var (
listener net.Listener
srvCloser io.Closer
err error
)
srv := &http.Server{Addr: addr, Handler: handler}
if addr == "" {
addr = ":http"
}
listener, err = net.Listen("tcp", addr)
if err != nil {
return nil, err
}
go func() {
err := srv.Serve(tcpKeepAliveListener{listener.(*net.TCPListener)})
if err != nil {
log.Println("HTTP Server Error - ", err)
}
}()
srvCloser = listener
return srvCloser, nil
}
Código completo disponible aquí .
El servidor HTTP se cerrará con el error
accept tcp [::]:8080: use of closed network connection