Java网络编程从入门到精通(14):多种多样的建立网络连接的方式

本文涉及的产品
.cn 域名,1个 12个月
简介: 本文为原创,如需转载,请注明作者和出处,谢谢! 上一篇:Java网络编程从入门到精通(13):使用Socket类接收和发送数据 在上一篇文章中我们讨论了Socket类的基本用法,并给出的例子中使用Socket类连接服务器时使用了一种最简单的连接方式,也就是通过IP和端口号来连接服务器。

本文为原创,如需转载,请注明作者和出处,谢谢!

上一篇:Java网络编程从入门到精通(13):使用Socket类接收和发送数据

在上一篇文章中我们讨论了Socket类的基本用法,并给出的例子中使用Socket类连接服务器时使用了一种最简单的连接方式,也就是通过IP和端口号来连接服务器。而为了使连接服务器的方式更灵活,Socket类不仅可以通过自身的构造方法连接服务器,而且也可以通过connect方法来连接数据库。

一、通过构造方法连接服务器

    我们可以通过6个重载构造函数以不同的方式来连接服务器。这6个重载的构造函数可以分为两类:

1.
自动选择IP

    这种方式是最常用的。所谓自动选择
IP,是指当本机有多块网卡或者在一个网卡上绑定了多个IP时,Socket类会自动为我们选择一个可用的IP。在上述6个构造方法中有4个是使用这种方法来连接服务器的。

(1)    public Socket(String host, int port)

这是最常用的构造方法,在前面的例子中就是使用的这个构造方法。在使用时只需要提供一个字符串类型的IP或域名以及一个整型的端口号即可。在这个构造方法中可能会抛出两个错误:UnknownHostExceptionIOException。发生第一个错误的原因是我们提供的host并不存在或不合法,而其它的错误被归为IO错误。因此,这个构造方法的完整定义是:

public  Socket(String host,  int  port)  throws  UnknownHostException, IOException

(2) public Socket(InetAddress inetaddress, int port)

这个构造方法和第一种构造方法类似只是将字符串形式的host改为InetAddress对象类型了。在这个构造方法中之所以要使用InetAddress类主要是因为考虑到在程序中可能需要使用Socket类多次连接同一个IP或域名,这样使用InetAddress类的效率比较高。另外,在使用字符串类型的host连接服务器时,可能会发生两个错误,但使用InetAddress对象来描述host,只会发生IOException错误,这是因为当你将IP或域名传给InetAddress时,InetAddress会自动检查这个IP或域名,如果这个IP或域名无效,那么InetAddress就会抛出UnknownHostException错误,而不会由Socket类的构造方法抛出。因此,这个构造方法的完整定义是:

public  Socket(InetAddress inetaddress,  int  port)  throws  IOException

(3) public Socket(String host, int port, boolean stream)

这个构造方法和第一种构造方法差不多,只是多了一个boolean类型的stream参数。如果这个streamtrue,那么这个构造方法和第一种构造方法完全一样。如果streamfalse,则使用UDP协议建立一个UDP连接(UDP将在下面的章节详细讨论,在这里只要知道它和TCP最大的区别是UDP是面向无连接的,而TCP是面向有连接的),也许是当初Sun的开发人员在编写Socket类时还未考虑编写处理UDP连接的DatagramSocket类,所以才将建立UDP连接的功能加入到Socket类中,不过Sun在后来的JDK中加入了DatagramSocket类,所以,这个构造方法就没什么用了,因此,Sun将其设为了Deprecated标记,也就是说,这个构造方法在以后的JDK版本中可以会被删除。其于以上原因,在使用Java编写网络程序时,尽量不要使用这个构造方法来建立UDP连接。

(4) public Socket(InetAddress inetaddress, int port, boolean flag)

这个构造方法和第三种构造方法的flag标记的含义一样,也是不建议使用的。

    下面的代码演示
上述4种构造方法的使用:

package mysocket;

import  java.net. * ;
import  java.io. * ;

