文章标题 java网络详解

¯网络基本概念
网络体系结构:国际标准化组织ISO于l978年提出“开放系统互连参考模型”,即著名的OSI(Open System Interconnection)模型。
该模型把计算机网络分成物理层、数据链路层、网络层、传输层、会话层、表示层、应用层等七层。
文章标题 java网络详解_第1张图片

至于各个层的具体描述在此省略

InetAddress
InetAddress类是Java中用于描述IP地址的类。它在java.net包中。在Java中分别用Inet4Address和Inet6Address类来描述IPv4和IPv6的地址。这两个类都是InetAddress的子类。由于InetAddress没有public的构造方法,因此,要想创建InetAddress对象,必须得依靠它的四个静态方法。InetAddress可以通过getLocalHost方法得到本机的InetAddress对象,也可以通过getByName、getAllByName和getByAddress得到远程主机的InetAddress对象
使用getLocalHost可以得到描述本机IP的InetAddress对象。这个方法的定义如下:

使用getLocalHost可以得到描述本机IP的InetAddress对象。这个方法的定义如下:

public static InetAddress getLocalHost() throws UnknownHostException

可以通过指定域名从DNS中得到相应的IP地址

public static InetAddress getByName() throws UnknownHostException

从DNS上得到域名对应的所有的IP

public static InetAddress[] getAllByName(String host) throws UnknownHostException

通过IP地址来创建InetAddress对象

public static InetAddress getByAddress(byte[] addr) throws UnknownHostException
public static InetAddress getByAddress(String host, byte[] addr) throws UnknownHostException

什么是UDP协议?
UDP( User Datagram Protocol )协议是用户数据报,在网络中它与TCP协议一样用于处理数据包。在OSI模型中,在第四层——传输层,处于IP协议的上一层。
UDP是一种无连接的协议,每个数据报都是一个独立的信息,包括完整的源或目的地址,它在网络上以任何可能的路径传往目的地,因此能否到达目的地,
到达目的地的时间以及内容的正确性都是不能被保证的。
¯为什么要使用UDP?
在网络质量令人不十分满意的环境下,UDP协议数据包丢失会比较严重。但是由于UDP的特性:它不属于连接型协议,因而具有资源消耗小,
处理速度快的优点,所以通常音频、视频和普通数据在传送时使用UDP较多,因为它们即使偶尔丢失一两个数据包,也不会对接收结果产生太大影响。
比如聊天用的ICQ和OICQ就是使用的UDP协议

¯在Java中操纵UDP
使用位于JDK中Java.net包下的DatagramSocket和DatagramPacket类,可以非常方便地控制用户数据报文。
DatagramSocket类:创建接收和发送UDP的Socket实例
DatagramSocket():创建实例。通常用于客户端编程,它并没有特定监听的端口,仅仅使用一个临时的。
DatagramSocket(int port):创建实例,并固定监听Port端口的报文。
DatagramSocket(int port, InetAddress localAddr):这是个非常有用的构建器,当一台机器拥有多于一个IP地址的时候,
由它创建的实例仅仅接收来自LocalAddr的报文

注意:在创建DatagramSocket类实例时,如果端口已经被使用,会产生一个SocketException的异常抛出,并导致程序非法终止,这个异常应该注意捕获
üreceive(DatagramPacket d):接收数据报文到d中。receive方法产生一个“阻塞”。
üsend(DatagramPacket d):发送报文d到目的地。
üsetSoTimeout(int timeout):设置超时时间,单位为毫秒。
üclose():关闭DatagramSocket。在应用程序退出的时候,通常会主动释放资源,关闭Socket,但是由于异常地退出可能造成资源无法回收。所以,
应该在程序完成时,主动使用此方法关闭Socket,或在捕获到异常抛出后关闭Socket。

“阻塞”是一个专业名词,它会产生一个内部循环,使程序暂停在这个地方,直到一个条件触发。

µDatagramPacket:用于处理报文,将byte数组、目标地址、目标端口等数据包装成报文或者将报文拆卸成byte数组。
üDatagramPacket(byte[] buf, int length, InetAddress addr, int port):从buf数组中,取出length长的数据创建数据包对象,目标是addr地址,port端口。
üDatagramPacket(byte[] buf, int offset, int length, InetAddress address, int port):从buf数组中,取出offset开始的、length长的数据创建数据包对象,目标是addr地址,port端口。
üDatagramPacket(byte[] buf, int offset, int length):将数据包中从offset开始、length长的数据装进buf数组。
üDatagramPacket(byte[] buf, int length):将数据包中length长的数据装进buf数组。
ügetData():它从实例中取得报文的byte数组编码。

实例1,编写程序演示使用UDP协议数据报的发送和接受。
发送端
建立udpsocket服务端点。该端点建立,系统会随机分配一个端口。如果不想随机配置,可以手动指定。
DatagramSocket ds = new DatagramSocket(9002);
将数据进行packet包的封装,必须要指定目的地地址和端口。
byte[] buf = “wo shi shu ju”.getBytes();
DatagramPacket dp =new DatagramPacket(buf,buf.length,InetAddress.getByName(“192.168.1.254”),9001);
通过socket服务的send方法将该包发出。
ds.send(dp);
将socket服务关闭。主要是关闭资源。
ds.close();

