黑马程序员-----网络编程

来源:互联网 发布:40本网络禁书 下载 编辑:程序博客网 时间:2024/06/07 07:49

------<a href="http://www.itheima.com" target="blank">Java培训、Android培训、iOS培训、.Net培训</a>、期待与您交流! -------


一、网络通讯要素:
IP地址:InetAddress
网络中设备的标识
不易记忆,可用主机名
本地回环地址:127.0.0.1 主机名:localhost
 
端口号
用于标识进程的逻辑地址,不同进程的标识
有效端口:0~65535,其中0~1024系统使用或保留端口

1、InetAddress
public class InetAddress  extends Object  implements Serializable

此类表示互联网协议 (IP) 地址。
IP 地址是 IP 使用的 32 位或 128 位无符号数字,它是一种低级协议,UDP 和 TCP 协议都是在它的基础上构建的。

InetAddress 类提供将主机名解析为其 IP 地址(或反之)的方法。

获取本机计算机名称和IP地址

InetAddress ia = InetAddress.getLocalHost();//getLocalHost是静态方法,直接调用

获取任意主机名称和IP地址

InetAddress i = InetAddress.getByName(String host);
sop(i.getHostAddress());
//sop(i.getHostName());

类似于ping可以直接输入域名,InetAddress也可以这样写:
InetAddress i = InetAddress.getByName("www.baidu.com");//速度相对较慢

反之,如果知道IP地址,也可通过该语句获取主机名称:
InetAddress i = InetAddress.getByName("192.168.0.102");
sop(i.getHostName());

2、Socket

Socket就是为网络服务提供的一种机制。
通信的两端都有Socket。
网络通信其实就是Socket间的通信。
数据在两个Socket间通过IO传输。
3、DatagramSocket 类 

java.net 包中

构造方法: 
   DatagramSocket() //构造数据报套接字并将其绑定到本地主机上任何可用的端口。

常用方法:void send(DatagramPacket p) //从此套接字发送数据报包

4、DatagramPacket 类

java.net 包中

构造方法:
   DatagramPacket(byte[] buf, int length) //构造 DatagramPacket,用来接收长度为 length 的数据包。
   DatagramPacket(byte[] buf, int length, InetAddress address, int port)//构造数据报包,用来将长度为 length 的包发送到指定主机上的指定端口号。 
常用方法:
    int getPort() //返回某台远程主机的端口号,此数据报将要发往该主机或者是从该主机接收到的。  
    InetAddress getAddress() //返回某台机器的 IP 地址,此数据报将要发往该机器或者是从该机器接收到的。     
    int getLength() //返回将要发送或接收到的数据的长度。
二、常见协议及示例代码

常见协议:TCP,UDP
UDP的特点:
1.面向无连接
2.数据会被封包.在64K内
3.不可靠,因为无连接.所以不可靠.
4.速度快,因为不需要连接
例如.聊天的时候就是UDP的,还有视频会议,桌面共享等.
TCP的特点:
1.建立连接,形成传输数据的通道.
2.在连接中进行大数据量传输
3.通过三次握手完成连接,是可靠的协议
4.必须建立连接,但是效率稍低
例如,打电话就是TCP的.
UDP就相当于对讲机,TCP相当于电话,
下载就是TCP的因为不能丢数据.聊天就是UDP的

Ⅰ UDP

    实例1:一个简单的UDP发送端程序

[java] view plaincopy
  1. import java.net.*;  
  2. class UdpDemo   
  3. {  
  4.     public static void main(String[] args) throws Exception  
  5.     {  
  6.         //1.创建UDP服务  
  7.         DatagramSocket ds = new DatagramSocket();  
  8.         //2.确定发送数据  
  9.         byte[] buf = "Hello UDP!".getBytes();  
  10.         //3.封装成包  
  11.         DatagramPacket dp =   
  12.             new DatagramPacket (buf,buf.length,InetAddress.getByName("Datura"),10000);  
  13.         //4.通过send发送  
  14.         ds.send(dp);  
  15.         //关闭资源  
  16.         ds.close();  
  17.     }  
  18. }  

