Java 网络编程
网络基本知识:
在Java中,网络程序有两种协议:TCP和UDP。TCP通过握手协议进行可靠的连接,UDP则是不可靠的连接。
IP地址:用于标记一台计算机的“身份证”。
IP地址由网络地址(确定网络)和主机地址(网络中的主机)组成。
子网掩码:为了区分网络地址和主机地址。
IP地址分为A类、B类、C类(常用)、D类和E类地址。
127.0.0.1 (localhost)是本机地址。
IPv4 和 IPv6
IPv4使用32位二进制表示,可以表示为4个十进制数。
SMTP是简单邮件传输协议,端口号是25.
Telnet用于连接远程计算机或互联网计算机提供的服务。每个服务都会设定一个端口。
给出类似 telnet ip port 即可与特定的服务进行通信
如果要连接互联网的服务,不仅要给出端口,还要给出计算机的名称,只有给出IP地址和端口号时,才能够请求服务,并接收到应答。
URL和URI
URI:统一资源标识符,用于标识一个web资源,包含了两个部分。
(1)URL:统一资源定位符。能够精确的定位数据的URI
(2)URN:统一资源名称。除了URL的URI。
在Java中URI和URL是分开的两个类,URI类专门用于解析,URL用于通信。
URL
1. URI分类
绝对和相对:
(1)绝对URI是指有确定的协议。比如http,ftp。后面以/进行分隔。
(2)相对URI是没有scheme的。
透明和不透明:
(1)不透明URI是不能够被解析的URI。不透明URI是绝对URI。Scheme后面的部分不是以/进行分割。
分层和不分层:
(1)分层是绝对透明URI或相对URI。
默认的网页端口是80.
2. URI的作用:
(1) 解析
[scheme:]scheme-specific-part[#fragment]
Scheme表示用的协议,可以是http\https\ftp\file等。
Scheme-specific-part是其余部分。
进一步细分:
[scheme:][//authority][path][?query][#fragment]
常用方法:
getScheme()获得scheme
getSchemeSpecificPart()
getPath()
getAuthority()
(2) 相对标识符和绝对标识符的转换
函数resolve和relative
示例代码:
任务1:取得特定网址的html代码。
任务2:分析地址信息。
任务3:绝对地址和相对地址转换
//示例代码1
package com.chenjo.net; import java.net.URI; import java.net.URL; import java.util.Scanner; public class URLTest1 { public static void main(String[] args) throws Exception { URL url = new URL("http://www.ecnu.edu.cn"); Scanner in = new Scanner(url.openStream()); while (in.hasNextLine()) { String str = in.nextLine(); System.out.println(str); } URI uri = new URI("http://kanglecrj.iteye.com"); System.out.println(uri.getScheme()); System.out.println(uri.getSchemeSpecificPart()); System.out.println(uri.getAuthority()); System.out.println(uri.getUserInfo()); System.out.println(uri.getHost()); System.out.println(uri.getPort()); System.out.println(uri.getPath()); System.out.println(uri.getQuery()); System.out.println(uri.getFragment()); String str = "/article/details/6705033"; URI combined = uri.resolve(str);// 根据uri的路径把str变成绝对地址 System.out.println(combined.getScheme() + combined.getSchemeSpecificPart()); URI relative = uri.relativize(new URI(str)); System.out.println(relative.getSchemeSpecificPart()); } } /* ...... http //blog.csdn.net/xiazdong blog.csdn.net null blog.csdn.net -1 /xiazdong null null http//blog.csdn.net/article/details/6705033 /article/details/6705033 */
URL和URLConnection
URL的作用
1. 如果想要获取每个网页的html源码,比如http://kanglecjr.iteye.com 则只需要:
(1)URL url = new URL("http://kanglecjr.iteye.com");
(2)Scanner in = new Scanner(url.openStream());
即可。
2. 获取消息头信息
URLConnection conn = url.openConnection
conn.getHeaderFields(); 返回一个Map<String, List<String>>
conn.getContentLength();
conn.getContentType();
conn.getDoOutput(true); 获得输出流
conn.getOutputStream();
conn.getInputStream();
//代码示例二
package com.chenjo.net; import java.net.*; //import sun.misc.*; import java.util.*; //import java.io.*; public class URLConnectionTest { public static void main(String[] args) throws Exception { String urlName = "http://java.sun.com"; URL url = new URL(urlName); URLConnection connection = url.openConnection(); Map<String, List<String>> map = connection.getHeaderFields(); for (Map.Entry<String, List<String>> entry : map.entrySet()) { String key = entry.getKey(); List<String> value = entry.getValue(); System.out.println(key + ":" + value); } } } /* null:[HTTP/1.1 200 OK] X-Frame-Options:[SAMEORIGIN] Content-Language:[en] Access-Control-Allow-Origin:[*] Date:[Mon, 21 Jul 2014 11:42:53 GMT] Content-Length:[47758] X-Akamai-Transformed:[9 - 0 pmb=mRUM,1] Content-Type:[text/html; charset=utf-8] Connection:[keep-alive] X-Powered-By:[Servlet/2.5 JSP/2.1] Server:[Oracle-Application-Server-11g Oracle-Web-Cache-11g/11.1.1.6.0 (TH;max-age=300+0;age=65;ecid=11957576370342462,0:1)] */
在网页中如果要提交数据给web服务器,通常要把数据发送给web服务器,然后web服务器委派一个脚本对数据进行处理,返回一个应答。
通常发送数据的方法有两种:get和post。
(1)get方法是直接把数据跟在url的后面,以name=value进行传输,每个数据之间用&进行分隔,value中的空格用+替换,非字母数字用%替换,并猴哥两个16进制数,这种编码方式称为URL编码。URLEncoder和URLDecoder
(2)post方法是通过URLConnection发送给服务器,编码方式和get一样。URLEncoder.encode(VALUE, "UTF-8); 一般在传输中文时会运用编码和解码。
示例:通过URLEncoder和URLDecoder编码和解码
(略)
InetAddress 根据域名得到IP地址或名称
没有构造方法,通过:
(1)InetAddress i1 = InetAddress.getByName(String) 返回一个InetAddress实例。
(2)如果一个主机有多个ip地址,比如google,有5个ip地址,就调用
InetAddress[] i2 = InetAddress.getAllByName(String);
InetAddress.getLocalhost() 获得本机的InetAddress实例。
//代码实例
package com.chenjo.net; import java.net.InetAddress; public class InetAddressTest { public static void main(String[] args) throws Exception{ InetAddress local = InetAddress.getLocalHost(); System.out.println("本机地址:"+local.getHostAddress()); System.out.println("本机名称:"+local.getHostName()); InetAddress[] remote = InetAddress.getAllByName("www.google.com"); for(InetAddress a : remote) { System.out.println("地址:"+a.getHostAddress()); System.out.println("名称:"+a.getHostName()); } } } /* 本机地址:172.16.4.18 本机名称:netbar-8f-08 地址:173.194.127.112 名称:www.google.com 地址:173.194.127.113 名称:www.google.com 地址:173.194.127.116 名称:www.google.com 地址:173.194.127.114 名称:www.google.com 地址:173.194.127.115 名称:www.google.com */
Socket(TCP)
Socket是一个用于机器之间通信的类。
Socket客户端:
(1)Socket s = new Socket(ip, port); 打开一个 套接字,发送请求
(2)InputStream istream = s.getInputStream(); 接收数据
(3)OutputStream ostream = s.getOutputStream(); 发送数据
需要用PrintWriter和Scanner进行包装,并且注意PrintWriter的自动缓冲。
Socket服务器:注意多个客户端同时访问服务器的问题:多线程
(1)ServerSocket server = new ServerSocket(port); 创建一个端口
(2)Socket s = server.accept(); 只有当有客户端请求并连接,函数才会返回
(3)InputStream istream = s.getInputStream(); 接收数据
(4)OutputStream ostream = s.getOutputStream(); 发送数据
需要用PrintWriter 和 Scanner 进行包装,并且注意PrintWriter的自动缓冲。
我们在使用PrintWriter时需要使用println()函数;
当服务器或客户端任意一方请求结束通信,则立刻停止。
问题1:在套接字中会发生阻塞的地方:
(1)实例化Socket时,会阻塞。
(2)在in.nextLine()类似操作时会阻塞。
解决方法:
(1)对于第一个问题,解决方法:
Socket s = new Socket(); 建立无连接socket
s.connect(new InetSocketAddress(host, pot), timeout); 设置超时。
(2)对于第二个问题,解决方法是设置s.setSoTimeout(long) 设置超时时长
问题2:当客户端想要关闭套接字时,但却不能确定服务器是否还在发送数据,但是只要一关闭就立刻断开。
解决方法:
socket.shutdownOutput() 关闭输出流
socket.shutdownInput() 关闭输入流
为半关闭的示例代码:客户端发送hello给服务器,同时关闭输出流,服务器接收到后关闭输入流,等待5秒发送ECHO hello给客户端。
package com.chenjo.net; import java.net.*; import java.io.*; import java.util.*; public class ShutdownOutputClient { public static void main(String[] args)throws Exception { Socket s = new Socket("localhost",8819); Scanner in = new Scanner(s.getInputStream()); PrintWriter out = new PrintWriter(s.getOutputStream(),true); out.println("Hello");//输出hello s.shutdownOutput(); //关闭输出流 System.out.println("关闭连接"); while(in.hasNextLine()){ System.out.println(in.nextLine()); } s.close(); } }
package com.chenjo.net; import java.net.*; import java.io.*; import java.util.*; public class ShutdownOutputServer { public static void main(String[] args)throws Exception { ServerSocket server = new ServerSocket(8819); Socket s = server.accept(); Scanner in = new Scanner(s.getInputStream()); PrintWriter out = new PrintWriter(s.getOutputStream(),true); String str = in.nextLine(); System.out.println(str); s.shutdownInput(); System.out.println("关闭输入流"); Thread.sleep(5000); out.println("Echo:"+str); s.close(); } }
综合代码举例:实现一个简单的对等通信程序,通过多线程,一个线程接收数据,一个线程发送数据。
//Client.java
package com.chenjo.net; import java.util.*; import java.io.*; import java.net.*; public class Client{ public static void main(String[]args)throws Exception{ Socket s = new Socket("localhost",8819); PrintWriter out = new PrintWriter(s.getOutputStream(),true); Thread t = new Thread(new Receive(s)); t.start(); //以下代码用于发送数据 Scanner in = new Scanner(System.in);//键盘输入 while(in.hasNextLine()){ //一直不断 out.println(in.nextLine()); //发送键盘输入数据 } } } class Receive implements Runnable //这个类用于接收数据 { private Socket s; public Receive(Socket s) { this.s = s; } public void run() { try{ Scanner in = new Scanner(s.getInputStream()); //in:接收数据 String str = null; while(true) { str = in.nextLine(); System.out.println("服务器说:"+str); //打印接收数据 } } catch(Exception e){} } } /* C: Hello 服务器说:S: Hi! C: Are you the boss? 服务器说:S: Yes, I am the boss. C: OK. See you tomorrow, my boss! 服务器说:S: See you! */
//Server.java
package com.chenjo.net; import java.util.*; import java.io.*; import java.net.*; public class Server{ public static void main(String[]args)throws Exception{ ServerSocket server = new ServerSocket(8819); Socket s = server.accept(); PrintWriter out = new PrintWriter(s.getOutputStream(),true); Thread t = new Thread(new Receive1(s)); t.start(); //以下代码用于发送数据 Scanner in = new Scanner(System.in);//键盘输入 while(in.hasNextLine()){ //一直不断 out.println(in.nextLine()); //发送键盘输入数据 } } } class Receive1 implements Runnable //这个类用于接收数据 { private Socket s; public Receive1(Socket s) { this.s = s; } public void run() { try{ Scanner in = new Scanner(s.getInputStream()); //in:接收数据 String str = null; while(true) { str = in.nextLine(); System.out.println("客户端说:"+str); //打印接收数据 } } catch(Exception e){} } } /* 客户端说:C: Hello S: Hi! 客户端说:C: Are you the boss? S: Yes, I am the boss. 客户端说:C: OK. See you tomorrow, my boss! S: See you! */
以上程序属于C/S,需要同时维护客户端和服务器的代码。
B/S: 浏览器和服务器,只需要维护一方代码即可。
聊天工具使用UDP非常多,因为我们通常也会遇到我们发给对方一条消息,对方却没有收到的情况。
DatagramPacket 和 DatagramSocket 数据报
代码举例:实现服务器发送数据报到客户端
package com.chenjo.net; import java.net.*; import java.io.*; public class DatagramClient { public static void main(String[] args) throws Exception{ byte[]buf = new byte[1024]; DatagramPacket packet = new DatagramPacket(buf,1024); DatagramSocket client = new DatagramSocket(9000); client.receive(packet); String str = new String(buf,0,packet.getLength()); System.out.println(packet.getAddress().getHostName()+":"+str); client.close(); } } /** Result: 172.16.4.18:hello world */
package com.chenjo.net; import java.net.DatagramPacket; import java.net.DatagramSocket; import java.net.InetAddress; public class DatagramServer { public static void main(String[] args)throws Exception { DatagramSocket server = new DatagramSocket(3000); String str = "hello world"; DatagramPacket packet = new DatagramPacket(str.getBytes(),str.length(),InetAddress.getLocalHost(),9000); server.send(packet); server.close(); } }
PrintWriter 的autoflush
如果 PrintWriter writer = new PrintWriter(out, true);
则调用println()、printf()、format()函数时会自动刷新,其他函数都不会,比如write()、print()函数时不会自动刷新。
网络编程常见异常
第1个异常是java.net.BindException: Address already in use: JVM_Bind。该异常发生在服务器端进行new ServerSocket(port) (port 是 一个 0-65536的整型值)操作时。异常的原因是因为该port端口已经被占用被监听。此时用netstat -an 命令,可以看到一个Listening 状态的端口。只需要找一个没有被占用的端口就能解决这个问题。
第2个异常是java.net.ConnectionException: Connection refused: connect。该异常发生在客户端进行new Socket(ip, port) 操作时。该异常发生的原因是或者具有ip地址的机器无法找到(也就是说当前机器不存在到指定ip的路由),或者该ip存在,但找不到指定的被监听端口。出行该问题,首先检查客户端的ip和port是否写错了,如果正确则从客户端ping一下服务器看能否ping通(服务器端把ping禁掉则需要另外的办法),则看在服务器端的指定监听端口的程序是有已启动,肯定能解决这个问题。
第3个异常时java.net.SocketException: Socket is closed,该异常在客户端和服务器均可能发生。异常的原因是己方主动关闭了连接后(调用了Socket的close方法)再对网络连接进行读写操作。
第4个异常是java.net.SocketException: (Connection reset 或者 Connect reset by peer: Socket write error)。该异常在客户端和服务器端均可能发生,引起该异常的原因有两个,第一个就是如果一端的Socket被关闭(或主动关闭或因为异常退出而引起的关闭),另一端仍发送数据,发送的第一个数据包引发该异常(Connection reset by peer)。另一个是一端退出,但退出时并未关闭该连接,另一端如果再从连接中读数据则被抛出该异常(Connection reset)。简单地说就是在连接断开后的读和写操作引起的。
第5个异常是java.net.SocketException: Broken pipe。该异常在客户端和服务器端均有可能发生。在第4个异常的第一种情况中(也就是抛出SocketException: Connect reset by peer: Socket write error后),如果再继续写数据则抛出该异常。前两个异常的解决方法是首先确保程序退出前关闭所有的网络连接,其次是要检测对方的关闭连接操作,发现对方关闭连接后自己也要关闭该连接。
类QQ客户端:
package com.chenjo.net; import java.awt.BorderLayout; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.io.BufferedReader; import java.io.InputStreamReader; import java.io.OutputStream; import java.io.PrintStream; import java.io.PrintWriter; import java.net.Socket; import javax.swing.JButton; import javax.swing.JFrame; import javax.swing.JPanel; import javax.swing.JScrollPane; import javax.swing.JTextArea; import javax.swing.JTextField; public class Client3 extends JFrame{ static JTextArea area; JTextField field; JButton button; static PrintWriter writer; public Client3(){ this.setTitle("客户端"); this.setSize(400,500); area = new JTextArea(25,30); area.setEditable(false); field = new JTextField(20); button = new JButton("提交"); JScrollPane sp = new JScrollPane(area); JPanel panel = new JPanel(); this.add(sp,BorderLayout.CENTER); panel.add(field); panel.add(button); this.add(panel,BorderLayout.SOUTH); this.setVisible(true); this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); button.addActionListener(new ActionListener(){ @Override public void actionPerformed(ActionEvent e) { String text = field.getText(); writer.println(text); area.append("我:"+text+"\n"); field.setText(""); } }); } public static void main(String[] args) throws Exception{ Client3 c = new Client3(); Socket socket = new Socket("127.0.0.1",8899); OutputStream out = socket.getOutputStream(); BufferedReader reader1 = new BufferedReader(new InputStreamReader(socket.getInputStream())); writer = new PrintWriter(out,true); System.out.println("已经成功和服务器连接..."); while(true){ String line = reader1.readLine(); area.append("服务器:"+line+"\n"); } } }
类QQ服务器:
package com.chenjo.net; import java.awt.BorderLayout; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.io.BufferedReader; import java.io.InputStreamReader; import java.io.PrintStream; import java.net.InetAddress; import java.net.ServerSocket; import java.net.Socket; import javax.swing.JButton; import javax.swing.JFrame; import javax.swing.JPanel; import javax.swing.JScrollPane; import javax.swing.JTextArea; import javax.swing.JTextField; public class Server3 extends JFrame{ static JTextArea area; JTextField field; JButton button; static PrintStream writer; public Server3(){ this.setTitle("服务器"); this.setSize(400,500); area = new JTextArea(25,30); area.setEditable(false); field = new JTextField(20); button = new JButton("提交"); JPanel panel = new JPanel(); JScrollPane sp = new JScrollPane(area); this.add(sp,BorderLayout.CENTER); panel.add(field); panel.add(button); this.add(panel,BorderLayout.SOUTH); this.setVisible(true); this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); button.addActionListener(new ActionListener(){ @Override public void actionPerformed(ActionEvent e) { String text = field.getText(); writer.println(text); area.append("我:"+text+"\n"); field.setText(""); } }); } public static void main(String[] args) throws Exception { Server3 s = new Server3(); ServerSocket server = new ServerSocket(8899); System.out.println("开始监听..."); Socket socket = server.accept(); InetAddress address = socket.getInetAddress(); String name = address.getLocalHost().getHostName(); System.out.println(name+"已连接"); BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream())); writer = new PrintStream(socket.getOutputStream(), true); while (true) { String line = null; line = reader.readLine(); if (line != null) { area.append("客户端:"+line+"\n"); } } } }
原文出处:
http://blog.csdn.net/xiazdong/article/details/6713691
相关推荐
Java网络编程案例教程习题参考答案 Java_network_programming是Java编程语言中一个基础组件,用于实现网络通信。以下是Java网络编程案例教程习题参考答案中涉及到的知识点: 1. Socket编程:Socket是Java网络编程...
Java网络编程是Java开发中的重要领域,它涵盖了网络应用程序的设计、实现和调试。在这个主题下,我们可以探讨多个关键知识点: 1. **Java Socket编程**:Java的Socket类提供了基于TCP/IP协议的网络通信能力。通过...
《Java网络编程(第四版)》是一本深入探讨Java在互联网环境下的编程技术的经典书籍。本书旨在帮助读者理解和掌握如何利用Java语言进行高效、安全的网络通信。书中内容覆盖了从基本的网络概念到复杂的多线程编程,是...
Java网络编程是计算机科学中的一个重要领域,特别是在软件开发中,它涉及到如何通过网络进行数据传输和通信。在Java中,网络编程主要依赖于Java的Socket编程、ServerSocket、URL类以及NIO(非阻塞I/O)等核心API。这...
"Java网络编程实验报告" 本实验报告主要介绍了Java网络编程的基本概念和实现方法,通过设计和实现一个简单的客户端/服务器应用程序,了解Java网络编程的基本原理和实现方法。 知识点1:Java 网络编程基础 Java ...
在本资料中,《Java网络编程》第三版提供了深入浅出的讲解,旨在帮助开发者提升对这一领域的理解。 1. **基础概念**: - **网络模型**:Java网络编程基于OSI七层模型和TCP/IP四层模型。理解这些模型有助于理解网络...
《Java网络编程(第4版)》是一本深入探讨Java平台上的网络编程技术的专业书籍,适合想要提升Java通讯技术的学者阅读。此书全面覆盖了Java网络编程的基础和高级概念,帮助开发者理解如何利用Java语言构建高效、可靠的...
Java网络编程
Java 网络编程实践课程设计报告 这是一份 Java 网络编程实践课程设计报告,旨在帮助学生掌握 Java 编程语言、图形化界面、多线程、网络和数据库等技术,并提高动手实践能力和书本知识学习。该课程设计报告涵盖了 ...
Java网络编程是开发分布式应用程序的关键技术,它允许程序通过网络发送和接收数据。《Java网络编程实例》这本书的源代码提供了丰富的示例,帮助读者深入理解这一领域。本压缩包包含的源代码覆盖了Java网络编程的各种...
《Java网络编程精解》是孙卫琴老师的一本经典教程,主要涵盖了Java语言在网络编程领域的深度解析。这本书深入浅出地介绍了如何使用Java进行网络通信,包括基本的TCP/IP协议、套接字编程、多线程技术以及HTTP、FTP等...
Java网络编程是开发分布式应用程序的关键技术,它使得Java程序能够与其他设备、系统和服务进行通信。本书《Java网络编程实例:Java网络编程实例》显然聚焦于通过实际案例来教授这一核心技能。以下是一些主要的知识点...
java网络编程.pdf
Java网络编程是开发分布式应用程序的关键技术,它允许Java程序与其他计算机进行通信,实现数据的交换。这份"Java网络编程资料"包含三个重要的学习资源:关于Socket套接字的"Java套接字编程.chm"、关于网络协议特别是...
本资源为"Java网络编程.chm",是一本关于Java网络编程的电子书,以实例驱动的方式深入讲解了相关知识。 首先,我们要理解Java在网络编程中的基础——IO流。Java中的IO流分为字节流和字符流,它们用于读写数据,包括...
Java网络编程是Java开发中的重要领域,它涵盖了网络通信的所有基本概念和技术,包括TCP/IP协议栈、套接字(Socket)编程、多线程、数据传输格式等。孙卫琴的《java网络编程》一书深入浅出地讲解了这些核心概念,并...
《JAVA网络编程大全》是一本全面且深入介绍Java网络编程技术的权威著作,旨在帮助读者掌握Java在构建网络应用程序中的核心概念和实践技巧。PDF版的书籍为学习者提供了便捷的电子阅读体验,随时随地都能深化对Java...
Java网络编程是Java开发中的重要领域,主要用于实现应用程序之间的通信,包括客户端-服务器架构、分布式系统以及互联网数据传输。这份“java网络编程 PPT”来自清华大学,无疑为学习这一主题提供了权威的参考资料。...
《Java网络编程》是编程领域的经典著作,尤其对于学习Java网络编程的初学者及进阶者来说,这本书具有很高的参考价值。它详细介绍了如何利用Java语言进行网络通信,涵盖了网络编程的基本概念、协议以及实现方法。 一...