Socket这个类不是java.io包中的,而在java.net包中。以至于Bruce Eckel在他的《Thinking in Java》中并没有怎么提及Socket,更不要说在IO章节中描述Socket。这位作者仅仅是在他的《Thinking in Enterprise Java》中非常简要地介绍了一下java.net.Socket这个类。
我们在本篇文章稍微详细一些地来看一下java.net.Socket这个类的情况。
先看一下类的声明:
1
|
public
class
Socket
implements
java.io.Closeable
|
可见Socket仅仅是实现了Closeable接口,给出了close()方法,并没有继承任何其他类(除Object)。这一点和SocketInputStream/SocketOutputStream继承于FileInputStream/FileOutputStream不同。
实际上,Socket这个类也仅仅是一层外在的包装。如果我们打开JDK中Socket的源码会看到,各类操作方法实际上都是对impl属性的一个封装。我们来看下,Socket类中有这样两个比较重要的属性:
- SocketImpl impl; 注意,这个是包级别可见。
- private static SocketImplFactory factory = null; 这个默认情况下是null。
除了属性相关的,Socket有connect()、bind()、getInputStream()、getOutputStream()等几个方法。而这样几个方法实际上又是调用了SocketImpl类的impl对象属性的同名方法,我们不得不看看SocketImpl和SocketImplFactory这两个类/接口。
首先,这俩都是java.net包中的。SocketImpl是一个public的抽象类,SocketImplFactory是一个接口。后者是前者的工厂类,即有构造前者对象的方法。
1
|
public
abstract
class
SocketImpl
implements
SocketOptions
|
和SocketImpl这个名字有些不符,这个类实际上是一个抽象类(Impl一般是具体实现)。SocketOptions定义了各个选项ID常量和相关的方法。实际上在Socket类中的各个set和get属性的方法中,属性也都是在SocketImpl中定义的。而对于实际的具体的操作,还都是由SocketImpl的具体子类来完成。
如图,可见Windows下JDK中的SocketImpl类及子类层次结构。像connect()这种方法,最终还是通过native方式依靠底层来实现。
回过头来,我们再来看下Socket的构造方法。Socket有10个重载的构造方法,其中有2个是@Deprecated的,还有1个protected和1个private重载。而这些方法中大部分都最终调用了这个private的实现。通常情况下,我们构造Socket的时候回给出要连接的服务端地址和端口号。这最终也会调用private重载构造方法实现中。