接收端
建立udp的socket服务。要监听一个端口。
DatagramSocket ds = new DatagramSocket(9001);
定义一个缓冲区,将该缓冲区封装到packet包中。
byte
[] buf = new byte[1024];
DatagramPacket dp = new DatagramPacket(buf,buf.length);
通过socket的receive方法将数据存入数据包中。
ds.receive(dp);
通过数据包dp的方法getData()、getAddress()、getPort()等方法获取包中的指定信息。
关闭socket。
ds.close();

实例2,完成一个聊天程序。
思路:聊天程序,就要使用到多线程package cn.com.net;

import java.awt.List;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;

import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextField;

public class ChatDemo2 extends JFrame{

/**
* @param args
*/

List list=new List(6);
//TextArea ta=new TextArea(10,40);
JTextField tfIp=new JTextField(15);
JTextField tfData=new JTextField(15);
JPanel p=new JPanel();

DatagramSocket ds=null;

public ChatDemo2(){
this.add(list,”Center”);

p.add(tfIp);
p.add(tfData);
this.add(p,”South”);
this.setSize(400,200);
this.setLocation(300,300);
this.setVisible(true);

try {
ds=new DatagramSocket(9009);
} catch (SocketException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}

new Thread(new Runnable(){

@Override
public void run() {
byte [] buf=new byte[1024];
DatagramPacket dp=new DatagramPacket(buf,buf.length);

while(true){

try {
ds.receive(dp);
//list.add(dp.getData().toString());
list.add(new String(buf,0,dp.getLength())+” from “+dp.getAddress().getHostAddress(),0);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}

}).start();

this.tfData.addActionListener(new ActionListener(){
@Override
public void actionPerformed(ActionEvent e) {
byte[] buf=tfData.getText().getBytes();
list.add(tfData.getText(),0);
DatagramPacket dp;
try {
dp = new DatagramPacket(buf,buf.length,InetAddress.getByName(tfIp.getText()),9009);
ds.send(dp);

} catch (Exception e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}

tfData.setText(“”);

}

});
}

public static void main(String[] args) {
ChatDemo2 chat=new ChatDemo2();
}
}rverSocket:编写TCP网络服务程序,首先要用到java.net.ServerSocket类用以创建服务器Socket
构造方法:
ServerSocket(int port):创建绑定到特定端口的服务器套接字
ServerSocket(int port, int backlog):利用指定的backlog(服务器忙时保持连接请求的等待客户数量),创建服务器套接字并将其绑定到指定的本地端口号。
ServerSocket(int port, int backlog, InetAddress bindAddr):使用指定的端口、侦听 backlog 和要绑定到的本地 IP 地址创建服务器。

¯Socket:客户端要与服务器建立连接,必须先创建一个Socket对象
常用构造方法
Socket(String host, int port):创建一个流套接字并将其连接到指定主机上的指定端口号。
Socket(InetAddress address, int port):创建一个流套接字并将其连接到指定 IP 地址的指定端口号。

服务器端程序调用ServerSocket类中的accept()方法等待客户端的连接请求,一旦accept()接收了客户端连接请求,
该方法返回一个与该客户端建立了专线连接的Socket对象,不用程序去创建这个Socket对象。建立了连接的两个Socket
是以IO流的方式进行数据交换的,Java提供了Socket类中的getInputStream()返回Socket的输入流对象,getOutputStream()
返回Socket的输出流对象。
文章标题 java网络详解_第2张图片


文章标题 java网络详解_第3张图片

小实例:

//接收由发送端键盘输入的文本
package cn.com.net.in;

import java.net.DatagramPacket;
import java.net.DatagramSocket;

//接收端
public class ReceiveDemo2 {

public static void main(String[] args) throws Exception {
//1.建立udpsocket
DatagramSocket ds=new DatagramSocket(9009);

while(true){
if(ds.equals(886)){
break;
}
//2.建立一个数据包来接收数据
byte[] buf=new byte[1024];

DatagramPacket dp=new DatagramPacket(buf,buf.length);

//3.接收
ds.receive(dp);

//4.通过数据包对象的方法获取数据
String ip=dp.getAddress().getHostAddress();
String data=new String(dp.getData(),0,dp.getLength());
int port=dp.getPort();
System.out.println(ip+”:”+data+”:”+port);
}
//5关闭
//ds.close();
}
}

//发送由键盘输入的的文本
package cn.com.net.in;

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;

public class SendDemo2 {

public static void main(String[] args) throws Exception{
// TODO Auto-generated method stub
DatagramSocket ds=new DatagramSocket();

BufferedReader br=new BufferedReader(new InputStreamReader(System.in));

String line=null;
while((line=br.readLine())!=null){
if(line.equals(“over”))
break;
byte[] buf=line.getBytes();
DatagramPacket dp=new DatagramPacket(buf,buf.length,InetAddress.getByName(“192.168.49.255”),9009);

ds.send(dp);
}
ds.close();
}

}

//接收上传的文package com.hbsi.tcp;

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

public class UploadServer {

public static void main(String[] args) throws Exception{

ServerSocket ss=new ServerSocket(9005);

Socket s=ss.accept();

String ip=s.getInetAddress().getHostAddress();
System.out.println(ip+”….connection”);

BufferedReader brin=new BufferedReader(new InputStreamReader(s.getInputStream()));

PrintWriter pw=new PrintWriter(new FileWriter(“f:\server.txt”),true);

String line=null;

while((line=brin.readLine())!=null){
/*if(line.equals(“over”))
break;*/
pw.println(line);
}

//向客户端发出上传成功提示
PrintWriter pwout=new PrintWriter(s.getOutputStream(),true);
pwout.println(“上传成功”);

pw.close();
s.close();
ss.close();
}
}

//发送上传的文本文件
package com.hbsi.tcp;
import java.net.*;
import java.io.*;

public class UploadClient {

public static void main(String[] args) throws Exception{

Socket s=new Socket(“192.168.49.64”,9005);

//读取要上传的文本文件,为了提高效率,使用缓冲

BufferedReader br=new BufferedReader(new FileReader(“h:\2.txt”));

//将读到的文件内容输出到服务器端
PrintWriter pwout=new PrintWriter(s.getOutputStream(),true);

String line=null;

while((line=br.readLine())!=null){
pwout.println(line);
}
//pwout.println(“over”);

s.shutdownOutput();

//读取服务端发过来的成功提示
BufferedReader brin=new BufferedReader(new InputStreamReader(s.getInputStream()));
String str=brin.readLine();
System.out.println(str);

br.close();
s.close();
}

}

//接收上传的图片
package com.hbsi.tcp;

import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class JpgServer2 {

public static void main(String[] args) throws Exception{

ServerSocket ss=new ServerSocket(10002);

while(true){

Socket s=ss.accept();

new Thread(new JpgThread(s)).start();
}
//ss.close();
}
}

class JpgThread implements Runnable{

private Socket s;

public JpgThread(Socket s){
this.s=s;
}

@Override
public void run() {
int count=1;
try{
String ip=s.getInetAddress().getHostAddress();

System.out.println(ip+”—connection”);
InputStream in=s.getInputStream();

File dir=new File(“F:\picture”);

File f=new File(dir,ip+”(“+count+”).jpg”);

while(f.exists())
f=new File(dir,ip+”(“+(count++)+”).jpg”);

FileOutputStream fos=new FileOutputStream(f);

byte[] buf=new byte[1024];

int len=0;

while((len=in.read(buf))!=-1){
fos.write(buf,0,len);
}

OutputStream out=s.getOutputStream();
out.write(“上传成功了”.getBytes());

fos.close();

s.close();
}catch(Exception e){
e.printStackTrace();
}
}
}

//指定文件 上传图片。发送端。
package com.hbsi.tcp;

import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;

public class JpgClient2 {

public static void main(String[] args)throws Exception {

if(args.length==0){
System.out.println(“请指定一个jpg文件”);
return;
}
File f=new File(args[0]);

Socket s=new Socket(“192.168.49.64”,10002);

FileInputStream fis=new FileInputStream(f);

OutputStream out=s.getOutputStream();

byte[] buf=new byte[1024];

int len=0;

while((len=fis.read(buf))!=-1){
out.write(buf, 0, len);
}

s.shutdownOutput();

InputStream in=s.getInputStream();
byte[] b=new byte[1024];
int num=in.read(b);
System.out.println(new String(b,0,num));

fis.close();

s.close();
}
}做为服务端向浏览发送信息url

package com.hbsi.tcp;

import java.io.InputStream;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;

public class ServerDemo {

public static void main(String[] args) throws Exception {
// TODO Auto-generated method stub
ServerSocket ss=new ServerSocket(10004);

Socket s=ss.accept();

System.out.println(s.getInetAddress().getHostAddress());

//-----------------------------------------------------
//浏览器返回的信息
InputStream in=s.getInputStream();

byte[] buf=new byte[1024];

int len=in.read(buf);
String str=new String(buf,0,len);
System.out.println(str);
//-----------------------------------------------------

PrintWriter pwout =new PrintWriter(s.getOutputStream(),true);
pwout.println("<font color='red' size='7'>访问成功</font>");

s.close();
ss.close();  
}
}

-------------------------------------------------------------------------------------------------------------------------------------
// url
//下载网页
package cn.com.url;

import java.io.BufferedReader;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.URL;

public class TestNet {

public static void main(String[] args)throws Exception {
// TODO Auto-generated method stub

URL url=new URL("http://www.sina.com.cn");

InputStream in=url.openStream();
BufferedReader brin=new BufferedReader(new InputStreamReader(in));

String line=null;
FileOutputStream fos=new FileOutputStream("f:\\chrp.html");

while((line=brin.readLine())!=null){
fos.write(line.getBytes());
System.out.println(line);
}
}
}

你可能感兴趣的:(java,网络)