9、網(wǎng)絡(luò)

一、基本概念

  • 1、網(wǎng)絡(luò)
    將不同區(qū)域的計算機(jī)連接到一起,按范圍的從小到大順序分成局域網(wǎng)、城域網(wǎng)、互聯(lián)網(wǎng)。
  • 2、地址:IP地址,確定網(wǎng)絡(luò)上一個絕對的地址
  • 3、端口號:區(qū)分計算機(jī)上各個軟件,共2個字節(jié),0-65535,共65536個端口號
    注意:
    • 3.1:在同一個協(xié)議下端口號不能重復(fù),不同協(xié)議下可以重復(fù)
    • 3.2:1024一下的端口號是預(yù)留給一些知名廠商的,最好不要使用。
  • 4、資源定位:URL統(tǒng)一資源定位符,URI表示統(tǒng)一資源,屬于URL的一部分。
  • 5、數(shù)據(jù)的傳輸
    • 5.1 協(xié)議:TCP協(xié)議和UDP協(xié)議
      TCP:必須先連接上再通信,“三次握手”。面向連接,安全可靠。但是效率相對低下。類似于打電話。
      UDP:非面向連接,安全性較低,但是效率較高。類似于發(fā)短信。
    • 5.2 傳輸
      需要先封裝,接收到的數(shù)據(jù)需要拆分。

二、相關(guān)類的使用

2.1 InetAddress類和InetSocketAddress類

2.1.1 InetAddress類

InetAddress類封裝計算機(jī)的ip地址和DNS,不包含端口。

  • 1、靜態(tài)方法獲取對象:
InetAddress.getLocalHost();
InetAddress.getByName("www.163.com");
InetAddress.getByName("192.168.2.23");

這單個方法都返回InetAddress實例對象。

  • 2、方法
getHostAddress()返回ip
getHostName()返回域名,本機(jī)為計算機(jī)名

相關(guān)例子:

package cn.itcast.day185.InetAddress_InetSocketAddress;
import java.net.InetAddress;
import java.net.UnknownHostException;
public class InetDemo01 {
    public static void main(String[] args) throws UnknownHostException {
        InetAddress addr = InetAddress.getLocalHost();
        System.out.println(addr.getHostAddress());//10.170.48.193
        System.out.println(addr.getHostName());//yj-PC
        
        addr = InetAddress.getByName("www.163.com");
        System.out.println(addr.getHostAddress());//219.145.184.60
        System.out.println(addr.getHostName());//www.163.com
        
        addr = InetAddress.getByName("219.145.184.60");
        System.out.println(addr.getHostAddress());//219.145.184.60
        System.out.println(addr.getHostName());//可能是219.145.184.60或者是www.163.com
    }
}

2.1.2 InetSocketAddress類

InetSocketAddress是在InetAddress基礎(chǔ)上封裝了端口號。

  • 1、靜態(tài)方法
public static InetSocketAddress createUnresolved(String host, int port)
  • 2、構(gòu)造器
public InetSocketAddress(InetAddress addr, int port)
public InetSocketAddress(String hostname, int port)//用的最多
public InetSocketAddress(int port)

相關(guān)例子:

package cn.itcast.day185.InetAddress_InetSocketAddress;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.UnknownHostException;

public class InetSocketDemo01 {
    public static void main(String[] args) throws UnknownHostException {
        InetSocketAddress address = new InetSocketAddress("localhost", 9999);
        System.out.println(address.getHostName());//localhost
        System.out.println(address.getPort());//9999
        
        InetAddress addr = address.getAddress();
        System.out.println(addr.getHostAddress());//0:0:0:0:0:0:0:1
        System.out.println(addr.getHostName());//localhost  
    }
}

2.2 URL和URI

  • URI:統(tǒng)一資源標(biāo)識符,用來唯一的標(biāo)識一個資源
  • URL:統(tǒng)一資源定位器,是一種具體的URI,即包含URI
  • URL:由四部分組成
    協(xié)議(默認(rèn)),存放資源的主機(jī)域名,端口號(默認(rèn)是80),資源文件名URL類。創(chuàng)建用的比較多的是:
public URL(String spec)//絕對路徑構(gòu)建
public URL(URL context, String spec)//相對路徑構(gòu)建

例1:

package cn.itcast.day186.URL;
import java.net.MalformedURLException;
import java.net.URL;
public class URLDemo01 {
    public static void main(String[] args) throws MalformedURLException {
        
        //絕對路徑構(gòu)建
        URL url = new URL("http://www.baidu.com:80/index.html#aa?username=csdn");
        System.out.println("協(xié)議:" + url.getProtocol());//http
        System.out.println("域名: " + url.getHost());//www.baidu.com
        System.out.println("端口: " + url.getPort());//80
        System.out.println("資源: " + url.getFile());//  /index.html
        System.out.println("相對路徑: " + url.getPath());// /index.html
        System.out.println("錨點: " + url.getRef());//aa?username=csdn
        System.out.println("參數(shù): " + url.getQuery());//存在錨點則為空null,否在返回正確
        
        //相對路徑構(gòu)建
        url = new URL("http://www.baidu.com/a/");
        url = new URL(url, "b.txt");
        System.out.println(url.toString());//http://www.baidu.com/a/b.txt
    }
}