实例2:一个简单的接收端程序

[java] view plaincopy
  1. import java.net.*;  
  2. class UpdReceiveDemo   
  3. {  
  4.     public static void main(String[] args) throws Exception  
  5.     {  
  6.         //创建数据报套接字并将其绑定到本地主机上的指定端口。  
  7.         DatagramSocket ds = new DatagramSocket(10000);  
  8.         byte[] buf = new byte[1024];  
  9.         //构造 DatagramPacket,用来接收长度为 length 的数据包  
  10.         DatagramPacket dp = new DatagramPacket(buf,buf.length);  
  11.         //调用receive方法:receive(DatagramPacket p) 从此套接字接收数据报包  
  12.         ds.receive(dp);  
  13.         String ip = dp.getAddress().getHostAddress();  
  14.         String data = new String(dp.getData(),0,dp.getLength());  
  15.         int port = dp.getPort();  
  16.         System.out.println("["+ip+":"+port+"]"+data);  
  17.     }  
  18. }  

实例3  聊天程序

[java] view plaincopy
  1. /* 
  2. 思路: 
  3. 有收数据的部分,和发数据的部分;这两部分需要同时执行;那就需要用到多线程技术;一个线程控制收,一个线程控制发 
  4.  因为收和发动作是不一致的,所以要定义两个run方法,而且这两个方法要封装到不同的类中 
  5. */  
  6. import java.io.*;  
  7. import java.net.*;  
  8. class ChatDemo   
  9. {  
  10.     public static void main(String[] args) throws Exception  
  11.     {  
  12.         int sendport=10003;  
  13.         int receport=10007;  
  14.   
  15.         new Thread(new Send(new DatagramSocket(sendport))).start();  
  16.         new Thread(new Receive(new DatagramSocket(receport))).start();  
  17.     }  
  18. }  
  19. class Send implements Runnable   
  20. {  
  21.     private DatagramSocket ds;  
  22.     Send(DatagramSocket ds)  
  23.     {  
  24.         this.ds = ds;  
  25.     }  
  26.     public void run()  
  27.     {  
  28.         try  
  29.         {  
  30.             BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));  
  31.             String line = null;  
  32.             while ((line=bufr.readLine())!=null)  
  33.             {  
  34.                 if ("exit".equals(line))  
  35.                     break;  
  36.                 byte[] buf = line.getBytes();  
  37.                 DatagramPacket dp = new DatagramPacket(buf,buf.length,InetAddress.getByName("32"),10003);  
  38.                 ds.send(dp);  
  39.             }             
  40.         }  
  41.         catch (Exception e)  
  42.         {  
  43.             System.out.println("发送失败");  
  44.         }  
  45.     }  
  46. }  
  47. class Receive implements Runnable  
  48. {  
  49.     private DatagramSocket ds;  
  50.     Receive(DatagramSocket ds)  
  51.     {  
  52.         this.ds = ds;  
  53.     }  
  54.     public void run()  
  55.     {  
  56.         try  
  57.         {  
  58.             //DatagramSocket ds = new DatagramSocket(10000);  
  59.             while (true)  
  60.             {  
  61.                 byte[] buf = new byte[1024];  
  62.                 DatagramPacket dp = new DatagramPacket(buf,buf.length);  
  63.                 ds.receive(dp);  
  64.                 String ip = dp.getAddress().getHostAddress();  
  65.                 String data = new String(dp.getData(),0,dp.getLength());  
  66.                 int port = dp.getPort();  
  67.                 System.out.println("["+ip+":"+port+"]"+data);  
  68.             }  
  69.               
  70.         }  
  71.         catch (Exception e)  
  72.         {  
  73.             System.out.println("接受失败");  
  74.         }  
  75.     }  
  76. }  

Ⅱ  TCP

示例一:创建客户端和服务端

