这篇博客侧重于了解OkHttp的网络部分,包括Socket的创建、连接,连接池等要点。OkHttp对Socket的流操作使用了Okio进行了封装,本篇博客不做介绍,想了解的朋友可以参考拆轮子系列:拆Okio

OkHttp中关于网络的几个概念

下面的主要翻译自OkHttp的官方文档,查看原文.

URL

URLs(比如https://github.com/square/okhttp)是HTTP和网络的基础,不止指定了Web上的资源,还指定了如何获取该资源。

Address

Address(比如github.com)指定了一个webserver和所有连接到该服务器的必需的静态配置:端口、HTTPS设置和首选网络协议(HTTP/2或SPDY)。
URLs属于同一个address的可以共享同一个底层的Socket连接。共享一个连接具有显著的性能优势:低延迟、高吞吐量(由于TCP慢启动)和省电。OkHttp使用连接池自动再利用HTTP/1.x的连接,复用HTTP/2和SPDY的连接。
在OkHttp中,address的一些字段来自URL(模式、主机名、端口),剩下的部分来自OkHttpClient。

Routes

Routes提供真正连接到一个网络服务器所需的动态信息。这指定了尝试的IP地址(或者进过DNS查询得到的地址)、使用的代理服务器(如果使用了ProxySelector)和使用哪个版本的TLS进行谈判。(对于HTTPS连接)
对于一个地址,可能有多个路由。举个例子,一个网路服务器托管在多个数据中心,那么在DNS中可能会产生多个IP地址。

Connections

当请求一个URL时,OkHttp会做以下几件事情:
1. 使用URL和配置好的OkHttpClient创建一个address。这个地址指明了我们将如何连接网络服务器。
2. 尝试从连接池中得到该地址的一条连接
3. 如果在连接池中没有找到一条连接,那么选择一个route进行尝试。通常这意味着做一个DNS请求得到服务器IP的地址,必要时会选择一个TLS版本和一个代理服务器。
4. 如果是一条新的路由,那么建立一条直接的socket连接或TLS通道(HTTPS使用HTTP代理)或一个直接的TLS连接。
5. 发送HTTP请求,读取响应。
如果连接出现了问题,OkHttp会选择另外一条路由进行再次尝试。这使得OkHttp在一个服务器的一些地址不可到达时仍然可用。
一旦读取到响应后,连接将会退还到连接池中以便可以复用。连接在池中闲置一段时间后将会被释放。

结合源码进行分析

Address的创建

Address的创建在RetryAndFollowupInterceptor中的createAddress方法中,代码如下:

  1. private Address createAddress(HttpUrl url) {
  2. SSLSocketFactory sslSocketFactory = null;
  3. HostnameVerifier hostnameVerifier = null;
  4. CertificatePinner certificatePinner = null;
  5. //如果是HTTPS协议
  6. if (url.isHttps()) {
  7. sslSocketFactory = client.sslSocketFactory();
  8. hostnameVerifier = client.hostnameVerifier();
  9. certificatePinner = client.certificatePinner();
  10. }
  11. //可以看到Address的构造方法中的一部分参数由URL提供,一部分由OkHttpClient提供
  12. return new Address(url.host(), url.port(), client.dns(), client.socketFactory(),
  13. sslSocketFactory, hostnameVerifier, certificatePinner, client.proxyAuthenticator(),
  14. client.proxy(), client.protocols(), client.connectionSpecs(), client.proxySelector());
  15. }

从代码中可以看出,Address的信息一部分由URL提供,主要包括主机名和端口;另一部分由OkHttpClient提供,如dns、socketFactory等等。
根据HttpUrl是否是HTTPS,创建sslSocketFactory等字段,而在Address的构造方法中,则根据sslSocketFactory是否为null判断是HTTP模式还是HTTPS模式。

StreamAllocation的创建

StreamAllocation类负责管理连接、流和请求三者之间的关系。其创建在RetryAndFollowupInterceptor的intercept方法中,使用OkHttpClient的连接池以及上面创建的Address进行初始化,代码如下:

  1. streamAllocation = new StreamAllocation(
  2. client.connectionPool(), createAddress(request.url()))

其中client的连接池是在OkHttpClient.Builder中设置的,而其设置在Builder的构造方法中,调用的是ConnectionPool的默认构造方法,代码如下:

  1. public Builder() {
  2. ...
  3. //默认连接池
  4. connectionPool = new ConnectionPool();
  5. dns = Dns.SYSTEM;
  6. followSslRedirects = true;
  7. followRedirects = true;
  8. retryOnConnectionFailure = true;
  9. connectTimeout = 10_000;
  10. readTimeout = 10_000;
  11. writeTimeout = 10_000;
  12. }
  • 1
  • 2
  • 3

下面是ConnectionPool的构造方法:

  1. /**
  2. * Create a new connection pool with tuning parameters appropriate for a single-user application.
  3. * The tuning parameters in this pool are subject to change in future OkHttp releases. Currently
  4. * this pool holds up to 5 idle connections which will be evicted after 5 minutes of inactivity.
  5. */
  6. public ConnectionPool() {
  7. this(5, 5, TimeUnit.MINUTES);
  8. }
  9. public ConnectionPool(int maxIdleConnections, long keepAliveDuration, TimeUnit timeUnit) {
  10. this.maxIdleConnections = maxIdleConnections;
  11. this.keepAliveDurationNs = timeUnit.toNanos(keepAliveDuration);
  12. // Put a floor on the keep alive duration, otherwise cleanup will spin loop.
  13. if (keepAliveDuration <= 0) {
  14. throw new IllegalArgumentException("keepAliveDuration <= 0: " + keepAliveDuration);
  15. }
  16. }

从上面可以看到,默认的连接池的最大空闲连接数为5,最长存活时间为5min。

HttpStream和Connection的创建

深入理解OkHttp源码(二)——获取响应中,我们知道了HttpStream以及Connection的创建都是在ConnectInterceptor拦截器中,代码如下:

  1. @Override public Response intercept(Chain chain) throws IOException {
  2. RealInterceptorChain realChain = (RealInterceptorChain) chain;
  3. Request request = realChain.request();
  4. StreamAllocation streamAllocation = realChain.streamAllocation();
  5. // We need the network to satisfy this request. Possibly for validating a conditional GET.
  6. boolean doExtensiveHealthChecks = !request.method().equals("GET");
  7. HttpStream httpStream = streamAllocation.newStream(client, doExtensiveHealthChecks);
  8. RealConnection connection = streamAllocation.connection();
  9. return realChain.proceed(request, streamAllocation, httpStream, connection);
  10. }

从上面的代码可以看到,首先调用StreamAllocation的newStream方法就可以得到HttpStream对象,同时也就得到了Connection对象。下面首选从StreamAllocation的newStream()方法看起:

  1. public HttpStream newStream(OkHttpClient client, boolean doExtensiveHealthChecks) {
  2. //得到连接时长、读超时以及写超时参数
  3. int connectTimeout = client.connectTimeoutMillis();
  4. int readTimeout = client.readTimeoutMillis();
  5. int writeTimeout = client.writeTimeoutMillis();
  6. boolean connectionRetryEnabled = client.retryOnConnectionFailure();
  7. try {
  8. //得到一个健康的连接
  9. RealConnection resultConnection = findHealthyConnection(connectTimeout, readTimeout,
  10. writeTimeout, connectionRetryEnabled, doExtensiveHealthChecks);
  11. HttpStream resultStream;
  12. //如果协议是HTTP 2.x协议
  13. if (resultConnection.framedConnection != null) {
  14. resultStream = new Http2xStream(client, this, resultConnection.framedConnection);
  15. }
  16. //协议是HTTP 1.x,设置连接底层的Socket属性
  17. else {
  18. resultConnection.socket().setSoTimeout(readTimeout);
  19. resultConnection.source.timeout().timeout(readTimeout, MILLISECONDS);
  20. resultConnection.sink.timeout().timeout(writeTimeout, MILLISECONDS);
  21. resultStream = new Http1xStream(
  22. client, this, resultConnection.source, resultConnection.sink);
  23. }
  24. synchronized (connectionPool) {
  25. stream = resultStream;
  26. return resultStream;
  27. }
  28. } catch (IOException e) {
  29. throw new RouteException(e);
  30. }
  31. }

从上面的代码可以看出,首先从OkHttpClient中获取连接超时、读取超时、写超时和是否连接失败重试参数,然后试图找到一条健康的连接,接下来是根据连接的framedConnection字段是否为null,得到Http2xStream或Http1xStram,前者是HTTP/2的实现,后者是HTTP/1.x的实现。
可以看到主要的逻辑肯定都在findHealthyConnection方法中,下面是findHeadlthyConnection方法的实现:

  1. /**
  2. * Finds a connection and returns it if it is healthy. If it is unhealthy the process is repeated
  3. * until a healthy connection is found.
  4. */
  5. private RealConnection findHealthyConnection(int connectTimeout, int readTimeout,
  6. int writeTimeout, boolean connectionRetryEnabled, boolean doExtensiveHealthChecks)
  7. throws IOException {
  8. //死循环
  9. while (true) {
  10. //得到一个候选的连接
  11. RealConnection candidate = findConnection(connectTimeout, readTimeout, writeTimeout,
  12. connectionRetryEnabled);
  13. // 如果是一个全新的连接,跳过额外的健康检查
  14. synchronized (connectionPool) {
  15. if (candidate.successCount == 0) {
  16. return candidate;
  17. }
  18. }
  19. //如果候选连接通不过额外的健康检查,那么继续寻找一个新的候选连接
  20. if (!candidate.isHealthy(doExtensiveHealthChecks)) {
  21. noNewStreams();
  22. continue;
  23. }
  24. return candidate;
  25. }
  26. }

从注释中可以看到,该方法用于查找一条健康的连接并返回,如果连接不健康,那么会重复查找,直到查找到健康的连接。可以看到方法内是一个死循环,首先调用findConnection方法得到候选的连接,如果该连接是一个全新的连接,那么就直接返回不需要验证是否健康,如果不是则需要验证是否健康,如果不健康调用noNewStreams()方法后继续下一次循环,否则返回。对于候选连接,总结一下就是下面几种情况:
1. 候选连接是一个全新的连接,那么直接返回;
2. 候选连接不是一个全新的连接,但是是健康的,那么直接返回;
3. 候选连接不是一个全新的连接,并且不健康,那么继续下一轮循环
经过上面的分析,我们查看findConnection()方法:

  1. /**
  2. * Returns a connection to host a new stream. This prefers the existing connection if it exists,
  3. * then the pool, finally building a new connection.
  4. */
  5. private RealConnection findConnection(int connectTimeout, int readTimeout, int writeTimeout,
  6. boolean connectionRetryEnabled) throws IOException {
  7. Route selectedRoute;
  8. //对连接池加锁,因为可能会有别的线程加入连接或移除连接
  9. synchronized (connectionPool) {
  10. if (released) throw new IllegalStateException("released");
  11. if (stream != null) throw new IllegalStateException("stream != null");
  12. if (canceled) throw new IOException("Canceled");
  13. //首先尝试使用本实例的连接
  14. RealConnection allocatedConnection = this.connection;
  15. if (allocatedConnection != null && !allocatedConnection.noNewStreams) {
  16. return allocatedConnection;
  17. }
  18. //其次,尝试从连接池中得到连接
  19. RealConnection pooledConnection = Internal.instance.get(connectionPool, address, this);
  20. if (pooledConnection != null) {
  21. this.connection = pooledConnection;
  22. return pooledConnection;
  23. }
  24. selectedRoute = route;
  25. }
  26. if (selectedRoute == null) {
  27. selectedRoute = routeSelector.next();
  28. synchronized (connectionPool) {
  29. route = selectedRoute;
  30. refusedStreamCount = 0;
  31. }
  32. }
  33. //根据路由创建新的连接
  34. RealConnection newConnection = new RealConnection(selectedRoute);
  35. acquire(newConnection);
  36. //将得到的新连接加入连接池中并设置本实例的连接
  37. synchronized (connectionPool) {
  38. Internal.instance.put(connectionPool, newConnection);
  39. this.connection = newConnection;
  40. if (canceled) throw new IOException("Canceled");
  41. }
  42. //底层Socket连接
  43. newConnection.connect(connectTimeout, readTimeout, writeTimeout, address.connectionSpecs(),
  44. connectionRetryEnabled);
  45. routeDatabase().connected(newConnection.route());
  46. return newConnection;
  47. }

从注释中可以看出,该方法返回一个拥有新流的连接。首先检查已存在的连接,其次连接池,最后建立一个新的连接。
从代码中可以看出,首先对连接池加锁,这儿的连接池是在创建StreamAllocation中传入的,而那个参数是在创建OkHttpClient时就创建的,我们一般使用OkHttpClient时,都会将其做成单例,那么连接池就是唯一的,由于可能存在别的线程从连接池中执行插入以及连接池自身连接的清除工作,所以需要对其进行加锁。首先获取本对象的connection,如果不为null并且noNewStreams为false,那么直接使用本连接;如果不能使用本连接,那么尝试从连接池中获取连接,如果可以得到,那么直接返回,否则将进行下一步创建新连接;首先根据路由创建一个新的连接,然后调用acquire方法使连接持有该StreamAllocation对象,接下来将新的连接添加就连接池,最后调用connect方法进行连接。

这里面有一个Internal.instance的实例,Internal是一个抽象类,其具体实现instance初始化是在OkHttpClient的静态初始化块中,如下:

  1. static {
  2. Internal.instance = new Internal() {
  3. @Override public void addLenient(Headers.Builder builder, String line) {
  4. builder.addLenient(line);
  5. }
  6. @Override public void addLenient(Headers.Builder builder, String name, String value) {
  7. builder.addLenient(name, value);
  8. }
  9. @Override public void setCache(OkHttpClient.Builder builder, InternalCache internalCache) {
  10. builder.setInternalCache(internalCache);
  11. }
  12. @Override public boolean connectionBecameIdle(
  13. ConnectionPool pool, RealConnection connection) {
  14. return pool.connectionBecameIdle(connection);
  15. }
  16. @Override public RealConnection get(
  17. ConnectionPool pool, Address address, StreamAllocation streamAllocation) {
  18. return pool.get(address, streamAllocation);
  19. }
  20. @Override public void put(ConnectionPool pool, RealConnection connection) {
  21. pool.put(connection);
  22. }
  23. @Override public RouteDatabase routeDatabase(ConnectionPool connectionPool) {
  24. return connectionPool.routeDatabase;
  25. }
  26. @Override public StreamAllocation callEngineGetStreamAllocation(Call call) {
  27. return ((RealCall) call).streamAllocation();
  28. }
  29. @Override
  30. public void apply(ConnectionSpec tlsConfiguration, SSLSocket sslSocket, boolean isFallback) {
  31. tlsConfiguration.apply(sslSocket, isFallback);
  32. }
  33. @Override public HttpUrl getHttpUrlChecked(String url)
  34. throws MalformedURLException, UnknownHostException {
  35. return HttpUrl.getChecked(url);
  36. }
  37. @Override public void setCallWebSocket(Call call) {
  38. ((RealCall) call).setForWebSocket();
  39. }
  40. };

首先看put方法,因为一开始时连接池中肯定是没有连接的,Internal.instance的put方法调用了连接池的put方法,下面是ConnectionPool的put方法:

  1. void put(RealConnection connection) {
  2. assert (Thread.holdsLock(this));
  3. //如果清理线程没有开启,则开启
  4. if (!cleanupRunning) {
  5. cleanupRunning = true;
  6. executor.execute(cleanupRunnable);
  7. }
  8. connections.add(connection);
  9. }

从代码中可以看出,当第一个连接被添加就线程池时,开启清除线程,主要清除那些连接池中过期的连接,然后将连接添加就connections对象中。下面看一下cleanupRunnable和connections的定义,其中connections是一个阻塞队列。

  1. private final Runnable cleanupRunnable = new Runnable() {
  2. @Override public void run() {
  3. while (true) {
  4. //得到下一次清除的等待时长
  5. long waitNanos = cleanup(System.nanoTime());
  6. //没有连接了,清除任务终结
  7. if (waitNanos == -1) return;
  8. //需要等待一定时长
  9. if (waitNanos > 0) {
  10. long waitMillis = waitNanos / 1000000L;
  11. waitNanos -= (waitMillis * 1000000L);
  12. synchronized (ConnectionPool.this) {
  13. try {
  14. ConnectionPool.this.wait(waitMillis, (int) waitNanos);
  15. } catch (InterruptedException ignored) {
  16. }
  17. }
  18. }
  19. }
  20. }
  21. };
  22. private final Deque<RealConnection> connections = new ArrayDeque<>();

可以看到cleadupRunnbale是一个死循环,调用cleanup方法进行清理工作并返回一个等待时长,如果有等待时长,那么让连接池进行休眠。其中清理工作在cleanup方法中,代码如下:

  1. /**
  2. * Performs maintenance on this pool, evicting the connection that has been idle the longest if
  3. * either it has exceeded the keep alive limit or the idle connections limit.
  4. *
  5. * <p>Returns the duration in nanos to sleep until the next scheduled call to this method. Returns
  6. * -1 if no further cleanups are required.
  7. */
  8. long cleanup(long now) {
  9. int inUseConnectionCount = 0;
  10. int idleConnectionCount = 0;
  11. RealConnection longestIdleConnection = null;
  12. long longestIdleDurationNs = Long.MIN_VALUE;
  13. // Find either a connection to evict, or the time that the next eviction is due.
  14. synchronized (this) {
  15. //检查每个连接
  16. for (Iterator<RealConnection> i = connections.iterator(); i.hasNext(); ) {
  17. RealConnection connection = i.next();
  18. //如果该连接正在运行,则跳过
  19. if (pruneAndGetAllocationCount(connection, now) > 0) {
  20. inUseConnectionCount++;
  21. continue;
  22. }
  23. idleConnectionCount++;
  24. //查找出空闲时间最长的连接
  25. long idleDurationNs = now - connection.idleAtNanos;
  26. if (idleDurationNs > longestIdleDurationNs) {
  27. longestIdleDurationNs = idleDurationNs;
  28. longestIdleConnection = connection;
  29. }
  30. }
  31. //如果时间超出规定的空闲时间或者数量达到最大空闲树,那么移除。关闭操作在后面
  32. if (longestIdleDurationNs >= this.keepAliveDurationNs
  33. || idleConnectionCount > this.maxIdleConnections) {
  34. connections.remove(longestIdleConnection);
  35. }
  36. //如果时间和数量都没到达上限,那么得到存活时间
  37. else if (idleConnectionCount > 0) {
  38. return keepAliveDurationNs - longestIdleDurationNs;
  39. }
  40. //如果所有连接都在使用中,返回最大存活时间
  41. else if (inUseConnectionCount > 0) {
  42. return keepAliveDurationNs;
  43. }
  44. //没有连接,关闭清除线程
  45. else {
  46. cleanupRunning = false;
  47. return -1;
  48. }
  49. }
  50. //关闭连接底层的Socket
  51. closeQuietly(longestIdleConnection.socket());
  52. // 再次执行清除
  53. return 0;
  54. }

从代码中可以看出,对当前连接池中保存的所有连接进行遍历,然后调用pruneAndGetAllocationCount()方法获取连接上可用的StreamAllocation的数量以及删除不可用的StreamAllocation,如果数量大于0,则表示该连接还在使用,那么继续下一次遍历;否则空闲连接数+1,需要查找出所有不可用的连接中最大的空闲时间。遍历做完后,根据不同情况不同的值返回不同的结果,一旦找到了最大的空闲连接,那么在同步块外部调用closeQuietly关闭连接。
pruneAndGetAllocationCount()方法用于删除连接上不可用的StreamAllocation以及可用的StreamAllocation的数量,下面是其具体实现:

  1. /**
  2. * Prunes any leaked allocations and then returns the number of remaining live allocations on
  3. * {@code connection}. Allocations are leaked if the connection is tracking them but the
  4. * application code has abandoned them. Leak detection is imprecise and relies on garbage
  5. * collection.
  6. */
  7. private int pruneAndGetAllocationCount(RealConnection connection, long now) {
  8. //得到关联在连接上StramAllocation对象列表
  9. List<Reference<StreamAllocation>> references = connection.allocations;
  10. for (int i = 0; i < references.size(); ) {
  11. Reference<StreamAllocation> reference = references.get(i);
  12. //可用
  13. if (reference.get() != null) {
  14. i++;
  15. continue;
  16. }
  17. // We've discovered a leaked allocation. This is an application bug.
  18. Platform.get().log(WARN, "A connection to " + connection.route().address().url()
  19. + " was leaked. Did you forget to close a response body?", null);
  20. references.remove(i);
  21. connection.noNewStreams = true;
  22. // If this was the last allocation, the connection is eligible for immediate eviction.
  23. if (references.isEmpty()) {
  24. connection.idleAtNanos = now - keepAliveDurationNs;
  25. return 0;
  26. }
  27. }
  28. return references.size();
  29. }

需要注意的是for循环,i的控制在循环内部,如果StreamAllocation为null,那么直接删除,如果连接没有一个可用的StreamAllocation,那么设置连接的idleAtNanos为now-keepAliveDurationNs,即5分钟之前。
至此,我们分析完了当创建了一个新连接,是如何被添加到线程池中的以及线程池的自动清除线程是如何工作的。下面看连接是如何建立连接的,在findConnection方法中,当创建了一个新的Connection后,调用了其connect方法,connect负责将客户端Socket连接到服务端Socket,代码如下:

  1. public void connect(int connectTimeout, int readTimeout, int writeTimeout,
  2. List<ConnectionSpec> connectionSpecs, boolean connectionRetryEnabled) {
  3. if (protocol != null) throw new IllegalStateException("already connected");
  4. RouteException routeException = null;
  5. ConnectionSpecSelector connectionSpecSelector = new ConnectionSpecSelector(connectionSpecs);
  6. //不是HTTPS协议
  7. if (route.address().sslSocketFactory() == null) {
  8. if (!connectionSpecs.contains(ConnectionSpec.CLEARTEXT)) {
  9. throw new RouteException(new UnknownServiceException(
  10. "CLEARTEXT communication not enabled for client"));
  11. }
  12. String host = route.address().url().host();
  13. if (!Platform.get().isCleartextTrafficPermitted(host)) {
  14. throw new RouteException(new UnknownServiceException(
  15. "CLEARTEXT communication to " + host + " not permitted by network security policy"));
  16. }
  17. }
  18. while (protocol == null) {
  19. try {
  20. if (route.requiresTunnel()) {
  21. buildTunneledConnection(connectTimeout, readTimeout, writeTimeout,
  22. connectionSpecSelector);
  23. } else {
  24. buildConnection(connectTimeout, readTimeout, writeTimeout, connectionSpecSelector);
  25. }
  26. } catch (IOException e) {
  27. closeQuietly(socket);
  28. closeQuietly(rawSocket);
  29. socket = null;
  30. rawSocket = null;
  31. source = null;
  32. sink = null;
  33. handshake = null;
  34. protocol = null;
  35. if (routeException == null) {
  36. routeException = new RouteException(e);
  37. } else {
  38. routeException.addConnectException(e);
  39. }
  40. if (!connectionRetryEnabled || !connectionSpecSelector.connectionFailed(e)) {
  41. throw routeException;
  42. }
  43. }
  44. }
  45. }

主要看while循环处,如果HTTPS通道使用HTTP代理,那么调用buildTunneledConnection方法,否则调用buildConnection方法,如果出现异常,那么就在catch中做了一些清理工作,然后会继续进入循环,因为将protocol置为了null。一般的请求都是直接调用buildConnection方法的,下面我们看buildConnection方法:

  1. /** Does all the work necessary to build a full HTTP or HTTPS connection on a raw socket. */
  2. private void buildConnection(int connectTimeout, int readTimeout, int writeTimeout,
  3. ConnectionSpecSelector connectionSpecSelector) throws IOException {
  4. connectSocket(connectTimeout, readTimeout);
  5. establishProtocol(readTimeout, writeTimeout, connectionSpecSelector);
  6. }

该方法做在raw socket上连接HTTP或HTTPS连接的准备工作,方法内部又是调用了另外两个方法,下面分别介绍。
connectSocket为创建Socket以及连接Socket,代码如下:

  1. private void connectSocket(int connectTimeout, int readTimeout) throws IOException {
  2. Proxy proxy = route.proxy();
  3. Address address = route.address();
  4. //创建Socket
  5. rawSocket = proxy.type() == Proxy.Type.DIRECT || proxy.type() == Proxy.Type.HTTP
  6. ? address.socketFactory().createSocket()
  7. : new Socket(proxy);
  8. rawSocket.setSoTimeout(readTimeout);
  9. try {
  10. //连接Socket
  11. Platform.get().connectSocket(rawSocket, route.socketAddress(), connectTimeout);
  12. } catch (ConnectException e) {
  13. throw new ConnectException("Failed to connect to " + route.socketAddress());
  14. }
  15. //使用Okio封装Socket的输入输出流
  16. source = Okio.buffer(Okio.source(rawSocket));
  17. sink = Okio.buffer(Okio.sink(rawSocket));
  18. }

从代码可以看出,首先获取代理和地址,然后根据代理的类型是使用SocketFactory工厂创建无参的rawSocket还是使用带代理参数的Socket构造方法,得到了rawSocket对象后,设置读超时,然后调用connectSocket进行Socket的连接,服务器的信息在route的socketAddress中,最后,得到rawSocket的输入流和输出流,这里使用了Okio进行了封装,就不做过多设计了。
其中Plateform.get()方法返回不同平台的信息,因为OkHttp是可以用于AndroidJava平台的,而Java又有多个版本,所以进行了平台判断。get()是一个单例,其初始化在findPlatform方法中,如下:

  1. private static Platform findPlatform() {
  2. Platform android = AndroidPlatform.buildIfSupported();
  3. if (android != null) {
  4. return android;
  5. }
  6. Platform jdk9 = Jdk9Platform.buildIfSupported();
  7. if (jdk9 != null) {
  8. return jdk9;
  9. }
  10. Platform jdkWithJettyBoot = JdkWithJettyBootPlatform.buildIfSupported();
  11. if (jdkWithJettyBoot != null) {
  12. return jdkWithJettyBoot;
  13. }
  14. // Probably an Oracle JDK like OpenJDK.
  15. return new Platform();
  16. }

可以看到findPlatform分为了android平台、jdk9、有JettyBoot的jdk还有默认的平台几类。这边看默认的Platform就可以了。下面看其socket方法:

  1. public void connectSocket(Socket socket, InetSocketAddress address,
  2. int connectTimeout) throws IOException {
  3. socket.connect(address, connectTimeout);
  4. }

可以看到就是调用socket的connect方法,至此,本地Socket与后台Socket建立了连接,并得到了输入输出流。
buildConnection方法中还有一个establishProtocol方法,该方法用于建立协议,设置protocol的值,这样上面的循环就可以跳出了。代码如下:

  1. private void establishProtocol(int readTimeout, int writeTimeout,
  2. ConnectionSpecSelector connectionSpecSelector) throws IOException {
  3. //如果是HTTPS协议
  4. if (route.address().sslSocketFactory() != null) {
  5. connectTls(readTimeout, writeTimeout, connectionSpecSelector);
  6. }
  7. //默认HTTP 1.1协议
  8. else {
  9. protocol = Protocol.HTTP_1_1;
  10. socket = rawSocket;
  11. }
  12. if (protocol == Protocol.SPDY_3 || protocol == Protocol.HTTP_2) {
  13. socket.setSoTimeout(0); // Framed connection timeouts are set per-stream.
  14. FramedConnection framedConnection = new FramedConnection.Builder(true)
  15. .socket(socket, route.address().url().host(), source, sink)
  16. .protocol(protocol)
  17. .listener(this)
  18. .build();
  19. framedConnection.start();
  20. // Only assign the framed connection once the preface has been sent successfully.
  21. this.allocationLimit = framedConnection.maxConcurrentStreams();
  22. this.framedConnection = framedConnection;
  23. } else {
  24. this.allocationLimit = 1;
  25. }
  26. }

可以看到该方法主要就是给protocol赋值,另外对于SPDY或HTTP/2协议有别的处理,这儿就不多介绍了。(==因为我自己目前也不懂,不过分析到这儿就已经足够了==)。
至此,我们分析完了是如何新建一个连接,然后将其放入连接池以及真正地与后台建立连接的,这一切都是发生在ConnectInterceptor中,所以也就可以理解为什么这个拦截器要命名为连接拦截器了。
上面的代码主要分析了新建连接,从上面的分析我们知道,还可以直接使用StreamAllocation的连接或从连接池中获取连接。我们知道当提交请求后,每个请求被封装成RealCall对象,而每个RealCall对象都只能被执行一次,RealCall对象持有RetryAndFollowupInterceptor,Connection又是RetryAndFollowupInterceptor持有的,那么如果发生重定向时,但是主机名相同,只是路径不同时,那么将会是重用之前创建的Connection;而如果是两个相同主机的不同请求,那么在第一个连接被创建放进线程池后,第二个请求的连接就可以从连接池中得到了。

findConnection方法中通过调用Internal.instance的get方法从连接池中获取连接,而其get方法又是通过调用连接池的get方法,具体代码如下:

  1. /** Returns a recycled connection to {@code address}, or null if no such connection exists. */
  2. RealConnection get(Address address, StreamAllocation streamAllocation) {
  3. assert (Thread.holdsLock(this));
  4. for (RealConnection connection : connections) {
  5. if (connection.allocations.size() < connection.allocationLimit
  6. && address.equals(connection.route().address)
  7. && !connection.noNewStreams) {
  8. streamAllocation.acquire(connection);
  9. return connection;
  10. }
  11. }
  12. return null;
  13. }

从上面代码中可以看出,get方法对连接池队列遍历,如果连接的StreamAllocation小于allocationLimit参数并且地址相等且连接的noNewStreams为false,那么将streamAllocation赋给连接。其中allocationLimit在协议为HTTP/1.x时为1,这也就意味着同一个Connection只能与一个StreamAllocation绑定,这就解释了为什么官方文档文档说连接池重用HTTP/1.x连接,复用HTTP/2或SPDY连接。

发送请求和获取响应

经过ConnectInterceptor后,为请求创建了Connection对象以及HttpStream对象,下面进入到CallServerInterceptor中发送请求和获取响应,首先看CallServerInterceptor的intercept方法:

  1. @Override public Response intercept(Chain chain) throws IOException {
  2. HttpStream httpStream = ((RealInterceptorChain) chain).httpStream();
  3. StreamAllocation streamAllocation = ((RealInterceptorChain) chain).streamAllocation();
  4. Request request = chain.request();
  5. long sentRequestMillis = System.currentTimeMillis();
  6. //发送HTTP首部信息
  7. httpStream.writeRequestHeaders(request);
  8. //如果HTTP方法允许有请求主体并且请求不为null,发送HTTP请求主体信息
  9. if (HttpMethod.permitsRequestBody(request.method()) && request.body() != null) {
  10. //Okio进行封装发送数据
  11. Sink requestBodyOut = httpStream.createRequestBody(request, request.body().contentLength());
  12. BufferedSink bufferedRequestBody = Okio.buffer(requestBodyOut);
  13. request.body().writeTo(bufferedRequestBody);
  14. bufferedRequestBody.close();
  15. }
  16. httpStream.finishRequest();
  17. //读响应首部构建Response对象
  18. Response response = httpStream.readResponseHeaders()
  19. .request(request)
  20. .handshake(streamAllocation.connection().handshake())
  21. .sentRequestAtMillis(sentRequestMillis)
  22. .receivedResponseAtMillis(System.currentTimeMillis())
  23. .build();
  24. if (!forWebSocket || response.code() != 101) {
  25. response = response.newBuilder()
  26. .body(httpStream.openResponseBody(response))
  27. .build();
  28. }
  29. //服务端不支持HTTP持久连接,那么需要关闭该连接
  30. if ("close".equalsIgnoreCase(response.request().header("Connection"))
  31. || "close".equalsIgnoreCase(response.header("Connection"))) {
  32. streamAllocation.noNewStreams();
  33. }
  34. int code = response.code();
  35. if ((code == 204 || code == 205) && response.body().contentLength() > 0) {
  36. throw new ProtocolException(
  37. "HTTP " + code + " had non-zero Content-Length: " + response.body().contentLength());
  38. }
  39. return response;
  40. }

可以看到写请求和读响应都是通过HttpStream对象,在前面的分析中知道了HttpStream的具体实现是Http1xStream或Http2xStream。我们主要看Http1xStream的各个实现,首先看写头部信息的writeRequestHeaders方法,下面是Http1xStream的具体实现:

  1. @Override public void writeRequestHeaders(Request request) throws IOException {
  2. //得到请求行
  3. String requestLine = RequestLine.get(
  4. request, streamAllocation.connection().route().proxy().type());
  5. writeRequest(request.headers(), requestLine);
  6. }

该方法用户将头信息发送给服务端,首先获取HTTP请求行(类似于“GET / HTTP/1.1”),然后调用writeRequest方法进行具体的写操作,下面是writeRequest的实现:

  1. public void writeRequest(Headers headers, String requestLine) throws IOException {
  2. if (state != STATE_IDLE) throw new IllegalStateException("state: " + state);
  3. sink.writeUtf8(requestLine).writeUtf8("\r\n");
  4. for (int i = 0, size = headers.size(); i < size; i++) {
  5. sink.writeUtf8(headers.name(i))
  6. .writeUtf8(": ")
  7. .writeUtf8(headers.value(i))
  8. .writeUtf8("\r\n");
  9. }
  10. sink.writeUtf8("\r\n");
  11. state = STATE_OPEN_REQUEST_BODY;
  12. }

从代码中可以看出,首先判断状态,状态初始值为STATE_IDLE,表明如果在写头部信息之前做了别的操作,那么将会报错,也就意味着必须首先进行写头部信息的操作;然后写入请求行以及换行符,接下来就是对头部信息做遍历,逐个写入,最后将状态置为STATE_OPEN_REQUEST_BODY。
在写完头部信息之后,如果需要写请求的主体部分,还会进行写主体部分操作,当请求发送完成后,调用finishRequest方法就行刷新输出流。

  1. @Override public void finishRequest() throws IOException {
  2. sink.flush();
  3. }

发送完请求之后,首先调用readResponseHeaders()获取响应的头部信息,然后构造Response对象,readResponseHeaders代码如下:

  1. @Override public Response.Builder readResponseHeaders() throws IOException {
  2. return readResponse();
  3. }
  4. ** Parses bytes of a response header from an HTTP transport. */
  5. public Response.Builder readResponse() throws IOException {
  6. if (state != STATE_OPEN_REQUEST_BODY && state != STATE_READ_RESPONSE_HEADERS) {
  7. throw new IllegalStateException("state: " + state);
  8. }
  9. try {
  10. while (true) {
  11. StatusLine statusLine = StatusLine.parse(source.readUtf8LineStrict());
  12. Response.Builder responseBuilder = new Response.Builder()
  13. .protocol(statusLine.protocol)
  14. .code(statusLine.code)
  15. .message(statusLine.message)
  16. .headers(readHeaders());
  17. if (statusLine.code != HTTP_CONTINUE) {
  18. state = STATE_OPEN_RESPONSE_BODY;
  19. return responseBuilder;
  20. }
  21. }
  22. } catch (EOFException e) {
  23. // Provide more context if the server ends the stream before sending a response.
  24. IOException exception = new IOException("unexpected end of stream on " + streamAllocation);
  25. exception.initCause(e);
  26. throw exception;
  27. }
  28. }

可以看到readResponseHeaders方法又调用了readResponse方法,而readResponse方法中首先对状态进行判断,然后进入一个死循环。首先获取响应的状态行(比如“H T T P / 1 . 1 2 0 0 T e m p o r a r y R e d i r e c t”)得到协议类型、状态码和消息,然后再调用readHeaders()方法读取头部信息,最后比较状态码不是100,那么说明请求发送完整了,那么将状态置为STATE_OPEN_RESPONSE_BODY,然后返回响应,这时的响应中只有协议类型、状态码、消息和头部信息。下面看一下readHeaders()方法是如何获取头部信息的:

  1. /** Reads headers or trailers. */
  2. public Headers readHeaders() throws IOException {
  3. Headers.Builder headers = new Headers.Builder();
  4. // parse the result headers until the first blank line
  5. for (String line; (line = source.readUtf8LineStrict()).length() != 0; ) {
  6. Internal.instance.addLenient(headers, line);
  7. }
  8. return headers.build();
  9. }
  • 1

可以看到每行遍历直到第一个空行,然后调用Internal.instance的addLenient方法将这一行的信息解析并添加到头部中,下面是addLenient方法的实现:

  1. @Override public void addLenient(Headers.Builder builder, String line) {
  2. builder.addLenient(line);
  3. }

可以看到只是简单的调用Builder的addLenient方法,那么继续看Builder的addLenient方法:

  1. Builder addLenient(String line) {
  2. int index = line.indexOf(":", 1);
  3. if (index != -1) {
  4. return addLenient(line.substring(0, index), line.substring(index + 1));
  5. } else if (line.startsWith(":")) {
  6. // Work around empty header names and header names that start with a
  7. // colon (created by old broken SPDY versions of the response cache).
  8. return addLenient("", line.substring(1)); // Empty header name.
  9. } else {
  10. return addLenient("", line); // No header name.
  11. }
  12. }
  • 1

从上面的代码可以看到,首先获取“:”的位置,如果存在“:”,那么调用addLenient将名和值添加进列表中,如果以”:”开宇,则头信息的名称为空,有值;如果都没有,那么没有头部信息名。三种情况都是调用addLenient方法,如下:

  1. /**
  2. * Add a field with the specified value without any validation. Only appropriate for headers
  3. * from the remote peer or cache.
  4. */
  5. Builder addLenient(String name, String value) {
  6. namesAndValues.add(name);
  7. namesAndValues.add(value.trim());
  8. return this;
  9. }

其中,nameAndValues是一个字符串的列表。
到上面为此,读取响应的头部信息已经完成,接下来在CallServerInterceptor中做的是调用openResponseBody方法读取响应的主体部分,方法如下:

  1. @Override public ResponseBody openResponseBody(Response response) throws IOException {
  2. Source source = getTransferStream(response);
  3. return new RealResponseBody(response.headers(), Okio.buffer(source));
  4. }

从代码中可以看出,首先调用getTransferStream方法就行流转换,因为传入的Response中有头部信息,而头部信息中可能会有编码的信息,所以需要就行转换,然后再创建RealResponseBody对象返回。先看getTransferStream()方法的实现:

  1. private Source getTransferStream(Response response) throws IOException {
  2. if (!HttpHeaders.hasBody(response)) {
  3. return newFixedLengthSource(0);
  4. }
  5. if ("chunked".equalsIgnoreCase(response.header("Transfer-Encoding"))) {
  6. return newChunkedSource(response.request().url());
  7. }
  8. long contentLength = HttpHeaders.contentLength(response);
  9. if (contentLength != -1) {
  10. return newFixedLengthSource(contentLength);
  11. }
  12. // Wrap the input stream from the connection (rather than just returning
  13. // "socketIn" directly here), so that we can control its use after the
  14. // reference escapes.
  15. return newUnknownLengthSource();
  16. }

从代码中可以看到一共可能有四种返回值,分别是以下四种情况:
1. 如果响应主体部分不应有内容,那么返回newFixedLengthSource(0)
2. 如果响应头部中Transfer-Encoding为chunked,即分块了,那么返回newChunkedSource
3. 如果响应中有个具体长度,那么返回newFixedLengthSource,并且指定长度
4. 以上情况均不满足,返回newUnknownLengthSource

总结

至此,OkHttp的网络部分讲解结束。OkHttp中涉及到了几个重要的类,StreamAllocation负责根据请求创建连接,可能是新建一个连接,可能是重用自己内部的连接,也有可能是从连接池中获取连接;而连接的建立就涉及到了Socket的创建以及连接;当连接创建好后,就创建了HttpStream对象,负责操作底层Socket的输出输入流。

在整个OkHttp的工作流程中,在RetryAndFollowupInterceptor中创建StreamAllocation,在ConnectInterceptor中创建连接以及HttpStream对象,在CallServerInterceptor中操作HttpStream进行发送请求和读取响应。

深入理解OkHttp源码(三)——网络操作的更多相关文章

  1. 深入理解OkHttp源码(一)——提交请求

    本篇文章主要介绍OkHttp执行同步和异步请求的大体流程.主要流程如下图: 主要分析到getResponseWidthInterceptorChain方法,该方法为具体的根据请求获取响应部分,留着后面 ...

  2. 深入理解OkHttp源码(二)——获取响应

    首先先看一张流程图,该图是从拆轮子系列:拆 OkHttp 中盗来的,如下: 在上一篇博客深入理解OkHttp源码(一)——提交请求中介绍到了getResponseWithInterceptorChai ...

  3. OKHttp源码解析

    http://frodoking.github.io/2015/03/12/android-okhttp/ Android为我们提供了两种HTTP交互的方式:HttpURLConnection 和 A ...

  4. 从设计模式角度看OkHttp源码

    前言 说到源码,很多朋友都觉得复杂,难理解. 但是,如果是一个结构清晰且完全解耦的优质源码库呢? OkHttp就是这样一个存在,对于这个原生网络框架,想必大家也看过很多很多相关的源码解析了. 它的源码 ...

  5. 【转载】okhttp源码解析

    转自:http://www.open-open.com/lib/view/open1472216742720.html https://blog.piasy.com/2016/07/11/Unders ...

  6. Okhttp源码分析--基本使用流程分析

    Okhttp源码分析--基本使用流程分析 一. 使用 同步请求 OkHttpClient okHttpClient=new OkHttpClient(); Request request=new Re ...

  7. AQS源码三视-JUC系列

    AQS源码三视-JUC系列 前两篇文章介绍了AQS的核心同步机制,使用CHL同步队列实现线程等待和唤醒,一个int值记录资源量.为上层各式各样的同步器实现画好了模版,像已经介绍到的ReentrantL ...

  8. [区块链\理解BTCD源码]GO语言实现一个区块链原型

    摘要 本文构建了一个使用工作量证明机制(POW)的类BTC的区块链.将区块链持久化到一个Bolt数据库中,然后会提供一个简单的命令行接口,用来完成一些与区块链的交互操作.这篇文章目的是希望帮助大家理解 ...

  9. 七、Spring之深入理解AOP源码

    Spring之深入理解AOP源码 ​ 在上一篇博文中,我们对AOP有了初步的了解,那么接下来我们就对AOP的实现原理进行深入的分析. ​ 在之前写的那个AOP示例代码当中有这样一个注解:@Enable ...

随机推荐

  1. 二维条码扫描模组在肯德基KFC的无纸化点餐解决方案

    在如今提倡节约资源的环境下,肯德基在品牌发展中,逐渐实现无纸化点餐,不仅节约了纸质点餐单,而且还具有节约资源的示范作用.而其中二维码扫描模组是这套无纸化点餐方案的重点,在整套设备中,加入二维码扫描模组 ...

  2. kafka 客户端 producer 配置参数

    属性 描述 类型 默认值 bootstrap.servers 用于建立与kafka集群的连接,这个list仅仅影响用于初始化的hosts,来发现全部的servers.格式:host1:port1,ho ...

  3. UOJ#348. 【WC2018】州区划分

    原文链接www.cnblogs.com/zhouzhendong/p/UOJ348.html 前言 第一次知道子集卷积可以自己卷自己. 题解 这是一道子集卷积模板题. 设 $sum[S]$ 表示点集 ...

  4. CF 552 Neko does Maths

    给出两个数a,b 求k     使得 a+k b+k有最小公倍数 a,b同时加上一个非负整数k,使得,a+k,b+k的最小公倍数最小 因为最小公公倍数=x*y / gcd(x,y),所以肯定离不开最大 ...

  5. Codeforces 126B. Password (KMP)

    <题目链接> 题目大意:给定一个字符串,从中找出一个前.中.后缀最长公共子串("中"代表着既不是前缀,也不是后缀的部分). 解题分析:本题依然是利用了KMP中next数 ...

  6. python文件的路径问题补充上一篇内容

    上次的路径问题还没解决就被勒索病毒的木马器给搞了两周多, 拖拖拖到现在又开始纠结路径问题...还是学习能力不足啊... 补充一下路径问题的知识, 毕竟jupyter notebook跟IDE测试的时候 ...

  7. SpringCloud使用Sofa-lookout监控(基于Eureka)

    本文介绍SpringCloud使用Sofa-lookout,基于Eureka服务发现. 1.前景 本文属于是前几篇文章的后续,其实一开始感觉这个没有什么必要写的,但是最近一个朋友问我关于这个的问题,所 ...

  8. POJ 3751 JAVA

    题意: 对于给定的采用”yyyy/mm/dd”加24小时制(用短横线”-”连接)来表示日期和时间的字符串, 请编程实现将其转换成”mm/dd/yyyy”加12小时制格式的字符串,末尾加上pm或者am. ...

  9. Logstash 6.4.3 导入 csv 数据到 ElasticSearch 6.4.3

    本文实践最新版的Logstash从csv文件导入数据到ElasticSearch. 本文目录: 1.初始化ES.Kibana.Logstash 2.安装logstash文件导入.过滤器等插件 3.配置 ...

  10. Python网络编程基础pdf

    Python网络编程基础(高清版)PDF 百度网盘 链接:https://pan.baidu.com/s/1VGwGtMSZbE0bSZe-MBl6qA 提取码:mert 复制这段内容后打开百度网盘手 ...