例2:獲取網(wǎng)絡(luò)資源

package cn.itcast.day186.URL;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.net.MalformedURLException;
import java.net.URL;
//獲取資源:源代碼
public class URLDemo02 {
    public static void main(String[] args) throws IOException {
        
        URL url = new URL("http://www.baidu.com");//訪問主頁,即默認(rèn)資源
        
        //獲取資源網(wǎng)絡(luò)流
        /*InputStream in = url.openStream();
        byte[] buffer = new byte[1024];
        int len = 0;
        while((len = in.read(buffer)) != -1){
            System.out.println(new String(buffer, 0, len));
        }
        in.close();*/
        
        //使用轉(zhuǎn)換流
        BufferedReader br = new BufferedReader(new InputStreamReader(url.openStream(), "UTF-8"));
        BufferedWriter bw = new BufferedWriter(
                new OutputStreamWriter(new FileOutputStream("C:\\Users\\yj\\Desktop\\test.html"), "UTF-8"));
        String message = null;
        while((message = br.readLine()) != null){
            //System.out.println(message);
            bw.append(message);
            bw.newLine();
        }
        bw.flush();
        bw.close();
        br.close();
    }
}

說明:這里獲取網(wǎng)絡(luò)資源比較簡單,主要注意url.openStream()的使用。

三、UDP

UDP以數(shù)據(jù)為中心,不安全,非面向連接,數(shù)據(jù)可能丟失,但是效率較高。

3.1 類DatagramSocket和DatagramPacket

使用步驟:

  • 1、客戶端
    (1)創(chuàng)建客戶端DatagramSocket類 + 指定端口
    (2)準(zhǔn)備數(shù)據(jù) 必須使用字節(jié)數(shù)組
    (3)打包DatagramPacket+服務(wù)器的地址和端口
    (4)發(fā)送
    (5)釋放資源

  • 2、服務(wù)器端
    (1)創(chuàng)建服務(wù)端DatagramSocket類+ 指定端口
    (2)準(zhǔn)備接收的容器(字節(jié)數(shù)組),封裝DatagramPacket
    (3)使用包接收數(shù)據(jù)
    (4)分析數(shù)據(jù)
    (5)釋放資源

  • 3、注意
    發(fā)送的數(shù)據(jù)一定要轉(zhuǎn)換成字節(jié)數(shù)組,同時udp不是面向連接的,就算服務(wù)端沒有開啟,客戶端這邊運(yùn)行也不會報錯。

下面通過一個例子進(jìn)行說明,
客戶端:

package cn.itcast.day187.UDP;
import java.io.ByteArrayOutputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetSocketAddress;
//客戶端
//注意:要先開啟服務(wù)端,但是先開啟客戶端也不會報錯,但是數(shù)據(jù)會丟失。因為不是面向連接的
public class MyClient {
    public static void main(String[] args) throws IOException {
        //1、創(chuàng)建客戶端+端口
        DatagramSocket client = new DatagramSocket(6666);//注意和服務(wù)端的端口不要沖突
        //2、準(zhǔn)備數(shù)據(jù),數(shù)據(jù)必須是字節(jié)數(shù)組
        String message = "udp編程";
        
        //double 類型,需要將double類型轉(zhuǎn)換為字節(jié)數(shù)組
        double num = 89.12;
        
        //byte[] data = message.getBytes();
        byte[] data = convert(num);
        
        //3、將數(shù)據(jù)打包, 需要發(fā)送地址和端口
        DatagramPacket packet = new DatagramPacket(data, data.length, new InetSocketAddress("localhost", 8888));
        //4、發(fā)送數(shù)據(jù)
        client.send(packet);
        //5、釋放資源
        client.close();
    }
    //double-->字節(jié)數(shù)組
    public static byte[] convert(double num) throws IOException{
        byte[] data = null;
        ByteArrayOutputStream bos = new ByteArrayOutputStream();
        DataOutputStream dos = new DataOutputStream(bos);
        dos.writeDouble(num);
        dos.flush();
        
        //獲得數(shù)據(jù)
        data = bos.toByteArray();
        dos.close();
        
        return data;
    }
}

服務(wù)端:

package cn.itcast.day187.UDP;
import java.io.ByteArrayInputStream;
import java.io.DataInputStream;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
//服務(wù)端
public class MyServer {
    