public   class MoreConnection
{
    
private   static   void  closeSocket(Socket socket)
    {
        
if  (socket  !=   null )
            
try
            {
                socket.close();
            }
            
catch  (Exception e) { }
    }

    
public   static   void  main(String[] args)
    {
        Socket socket1 
=   null , socket2  =   null , socket3  =   null , socket4  =   null ;
        
try
        {
            
//  如果将www.ptpress.com.cn改成其它不存在的域名,将抛出UnknownHostException错误
            
//  测试public Socket(String host, int port)
            socket1  =   new  Socket( " www.ptpress.com.cn " 80 );
            System.out.println(
" socket1连接成功! " );
            
//  测试public Socket(InetAddress inetaddress, int port)
            socket2  =   new  Socket(InetAddress.getByName( " www.ptpress.com.cn " ), 80 );
            System.out.println(
" socket2连接成功! " );

            
//  下面的两种建立连接的方式并不建议使用
            
//  测试public Socket(String host, int port, boolean stream)
            socket3  =   new  Socket( " www.ptpress.com.cn " 80 false );
            System.out.println(
" socket3连接成功! " );
            
//  测试public Socket(InetAddress inetaddress, int i, boolean flag)
            socket4  =   new  Socket(InetAddress.getByName( " www.ptpress.com.cn " ), 80 false );
            System.out.println(
" socket4连接成功! " );
        }
        
catch  (UnknownHostException e)
        {
            System.out.println(
" UnknownHostException 被抛出! " );
        }
        
catch  (IOException e)
        {
            System.out.println(
" IOException 被抛出! " );
        }
        
finally
        {
            closeSocket(socket1);
            closeSocket(socket2);
            closeSocket(socket3);
            closeSocket(socket4);
        }
    }
}

在上面代码中的最后通过finally关闭了被打开的Socket连接,这是一个好习惯。因为只有在将关闭Socket连接的代码写在finally里,无论是否出错,都会执行这些代码。但要注意,在关闭Socket连接之前,必须检查Socket对象是否为null,这是因为错误很可能在建立连接时发生,这样Socket对象就没有建立成功,也就用不着关闭了。

1.       手动绑定IP

当本机有多个IP时(这些IP可能是多块网卡上的,也可能是一块网卡上绑定的多个IP),在连接服务器时需要由客户端确定需要使用哪个IP。这样就必须使用Socket类的另外两个构方法来处理。下面让我们来看看这两个构造方法是如何来使用特定的IP来连接服务器的。

public Socket(String host, int port, InetAddress inetaddress, int localPort)

这个构造方法的参数分为两部分,第一部分为前两个参数:hostport,它们分别表示要连接的服务器的IP和端口号。第二部分为后两个参数:inetaddresslocalPort。其中inetaddress则表示要使用的本地的IP,而localPort则表示要绑定的本地端口号。这个localPort这以设置为本机的任何未被绑定的端口号。如果将localPort的值设为0java将在102465,535之间随即选择一个未绑定的端口号。因此,在一般情况下将localPort设为0

public Socket(InetAddress inetaddress, int port, InetAddress inetaddress1, int localPort)

这个构造方法和第一个构造方法基本相同,只是将第一个参数host换成了inetaddress。其它的使用方法和第一个构造方法类似。

下面的代码中将使用这两个构造方法来做一个实验。我们假设有两台计算机:PC1PC2PC1PC2各有一块网卡。PC1绑定有两个IP192.168.18.252200.200.200.200PC2绑定有一个IP200.200.200.4PC1PC2的子网掩码都是255.255.255.0。而PC1的默认网关为:192.168.28.254。下面的代码需要在PC1上运行。

package mysocket;

import  java.net. * ;

public   class MoreConnection1
{
    
public   static   void  main(String[] args)
    {
        
try
        {
            InetAddress localAddress1 
=  InetAddress.getByName( " 200.200.200.200 " );
            InetAddress localAddress2 
=  InetAddress.getByName( " 192.168.18.252 " );
            // 如果将localAddress1改成localAddress2,socket1无法连接成功
            Socket socket1  =   new  Socket( " 200.200.200.4 " 80 , localAddress1,  0 );
            System.out.println(
" socket1连接成功! " );
            Socket socket2 
=   new  Socket( " www.ptpress.com.cn " 80 , localAddress2,  0 );
            System.out.println(
" socket2连接成功! " );
            
//  下面的语句将抛出一个IOException错误
            Socket socket3  =   new  Socket( " www.ptpress.com.cn " 80 , localAddress1,  0 );
            System.out.println(
" socket3连接成功! " );
            socket1.close();
            socket2.close();
            socket3.close();
        }
        
catch  (Exception e)
        {
            System.out.println(e.getMessage());
        }
    }
}

运行上面代码的输出结果如下:

socket1连接成功 !
socket2连接成功
!
Connection timed out: connect

从上面的输出结果可以看出socket1socket2已经连接成功socket3并未连接成功。从例程4-8可以看出,socket1在连接时使用localAddress1绑定到了200.200.200.200上,而PC2IP200.200.200.4,因此,socket1所使用的IPPC2IP在同一个网段,所以socket1可以连接成功。如果将localAddress1改成localAddress2后,socket1将无法连接成功。另外两个Socket连接socket2socket3是通过Internet连接www.ptpress.com.cn。它们所不同的是socket2绑定的是192.168.18.252,而socket3绑定的是200.200.200.200。它们执行的结果是socket2可以连接成功,而socket3连接失败。这是因为socket2所绑定的IPPC1的默认网关192.168.18.254在同一个网段,因此,socket2可以连接到Internet。而socket3所绑定的IPPC1IP不在同一个网段,因此,socket3将无法连接到Internet