[java] view plaincopy
  1. /* 
  2. 客户端对应的对象是Socket 
  3. 服务端对应的对象是ServerSocket 
  4. */  
  5.   
  6. import java.net.*;  
  7. import java.io.*;  
  8. class  ClientDemo  
  9. {  
  10.     public static void main(String[] args) throws Exception  
  11.     {  
  12.         //创建客户端的Socket服务,指定目的主机和端口  
  13.         Socket s = new Socket("192.168.0.102",10006);  
  14.         //为了发送数据,应该获取Socket流中的输出流  
  15.         OutputStream out = s.getOutputStream();  
  16.         out.write("hello TCP".getBytes());  
  17.         s.close();  
  18.     }  
  19. }  
  20. class TCPServer   
  21. {  
  22.     public static void main(String[] args) throws Exception  
  23.     {  
  24.         //建立服务端Socket服务,并监听一个端口  
  25.         ServerSocket ss = new ServerSocket(10006);  
  26.         //通过accept方法获取连接过来的客户端对象  
  27.         Socket s = ss.accept();  
  28.         String ip = s.getInetAddress().getHostAddress();  
  29.         System.out.println(ip+"......is connected.");  
  30.         //获取客户端发送过来的数据,那么要使用客户端对象的读取流来读取数据  
  31.         InputStream in = s.getInputStream();  
  32.         byte[] buf = new byte[1024];  
  33.         int len = in.read(buf);  
  34.         System.out.println(new String(buf,0,len));  
  35.         s.close();  
  36.     }  
  37. }  

示例二:上传txt文件
[java] view plaincopy
  1. import java.io.*;  
  2. import java.net.*;  
  3. class  TextClient  
  4. {  
  5.     public static void main(String[] args) throws Exception  
  6.     {  
  7.         Socket s = new Socket("192.168.1.254",10006);  
  8.         BufferedReader bufr =   
  9.             new BufferedReader(new FileReader("IPDemo.java"));  
  10.         PrintWriter out = new PrintWriter(s.getOutputStream(),true);  
  11.         String line = null;  
  12.         while((line=bufr.readLine())!=null)  
  13.         {  
  14.             out.println(line);  
  15.         }  
  16.         s.shutdownOutput();//关闭客户端的输出流。相当于给流中加入一个结束标记-1.  
  17.         BufferedReader bufIn = new BufferedReader(new InputStreamReader(s.getInputStream()));  
  18.   
  19.         String str = bufIn.readLine();  
  20.         System.out.println(str);  
  21.   
  22.         bufr.close();  
  23.         s.close();  
  24.     }  
  25. }  
  26. class  TextServer  
  27. {  
  28.     public static void main(String[] args) throws Exception  
  29.     {  
  30.         ServerSocket ss = new ServerSocket(10006);  
  31.   
  32.         Socket s = ss.accept();  
  33.         String ip = s.getInetAddress().getHostAddress();  
  34.         System.out.println(ip+"....connected");  
  35.   
  36.         BufferedReader bufIn = new BufferedReader(new InputStreamReader(s.getInputStream()));  
  37.         PrintWriter out  = new PrintWriter(new FileWriter("server.txt"),true);  
  38.   
  39.         String line = null;  
  40.         while((line=bufIn.readLine())!=null)  
  41.         {  
  42.             //if("over".equals(line))  
  43.                 //break;  
  44.             out.println(line);  
  45.         }  
  46.         PrintWriter pw = new PrintWriter(s.getOutputStream(),true);  
  47.         pw.println("上传成功");  
  48.   
  49.         out.close();  
  50.         s.close();  
  51.         ss.close();  
  52.     }  
  53. }  