    public static void main(String[] args) throws IOException {
        //1、創(chuàng)建服務(wù)器+ 端口
        DatagramSocket server = new DatagramSocket(8888);//有可能端口被占用
        //2、準(zhǔn)備接收容器
        byte[] container = new byte[1024];
        //3、封裝成包,下面這個構(gòu)造函數(shù)是用來接收的
        DatagramPacket packet = new DatagramPacket(container, container.length);
        //4、接收數(shù)據(jù)
        server.receive(packet);
        //5、分析數(shù)據(jù)
        //byte[] data = packet.getData();
        
        double data = convert(packet.getData());
        int len = packet.getLength();
        
        //System.out.println(new String(data, 0, len));
        System.out.println(data);
        
        //6、釋放資源
        server.close();
    }
    
    //字節(jié)數(shù)組-->Data 輸入流
    public static double convert(byte[] data) throws IOException{
        DataInputStream dis = new DataInputStream(new ByteArrayInputStream(data));
        double num = dis.readDouble();
        dis.close();
        return num;
    }
}

說明:主要的使用步驟在程序中已經(jīng)標(biāo)明,要注意基本數(shù)據(jù)類型轉(zhuǎn)換成字節(jié)數(shù)組的方法。

四、TCP

socket通信:我們可以將socket理解為服務(wù)端和客戶端之間的一條通道

  • (1)基于tcp協(xié)議,建立穩(wěn)定連接的點對點的通信
    實時、快速、安全性高、占用系統(tǒng)資源多、效率低
  • (2)請求-響應(yīng) 模式
    • 客戶端(Socket
      在網(wǎng)絡(luò)通信中,第一次主動發(fā)起通訊的程序被稱作客戶端程序,客戶端的端口是自動分配的
    • 服務(wù)端(ServerSocket
      第一次通訊中等待連接的程序被稱作服務(wù)端程序,服務(wù)端的端口需要我們自己指定

相關(guān)例子
客戶端:

package cn.itcast.day189.socket_;
import java.io.DataInputStream;
import java.io.IOException;
import java.net.Socket;
import java.net.UnknownHostException;
//必須先啟動服務(wù)器后連接
public class MyClient {
    public static void main(String[] args) throws UnknownHostException, IOException {
        //1、創(chuàng)建客戶端,必須指定服務(wù)器+端口,此時就在連接,如果不啟動服務(wù)器則會報異常
        Socket socket = new Socket("localhost", 8888);
        //2.接收數(shù)據(jù)
        /*BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        String message = br.readLine();//阻塞式方法,在讀取數(shù)據(jù)時必須要有行結(jié)束符*/
        DataInputStream dis = new DataInputStream(socket.getInputStream());
        String message = dis.readUTF();
        System.out.println(message);
    }
}

服務(wù)端:

package cn.itcast.day189.socket_;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

public class MyServer {
    
    public static void main(String[] args) throws IOException {
        //1、創(chuàng)建服務(wù)端,指定端口,這里可以指定端口為8888,即使客戶端也是用8888,因為不同協(xié)議端口可以重復(fù)
        ServerSocket serverSocket = new ServerSocket(8888);
        //2、接收客戶端的連接,阻塞式的
        Socket socket = serverSocket.accept();
        System.out.println("一個客戶端建立連接");//當(dāng)運(yùn)行程序,然后在瀏覽器訪問此端口,則會打印
        
        //3.發(fā)送數(shù)據(jù)和接收數(shù)據(jù)
        String message = "歡迎使用";
        /*
        //輸出流
        BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
        bw.write(message);
        bw.newLine();//向流中添加一個行結(jié)束符
        bw.flush();
        //不要關(guān)閉
        */
        DataOutputStream dos = new DataOutputStream(socket.getOutputStream());
        dos.writeUTF(message);
        dos.flush();    
    }
}

說明:這里我們只是給出了最基本的使用方法,更深層次的內(nèi)容還需要再研究。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,914評論 18 139
  • 1. 網(wǎng)絡(luò)編程概述 1.1 計算機(jī)網(wǎng)絡(luò) 是指將地理位置不同的具有獨(dú)立功能的多臺計算機(jī)及其外部設(shè)備,通過通信線路連接...
    JackChen1024閱讀 1,049評論 0 3
  • 情人節(jié)那天起,不再擔(dān)心沒有情人,而是擔(dān)心自己的命運(yùn),人生一輩子轉(zhuǎn)折點就那么幾次。且行且珍惜。 ??? 上...
    大朵叨叨閱讀 344評論 2 3
  • 今天是特種兵培訓(xùn)第二天,今天的主題一直都是我們做微商講課的重點話題,今天再次聽一遍,收貨還是很大。 今天下午出門去...
    希希媽咪愛閱讀 195評論 0 0
  • 匯總 滑動條控件.步進(jìn)控件.開關(guān)控件.選項卡控件 UISlider 滑動條 //1.UISlider滑塊控件UIS...
    nothing_c閱讀 273評論 0 0