二、通过connect方法连接服务器

Socket类不仅可以通过构造方法直接连接服务器,而且还可以建立未连接的Socket对象,并通过connect方法来连接服务器。Socket类的connect方法有两个重载形式:

1. public void connect(SocketAddress endpoint) throws IOException

Socket类的connect方法和它的构造方法在描述服务器信息IP和端口上有一些差异。在connect方法中并未象构造方法中以字符串形式的host和整数形式的port作为参数,而是直接将IP和端口封装在了SocketAddress类的子类InetSocketAddress中。可按如下形式使用这个connect方法:

Socket socket  =   new  Socket();
socket.connect(
new  InetSocketAddress(host, port));

2.  public void connect(SocketAddress endpoint, int timeout) throws IOException

    这个connect方法和第一个connect类似,只是多了一个timeout参数。这个参数表示连接的超时时间,单位是毫秒。使用timeout设为0,则使用默认的超时时间。

在使用Socket类的构造方法连接服务器时可以直接通过构造方法绑定本地IP,而connect方法可以通过Socket类的bind方法来绑定本地IP。例程4-9演示如何使用connect方法和bind方法。

package mysocket;

import  java.net. * ;

public   class MoreConnection2
{
    
public   static   void  main(String[] args)
    {
        
try
        {
            Socket socket1 
=   new  Socket();
            Socket socket2 
=   new  Socket();
            Socket socket3 
=   new  Socket();
            socket1.connect(
new  InetSocketAddress( " 200.200.200.4 " 80 ));
            socket1.close();
            System.out.println(
" socket1连接成功! " );             
            
/*
               将socket2绑定到192.168.18.252将产生一个IOException错误  
            socket2.bind(new InetSocketAddress("192.168.18.252", 0));
            
*/
            socket2.bind(
new  InetSocketAddress( " 200.200.200.200 " 0 ));
            socket2.connect(
new  InetSocketAddress( " 200.200.200.4 " 80 ));
             
            socket2.close();
            System.out.println(
" socket2连接成功! " );

            socket3.bind(
new  InetSocketAddress( " 192.168.18.252 " 0 ));
            socket3.connect(
new  InetSocketAddress( " 200.200.200.4 " 80 ),  2000 );            
            socket3.close();
            System.out.println(
" socket3连接成功! " );
        }
        
catch  (Exception e)
        {
            System.out.println(e.getMessage());
        }
    }
}

上面的代码的输出结果为:


socket1连接成功 !
socket2连接成功
!
Connection timed out: connect

在上面代码中的socket3连接服务器时为其设置了超时时间(2000毫秒),因此,socket3在非常短的时间就抛出了IOException错误。

下一篇:
Java网络编程从入门到精通(15):为什么要使用SocketAddress来管理网络地址



国内最棒的Google Android技术社区(eoeandroid),欢迎访问!

《银河系列原创教程》发布

《Java Web开发速学宝典》出版,欢迎定购