示例三:客户并发上传图片:☆☆☆☆☆
[java] view plaincopy
  1. /* 
  2. 客户端。 
  3. 1,服务端点。 
  4. 2,读取客户端已有的图片数据。 
  5. 3,通过socket 输出流将数据发给服务端。 
  6. 4,读取服务端反馈信息。 
  7. 5,关闭。 
  8. */  
  9. import java.io.*;  
  10. import java.net.*;  
  11. class  PicClient  
  12. {  
  13.     public static void main(String[] args)throws Exception   
  14.     {  
  15.         if(args.length!=1)  
  16.         {  
  17.             System.out.println("请选择一个jpg格式的图片");  
  18.             return ;  
  19.         }  
  20.         File file = new File(args[0]);  
  21.         if(!(file.exists() && file.isFile()))  
  22.         {  
  23.             System.out.println("该文件有问题,要么不存在,要么不是文件");  
  24.             return ;  
  25.         }  
  26.         if(!file.getName().endsWith(".jpg"))  
  27.         {  
  28.             System.out.println("图片格式错误,请重新选择");  
  29.             return ;  
  30.         }  
  31.         if(file.length()>1024*1024*5)  
  32.         {  
  33.             System.out.println("文件过大,请升级权限");  
  34.             return ;  
  35.         }  
  36.         Socket s = new Socket("192.168.1.254",10007);  
  37.         FileInputStream fis = new FileInputStream(file);  
  38.         OutputStream out = s.getOutputStream();  
  39.         byte[] buf = new byte[1024];  
  40.         int len = 0;  
  41.         while((len=fis.read(buf))!=-1)  
  42.         {  
  43.             out.write(buf,0,len);  
  44.         }  
  45.         //告诉服务端数据已写完  
  46.         s.shutdownOutput();  
  47.         InputStream in = s.getInputStream();  
  48.         byte[] bufIn = new byte[1024];  
  49.         int num = in.read(bufIn);  
  50.         System.out.println(new String(bufIn,0,num));  
  51.         fis.close();  
  52.         s.close();  
  53.     }  
  54. }  
  55. /* 
  56. 服务端 
  57. 这个服务端有个局限性。当A客户端连接上以后。被服务端获取到。服务端执行具体流程。 
  58. 这时B客户端连接,只有等待。 
  59. 因为服务端还没有处理完A客户端的请求,还有循环回来执行下次accept方法。所以 
  60. 暂时获取不到B客户端对象。 
  61.  
  62. 那么为了可以让多个客户端同时并发访问服务端。 
  63. 那么服务端最好就是将每个客户端封装到一个单独的线程中,这样,就可以同时处理多个客户端请求。 
  64.  
  65. 如何定义线程呢? 
  66. 只要明确了每一个客户端要在服务端执行的代码即可。将该代码存入run方法中。 
  67. */  
  68.   
  69.   
  70. class PicThread implements Runnable  
  71. {  
  72.     private Socket s;  
  73.     PicThread(Socket s)  
  74.     {  
  75.         this.s = s;  
  76.     }  
  77.     public void run()  
  78.     {  
  79.         int count = 1;  
  80.         String ip  = s.getInetAddress().getHostAddress();  
  81.         try  
  82.         {  
  83.             System.out.println(ip+"....connected");  
  84.             InputStream in = s.getInputStream();  
  85.             File dir =  new File("d:\\pic");  
  86.             File file = new File(dir,ip+"("+(count)+")"+".jpg");  
  87.             while(file.exists())  
  88.                 file = new File(dir,ip+"("+(count++)+")"+".jpg");  
  89.             FileOutputStream fos = new FileOutputStream(file);  
  90.             byte[] buf = new byte[1024];  
  91.             int len = 0;  
  92.             while((len=in.read(buf))!=-1)  
  93.             {  
  94.                 fos.write(buf,0,len);  
  95.             }  
  96.             OutputStream out = s.getOutputStream();  
  97.             out.write("上传成功".getBytes());  
  98.             fos.close();  
  99.             s.close();  
  100.         }  
  101.         catch (Exception e)  
  102.         {  
  103.             throw new RuntimeException(ip+"上传失败");  
  104.         }  
  105.     }  
  106. }  
  107.   
  108. class  PicServer  
  109. {  
  110.     public static void main(String[] args) throws Exception  
  111.     {  
  112.         ServerSocket ss = new ServerSocket(10007);  
  113.         while(true)  
  114.         {  
  115.             Socket s = ss.accept();  
  116.             new Thread(new PicThread(s)).start();  
  117.         }  
  118.         //ss.close();  
  119.     }  
  120. }  

