Java_常瑞鹏 java 深入理解 网络编程


Java是一种平台无关的编程语言,具有“一次编写、到处运行”的特点,所以非常适合网络编程。
可以说,在网络编程方面没有任何一门语言比Java更优秀。

¯网络基本概念
µ

计算机网络,就是把分布在不同地理区域的计算机与专门的外部设备用通信线路互连成一个规模大、

功能强的网络系统,从而使众多的计算机可以方便地互相传递信息,共享硬件、软件、数据信息等资源。

µ

网络体系结构:国际标准化组织ISO于l978年提出“开放系统互连参考模型”,即著名的OSI(Open System Interconnection)模型。

该模型把计算机网络分成物理层、数据链路层、网络层、传输层、会话层、表示层、应用层等七层。

¯
通信协议

计算机网络中实现通信必须有一些约定,即通信协议。对速率、传输代码、代码结构、传输控制步骤、出错控制等制定标准。

TCP协议:提供可靠的数据传输服务的规则。
IP协议进行IP数据包的分割和组装。

但是通过IP协议并不能清楚地了解到数据包是否顺利地发送给目标计算机。而使用TCP协议,它将数据包成功发送给目标计算机
后,会要求发送一个确认,如果在某个时间内没有收到确认,TCP将重新发送数据包。

¯IP地址和端口号

IP地址:为实现网络中不同的计算机之间的通信,在网络中的每台机器都必须有一个与众不同的标识,这就是IP地址(IP Address)

格式:数字型、32位、由4段8位的二进制数组成。一般表示为十进制形式(4个0~255的十进制整数),中间用圆点隔开,如:166.111.78.98。

域名地址:也是分段表示的,便于记忆的、字符串形式。

端口:一个16位的整数,用于表示数据交给哪个通信程序处理。因此,端口就是应用程序与外界交流的出入口,它是一种抽象的软件结构,

包括一些数据结构和I/O(基本输入/输出)缓冲区。


不同的应用程序处理不同端口上的数据,同一台机器上不能有两个程序使用同一个端口,端口号可以从0到65535,
通常将它分为三类:

公认端口(Well Known Ports):从0到1023,它们紧密绑定(Binding)一些服务。

注册端口(Registered Ports):从1024到49151。它们松散地绑定一些服务。

动态和/或私有端口(Dynamic and/or Private Ports):从49152到65535,这些端口是应用程序使用的动态端口,应用程序一般不会主动使用这些端口。


JDK中为开发网络应用程序提供了java.net包,该包下的类和接口几乎都是为网络编程服务的。
¯
InetAddress:用于描述IP地址的对象
InetAddress类没有提供构造方法,而是提供了两个静态方法来获取InetAddress实例
getByName(String host):根据主机获取对应的InetAddress对象。
getByAddress(byte[] addr):根据原始IP地址来获取对应的InetAddress对象。

¯什么是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();
}
}


¯ServerSocket:编写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的输出流对象。











小实例:

//接收由发送端键盘输入的文本
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);
}
}
}


posted @ 2012-09-14 15:30  chrp99  阅读(178)  评论(0编辑  收藏  举报