目录
相关文章
|
10天前
|
监控 安全 数据安全/隐私保护
智能家居安全入门:保护你的网络家园
本文旨在为初学者提供一份简明扼要的指南,介绍如何保护自己的智能家居设备免受网络攻击。通过分析智能家居系统常见的安全漏洞,并提供实用的防御策略,帮助读者建立起一道坚固的数字防线。
|
8天前
|
网络协议 安全 网络安全
C语言 网络编程(四)常见网络模型
这段内容介绍了目前被广泛接受的三种网络模型:OSI七层模型、TCP五层模型以及TCP/IP四层模型,并简述了多个网络协议的功能与特性,包括HTTP、HTTPS、FTP、DNS、SMTP、TCP、UDP、IP、ICMP、ARP、RARP及SSH协议等,同时提到了ssh的免费开源实现openssh及其在Linux系统中的应用。
|
11天前
|
图形学 C#
超实用!深度解析Unity引擎,手把手教你从零开始构建精美的2D平面冒险游戏,涵盖资源导入、角色控制与动画、碰撞检测等核心技巧,打造沉浸式游戏体验完全指南
【8月更文挑战第31天】本文是 Unity 2D 游戏开发的全面指南,手把手教你从零开始构建精美的平面冒险游戏。首先,通过 Unity Hub 创建 2D 项目并导入游戏资源。接着,编写 `PlayerController` 脚本来实现角色移动,并添加动画以增强视觉效果。最后,通过 Collider 2D 组件实现碰撞检测等游戏机制。每一步均展示 Unity 在 2D 游戏开发中的强大功能。
50 6
|
2天前
|
机器学习/深度学习 人工智能 TensorFlow
神经网络入门到精通:Python带你搭建AI思维,解锁机器学习的无限可能
【9月更文挑战第10天】神经网络是开启人工智能大门的钥匙,不仅是一种技术,更是模仿人脑思考的奇迹。本文从基础概念入手,通过Python和TensorFlow搭建手写数字识别的神经网络,逐步解析数据加载、模型定义、训练及评估的全过程。随着学习深入,我们将探索深度神经网络、卷积神经网络等高级话题,并掌握优化模型性能的方法。通过不断实践,你将能构建自己的AI系统,解锁机器学习的无限潜能。
8 0
|
11天前
|
开发者 图形学 API
从零起步,深度揭秘:运用Unity引擎及网络编程技术,一步步搭建属于你的实时多人在线对战游戏平台——详尽指南与实战代码解析,带你轻松掌握网络化游戏开发的核心要领与最佳实践路径
【8月更文挑战第31天】构建实时多人对战平台是技术与创意的结合。本文使用成熟的Unity游戏开发引擎,从零开始指导读者搭建简单的实时对战平台。内容涵盖网络架构设计、Unity网络API应用及客户端与服务器通信。首先,创建新项目并选择适合多人游戏的模板,使用推荐的网络传输层。接着,定义基本玩法,如2D多人射击游戏,创建角色预制件并添加Rigidbody2D组件。然后,引入网络身份组件以同步对象状态。通过示例代码展示玩家控制逻辑,包括移动和发射子弹功能。最后,设置服务器端逻辑,处理客户端连接和断开。本文帮助读者掌握构建Unity多人对战平台的核心知识,为进一步开发打下基础。
32 0
|
11天前
|
网络协议 C# 开发者
WPF与Socket编程的完美邂逅:打造流畅网络通信体验——从客户端到服务器端,手把手教你实现基于Socket的实时数据交换
【8月更文挑战第31天】网络通信在现代应用中至关重要,Socket编程作为其实现基础,即便在主要用于桌面应用的Windows Presentation Foundation(WPF)中也发挥着重要作用。本文通过最佳实践,详细介绍如何在WPF应用中利用Socket实现网络通信,包括创建WPF项目、设计用户界面、实现Socket通信逻辑及搭建简单服务器端的全过程。具体步骤涵盖从UI设计到前后端交互的各个环节,并附有详尽示例代码,助力WPF开发者掌握这一关键技术,拓展应用程序的功能与实用性。
31 0
|
11天前
|
机器学习/深度学习 人工智能 自然语言处理
深度学习中的卷积神经网络(CNN)入门
【8月更文挑战第31天】在人工智能的浪潮中,深度学习以其强大的数据处理能力成为时代的宠儿。本文将引导你走进深度学习的核心组件之一——卷积神经网络(CNN),并带你一探其背后的奥秘。通过简明的语言和直观的代码示例,我们将一起构建一个简易的CNN模型,理解它在图像处理领域的应用,并探索如何利用Python和TensorFlow实现它。无论你是初学者还是有一定基础的开发者,这篇文章都将为你打开一扇通往深度学习世界的大门。
|
11天前
|
Java 程序员 UED
Java 中的异常处理:从入门到精通
【8月更文挑战第31天】在Java编程的世界中,异常处理是保持应用稳定性的重要机制。本文将引导你理解异常的本质,学会如何使用try-catch语句来捕获和处理异常,并探索自定义异常类的魅力。我们将一起深入异常的世界,让你的代码更加健壮和用户友好。
|
11天前
|
机器学习/深度学习 人工智能 TensorFlow
深度学习入门:使用Python和TensorFlow构建你的第一个神经网络
【8月更文挑战第31天】 本文是一篇面向初学者的深度学习指南,旨在通过简洁明了的语言引导读者了解并实现他们的第一个神经网络。我们将一起探索深度学习的基本概念,并逐步构建一个能够识别手写数字的简单模型。文章将展示如何使用Python语言和TensorFlow框架来训练我们的网络,并通过直观的例子使抽象的概念具体化。无论你是编程新手还是深度学习领域的新兵,这篇文章都将成为你探索这个激动人心领域的垫脚石。
|
11天前
|
机器学习/深度学习 自动驾驶 算法框架/工具
深度学习中的卷积神经网络(CNN)入门
【8月更文挑战第31天】 本文旨在通过浅显易懂的方式,引导初学者步入卷积神经网络(CNN)的神秘世界。我们将从CNN的基础概念出发,逐步深入到其在图像处理中的应用实例,最后通过一个简单的Python代码示例,展示如何实现一个基础的CNN模型。无论你是编程新手还是深度学习领域的初探者,这篇文章都将为你打开一扇了解和掌握CNN的大门。