示例四:客户并发登陆

[java] view plaincopy
  1. /* 
  2. 客户端通过键盘录入用户名。 
  3. 服务端对这个用户名进行校验。 
  4.  
  5. 如果该用户存在,在服务端显示xxx,已登陆。 
  6. 并在客户端显示 xxx,欢迎光临。 
  7.  
  8. 如果该用户存在,在服务端显示xxx,尝试登陆。 
  9. 并在客户端显示 xxx,该用户不存在。 
  10.  
  11. 最多就登录三次。 
  12. */  
  13. import java.io.*;  
  14. import java.net.*;  
  15. class  LoginClient  
  16. {  
  17.     public static void main(String[] args) throws Exception  
  18.     {  
  19.         Socket s = new Socket("192.168.1.254",10008);  
  20.         BufferedReader bufr =   
  21.             new BufferedReader(new InputStreamReader(System.in));  
  22.         PrintWriter out = new PrintWriter(s.getOutputStream(),true);  
  23.         BufferedReader bufIn =  
  24.             new BufferedReader(new InputStreamReader(s.getInputStream()));  
  25.         for(int x=0; x<3; x++)  
  26.         {  
  27.             String line = bufr.readLine();  
  28.             if(line==null)  
  29.                 break;  
  30.             out.println(line);  
  31.   
  32.             String info = bufIn.readLine();  
  33.             System.out.println("info:"+info);  
  34.             if(info.contains("欢迎"))  
  35.                 break;  
  36.         }  
  37.   
  38.         bufr.close();  
  39.         s.close();  
  40.     }  
  41. }  
  42. class UserThread implements Runnable  
  43. {  
  44.     private Socket s;  
  45.     UserThread(Socket s)  
  46.     {  
  47.         this.s = s;  
  48.     }  
  49.     public void run()  
  50.     {  
  51.         String ip = s.getInetAddress().getHostAddress();  
  52.         System.out.println(ip+"....connected");  
  53.         try  
  54.         {  
  55.             for(int x=0; x<3; x++)  
  56.             {  
  57.                 BufferedReader bufIn = new BufferedReader(new InputStreamReader(s.getInputStream()));  
  58.   
  59.                 String name = bufIn.readLine();  
  60.                 if(name==null)  
  61.                     break;  
  62.   
  63.                 BufferedReader bufr = new BufferedReader(new FileReader("user.txt"));  
  64.                 PrintWriter out = new PrintWriter(s.getOutputStream(),true);  
  65.                 String line = null;  
  66.                 boolean flag = false;  
  67.                 while((line=bufr.readLine())!=null)  
  68.                 {  
  69.                     if(line.equals(name))  
  70.                     {  
  71.                         flag = true;  
  72.                         break;  
  73.                     }                 
  74.                 }  
  75.                   
  76.                 if(flag)  
  77.                 {  
  78.                     System.out.println(name+",已登录");  
  79.                     out.println(name+",欢迎光临");  
  80.                     break;  
  81.                 }  
  82.                 else  
  83.                 {  
  84.                     System.out.println(name+",尝试登录");  
  85.                     out.println(name+",用户名不存在");  
  86.                 }  
  87.   
  88.             }  
  89.             s.close();  
  90.         }  
  91.         catch (Exception e)  
  92.         {  
  93.             throw new RuntimeException(ip+"校验失败");  
  94.         }  
  95.     }  
  96. }  
  97. class  LoginServer  
  98. {  
  99.     public static void main(String[] args) throws Exception  
  100.     {  
  101.         ServerSocket ss = new ServerSocket(10008);  
  102.         while(true)  
  103.         {  
  104.             Socket s = ss.accept();  
  105.             new Thread(new UserThread(s)).start();  
  106.         }  
  107.     }  
  108. }  

总结:网络编程具体操作过程特别是涉及到多线程的处理时,相对难理解。TCP部分的示例三、四部分的代码很重要,要理解透彻。
0 0