// Transport can also cache connections for future re-use.
 type Transport struct {
        idleMu     sync.Mutex
-       idleConn   map[string][]*persistConn
-       idleConnCh map[string]chan *persistConn
+       idleConn   map[connectMethodKey][]*persistConn
+       idleConnCh map[connectMethodKey]chan *persistConn
        reqMu      sync.Mutex
        reqConn    map[*Request]*persistConn
        altMu      sync.RWMutex
        e.val = ""
 }
 
-func (t *Transport) connectMethodForRequest(treq *transportRequest) (*connectMethod, error) {
-       cm := &connectMethod{
-               targetScheme: treq.URL.Scheme,
-               targetAddr:   canonicalAddr(treq.URL),
-       }
+func (t *Transport) connectMethodForRequest(treq *transportRequest) (cm connectMethod, err error) {
+       cm.targetScheme = treq.URL.Scheme
+       cm.targetAddr = canonicalAddr(treq.URL)
        if t.Proxy != nil {
-               var err error
                cm.proxyURL, err = t.Proxy(treq.Request)
-               if err != nil {
-                       return nil, err
-               }
        }
        return cm, nil
 }
                }
        }
        if t.idleConn == nil {
-               t.idleConn = make(map[string][]*persistConn)
+               t.idleConn = make(map[connectMethodKey][]*persistConn)
        }
        if len(t.idleConn[key]) >= max {
                t.idleMu.Unlock()
 // getIdleConnCh returns a channel to receive and return idle
 // persistent connection for the given connectMethod.
 // It may return nil, if persistent connections are not being used.
-func (t *Transport) getIdleConnCh(cm *connectMethod) chan *persistConn {
+func (t *Transport) getIdleConnCh(cm connectMethod) chan *persistConn {
        if t.DisableKeepAlives {
                return nil
        }
        t.idleMu.Lock()
        defer t.idleMu.Unlock()
        if t.idleConnCh == nil {
-               t.idleConnCh = make(map[string]chan *persistConn)
+               t.idleConnCh = make(map[connectMethodKey]chan *persistConn)
        }
        ch, ok := t.idleConnCh[key]
        if !ok {
        return ch
 }
 
-func (t *Transport) getIdleConn(cm *connectMethod) (pconn *persistConn) {
+func (t *Transport) getIdleConn(cm connectMethod) (pconn *persistConn) {
        key := cm.key()
        t.idleMu.Lock()
        defer t.idleMu.Unlock()
                        // 2 or more cached connections; pop last
                        // TODO: queue?
                        pconn = pconns[len(pconns)-1]
-                       t.idleConn[key] = pconns[0 : len(pconns)-1]
+                       t.idleConn[key] = pconns[:len(pconns)-1]
                }
                if !pconn.isBroken() {
                        return
 // specified in the connectMethod.  This includes doing a proxy CONNECT
 // and/or setting up TLS.  If this doesn't return an error, the persistConn
 // is ready to write requests to.
-func (t *Transport) getConn(cm *connectMethod) (*persistConn, error) {
+func (t *Transport) getConn(cm connectMethod) (*persistConn, error) {
        if pc := t.getIdleConn(cm); pc != nil {
                return pc, nil
        }
        }
 }
 
-func (t *Transport) dialConn(cm *connectMethod) (*persistConn, error) {
+func (t *Transport) dialConn(cm connectMethod) (*persistConn, error) {
        conn, err := t.dial("tcp", cm.addr())
        if err != nil {
                if cm.proxyURL != nil {
        targetAddr   string   // Not used if proxy + http targetScheme (4th example in table)
 }
 
-func (ck *connectMethod) key() string {
-       return ck.String() // TODO: use a struct type instead
-}
-
-func (ck *connectMethod) String() string {
+func (cm *connectMethod) key() connectMethodKey {
        proxyStr := ""
-       targetAddr := ck.targetAddr
-       if ck.proxyURL != nil {
-               proxyStr = ck.proxyURL.String()
-               if ck.targetScheme == "http" {
+       targetAddr := cm.targetAddr
+       if cm.proxyURL != nil {
+               proxyStr = cm.proxyURL.String()
+               if cm.targetScheme == "http" {
                        targetAddr = ""
                }
        }
-       return strings.Join([]string{proxyStr, ck.targetScheme, targetAddr}, "|")
+       return connectMethodKey{
+               proxy:  proxyStr,
+               scheme: cm.targetScheme,
+               addr:   targetAddr,
+       }
 }
 
 // addr returns the first hop "host:port" to which we need to TCP connect.
        return h
 }
 
+// connectMethodKey is the map key version of connectMethod, with a
+// stringified proxy URL (or the empty string) instead of a pointer to
+// a URL.
+type connectMethodKey struct {
+       proxy, scheme, addr string
+}
+
+func (k connectMethodKey) String() string {
+       // Only used by tests.
+       return fmt.Sprintf("%s|%s|%s", k.proxy, k.scheme, k.addr)
+}
+
 // persistConn wraps a connection, usually a persistent one
 // (but may be used for non-keep-alive requests as well)
 type persistConn struct {
        t        *Transport
-       cacheKey string // its connectMethod.String()
+       cacheKey connectMethodKey
        conn     net.Conn
        closed   bool                // whether conn has been closed
        br       *bufio.Reader       // from conn