Java攻城狮的入门课程@IT·互联网Java学习笔记

(一)Java中网络相关API的应用

2017-05-15  本文已影响95人  黒猫

1、网络基础知识

在介绍后续Java Socket的相关知识之前,需要各位对于网络的基础知识有一定了解,关于此部分内容可以到已整理完成的网络基础知识专题查看。
  
  这里再补充以下内容:

  • 所谓的Socket,可以简单理解为网络上运行的程序之间双向通讯链路的终结点,由IP地址端口号组成,是TCP和UDP的基础,也是整个网络通信的基础。

2、InetAddress的应用

InetAddress类用于标识网络上的硬件资源,表示互联网协议(IP)地址,虽然InetAddress类没有静态方法,不能直接new一个对象,但是可以通过其静态方法来获取一个对象,示例代码如下:

import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.Arrays;

public class Deom1 {

    public static void main(String[] args) throws UnknownHostException {
        /*
         * 获取本机的InetAddress实例 
         * 会提示出现未知主机异常 
         * 可以选择直接抛出异常
         */
        InetAddress address = InetAddress.getLocalHost();
        // 获取计算机名
        System.out.println("计算机名:" + address.getHostName());
        // 获取IP地址
        System.out.println("IP地址:" + address.getHostAddress());
        // 获取字节数组形式的IP地址
        byte[] bytes = address.getAddress();
        System.out.println("字节数组形式的IP地址:" + Arrays.toString(bytes));
    }

}

此时的结果为:

之所以字节数组形式的IP地址出现负值,是因为byte类型长度只有8个二进制位,取值范围是-128—127,并且是有符号类型的。而IP地址的每一段八位二进制位是无符号的,例如十进制数192的二进制形式是11000000,但是放在字节数组中,第一位的1就变成了代表负号,即-1000000,关于二进制数转化成十进制的计算方法可参见进制转换,1000000转换成十进制数是64,因此-1000000即十进制数-64。
  
  解决该问题的方式也很简单,代码如下:

        System.out.print("转换后的IP地址:");
        // 使用增强for循环遍历数组形式的IP地址
                for (byte before : bytes) {
                    /*
                     * 使用三元运算符 
                     * 当遍历出的元素小于0时 
                     * 就让该元素加上256 
                     * 反之就保留该值
                     */
                    int after = (before < 0) ? 256 + before : before;
                    System.out.print(after + " ");
                }

此时的结果为:

还可以通过其他途径获取InetAddress实例,示例代码如下:

import java.net.InetAddress;
import java.net.UnknownHostException;

public class Demo2 {

    public static void main(String[] args) throws UnknownHostException {
        // 通过计算机名获取InetAddress实例
        InetAddress address = InetAddress.getByName("Z7-SL7-S3");
        // 获取计算机名
        System.out.println("计算机名:" + address.getHostName());
        // 获取IP地址
        System.out.println("IP地址:" + address.getHostAddress());

        // 通过IP地址获取InetAddress实例
        InetAddress address2 = InetAddress.getByName("192.168.199.202");
        // 获取计算机名
        System.out.println("计算机名:" + address2.getHostName());
        // 获取IP地址
        System.out.println("IP地址:" + address2.getHostAddress());

    }
}

此时的结果为:

至于通过IP地址获取的实例得出的主机名带有“.lan”的后缀,是微软自动添加的,其他后缀还有“.net”等。


3、URL的应用

URL(Uniform Resource Locator)即统一资源定位符,表示网络上某一资源的地址;URL由两部分组成,分别是协议名称和资源名称,二者之间用“:”隔开,以简书的网站为例:

http://www.jianshu.com

“http”指的是协议名称,即超文本传输协议,而“www.jianshu.com”则表示资源名称。在java.net包中,提供了URL类来表示URL,下面通过示例代码来介绍URL的常用方法:

import java.net.MalformedURLException;
import java.net.URL;

public class Demo3 {

    public static void main(String[] args) {

        try {
            /*
             * 创建一个URL实例
             * 此时会出现异常,使用try-catch块捕获该异常
             */
            URL jianshu = new URL("http://www.jianshu.com");
            /*
             * 使用构造方法:URL(URL context, String spec) 
             * 根据已经存在的URL
             * 以及相应的字符串spec,也就是其他资源的相关信息
             * 来创建新的URL实例
             */
            URL url = new URL(jianshu, "/index.html?username=hm#test");
            /*
             * 其中?起连接作用
             * 后面的字符表示参数信息,例如用户名为hm
             * 
             * #代表网页中的一个位置
             * 后面的字符,就是该位置的标识符,即锚点
             */
            
            //之后就可以使用URL类所提供的方法来获取相关信息
            System.out.println("协议:"+url.getProtocol());//获取协议名称
            System.out.println("主机:"+url.getHost());//获取主机地址
            System.out.println("端口:"+url.getPort());//获取端口号
            System.out.println("文件路径:"+url.getPath());//获取文件路径
            System.out.println("文件名:"+url.getFile());//文件名包括文件路径及参数
            System.out.println("相对路径:"+url.getRef());//相对路径即锚点
            System.out.println("查询字符串:"+url.getQuery());//查询字符串即参数信息
            

        } catch (MalformedURLException e) {
            e.printStackTrace();
        }

    }

}

此时的结果为:



  这里要注意,端口显示为-1的原因是:当指定URL资源时,未指定端口号的情况下,会根据不同的协议使用默认的端口号,即此时根据http协议使用80端口;对getPort()方法而言,当使用默认端口号时,返回值就是-1。

还可以使用URL读取网页上的内容,通过URL实例提供的openStream()方法可以得到指定资源的输入流,通过该输入流可以访问并读取网络上对应的数据,关于输入流等此部分内容可以到已整理完成的文件传输基础——Java IO流专题查看,示例代码如下:

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.MalformedURLException;
import java.net.URL;

public class Demo4 {

    public static void main(String[] args) {

        try {
            //创建一个URL实例
            URL url = new URL("http://www.jianshu.com");
            /*
             * 通过openStream()方法获取URL对象所表示的资源的字节输入流
             * 此时会出现异常,使用try-catch块捕获该异常
             * 注意导入java.io包中的InputStream
             */
                InputStream is = url.openStream();
                //将字节输入流转换为字符输入流
                InputStreamReader isr = new InputStreamReader(is);
                //为字符输入流添加缓冲
                BufferedReader br = new BufferedReader(isr);
                //读取数据,一次读取一行
                String data = br.readLine();
                //只要数据不为空,就一直读取
                while(data!=null){
                    System.out.println(data);
                    data = br.readLine();
                }
                //及时关闭相应资源
                br.close();
                is.close();
        } catch (MalformedURLException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }

    }

}

此时的结果为:

实际上获取的结果就是简书网站的首页页面,但是可以看到页面中的中文汉字显示乱码,在浏览结果时可以看到:

因此可进行如下操作:

//在将字节输入流转换为字符输入流时,指定字符集
InputStreamReader isr = new InputStreamReader(is,"utf-8");

此时的结果为:


版权声明:欢迎转载,欢迎扩散,但转载时请标明作者以及原文出处,谢谢合作!             ↓↓↓
上一篇下一篇

猜你喜欢

热点阅读