Java 网络编程(二)
以下例开始本文的内容:
例1,需求:上传图片。
客户端:
- 服务端点。
- 读取客户端已有的图片数据。
- 通过socket输出流将数据发给服务端。
- 读取服务端反馈信息。
- 关闭。
class PicClient {
public static void main(String[] args) throws IOException {
/*
* 一系列判断
*/
if(args.length != 1) {
System.out.println("请选择一个jpg格式的图片");
return;
}
File file = new File(args[0]);
if(!(file.exists() && file.isFile())) {
System.out.println("该文件有问题,要么不存在,要么不是文件");
return;
}
if(!file.getName().endsWith(".jpg")) {
System.out.println("图片格式错误,请重新选择");
return;
}
if(file.length() > 1024*1024*10) {
System.out.println("文件过大,没安好心");
return;
} Socket s = new Socket("10.48.62.209", 10007); FileInputStream fis = new FileInputStream(file); 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[] bufIn = new byte[1024];
int num = in.read(bufIn);
System.out.println(new String(bufIn, 0, num));
fis.close();
s.close();
}
}
服务端:
这个服务端有个局限性,当A客户端连接上以后,被服务端获取到,服务端执行具体流程。这时B客户端连接,只有等待,因为服务端还没有处理完A客户端的请求,还没有循环回来执行下一次accpet()方法,所以暂时获取不到B客户端对象。
那么为了可以让多个客户端同时并发访问服务端,服务端最好就是将每个客户端封装到一个单独的线程中,这样就可以同时处理多个客户端请求。
那么如何定义线程呢?
只要明确了每一个客户端在服务端执行的代码即可。将该代码存入run()方法中。
class PicThread implements Runnable {
private Socket s;
PicThread(Socket s) {
this.s = s;
}
@Override
public void run() {
int count = 1;
String ip = s.getInetAddress().getHostAddress();
try { System.out.println(ip+"......connected."); InputStream in = s.getInputStream(); File file = new File(ip+"("+(count)+")"+".jpg");//10.48.62.209(1).jpg
while(file.exists()) {
file = new File(ip+"("+(count++)+")"+".jpg");
} FileOutputStream fos = new FileOutputStream(file); 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 (IOException e) {
throw new RuntimeException(ip+"上传失败");
}
} }
class PicServer {
public static void main(String[] args) throws IOException {
ServerSocket ss = new ServerSocket(10007);
while(true) {
Socket s = ss.accept(); new Thread(new PicThread(s)).start();
}
//ss.close();
} }
我觉得多线程是比较难理解的,所以图示以上原理:
例2,客户端通过键盘录入用户名。服务端对这个用户名进行校验。如果该用户存在,在服务端显示XXX,已登录。并在客户端显示XXX,欢迎光临。如果该用户不存在,在服务端显示XXX,尝试登录。并在客户端显示XXX,该用户不存在。最多就登录3次。
以下为代码,导包就不导了。
客户端:
class LoginClient {
public static void main(String[] args) throws IOException {
Socket s = new Socket("10.48.62.209", 10008); BufferedReader bufr =
new BufferedReader(new InputStreamReader(System.in)); PrintWriter out = new PrintWriter(s.getOutputStream(), true); BufferedReader bufIn =
new BufferedReader(new InputStreamReader(s.getInputStream())); for(int x = 0; x < 3; x++) {
/*
* 在键盘敲Ctrl+C键,即代表结束录入
* read()方法返回-1,readLine()方法返回null
*/
String line = bufr.readLine();//Ctrl+C,那么就是-1,readLine()返回null
if(line == null)
break;
out.println(line);
String info = bufIn.readLine();
System.out.println("info:"+info);
if(info.contains("欢迎"))
break;
}
bufr.close();
s.close();//socket流中加了一个-1,相当于给流中加入一个结束标记,即-1
}
}
服务端:
class UserThread implements Runnable {
private Socket s;
UserThread(Socket s) {
this.s = s;
}
@Override
public void run() {
String ip = s.getInetAddress().getHostAddress();
System.out.println(ip+"........connected.");
try {
for(int x = 0; x < 3; x++) {
BufferedReader bufIn = new BufferedReader(
new InputStreamReader(s.getInputStream()));
/*
* 客户端那边敲入Ctrl+C键键时,
* read()返回-1,readLine()返回null。
* 所以name得加一个判断
*/
String name = bufIn.readLine();
if(name == null)
break; /*
* 校验
*/
BufferedReader bufr = new BufferedReader(new FileReader("user.txt")); PrintWriter out = new PrintWriter(s.getOutputStream(), true); String line = null; boolean flag = false;//定义标记
while((line = bufr.readLine()) != null) {
if(line.equals(name)) {
flag = true;
break;
}
}
if(flag) {
System.out.println(name+",已登录");
out.println(name+",欢迎光临");
break;
} else {
System.out.println(name+"尝试登录");
out.println(name+",用户名不存在");
}
}
s.close();
} catch (IOException e) {
throw new RuntimeException(ip+"校验失败");
}
} }
class LoginServer {
public static void main(String[] args) throws IOException {
ServerSocket ss = new ServerSocket(10008);
while(true) {
Socket s = ss.accept();
new Thread(new UserThread(s)).start();
}
} }
演示客户端和服务端
1、
客户端:浏览器、telnet(远程登录命令——在DOS命令行下连接网络上任意一台主机)
形如:
telnet 10.48.62.209 11000
示例代码:
public class ServerDemo { public static void main(String[] args) throws IOException {
ServerSocket ss = new ServerSocket(11000); Socket s = ss.accept();
System.out.println(s.getInetAddress().getHostAddress()); InputStream in = s.getInputStream();
byte[] buf = new byte[1024];
int len = in.read(buf);
System.out.println(new String(buf, 0, len)); PrintWriter out = new PrintWriter(s.getOutputStream(), true);
out.println("<font color='red' size='7'>Client, hello!!!</font>");
s.close();
ss.close();
} }
控制台输出类似诸如以下信息(头信息):
GET / HTTP/1.1
Host: 127.0.0.1:11000
Connection: keep-alive
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/48.0.2564.116 Safari/537.36
Accept-Encoding: gzip, deflate, sdch
Accept-Language: zh-CN,zh;q=0.8
分析如下:
http(http协议)://127.0.0.1(主机名):11000(端口号)/myWeb(资源路径)/demo.html(资源)
Host: 127.0.0.1:11000
Connection: keep-alive
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*(代表可接受任意类型的东东);q=0.8
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/48.0.2564.116 Safari/537.36
Accept-Encoding: gzip(告诉服务器的压缩方式), deflate, sdch
Accept-Language: zh-CN,zh;q=0.8
2、
客户端:浏览器。
服务器:Tomcat服务器。
3、
客户端:自定义。
服务器:Tomcat服务器。
示例代码:
public class MyIE { public static void main(String[] args) throws IOException {
Socket s = new Socket("127.0.0.1", 8888); PrintWriter out = new PrintWriter(s.getOutputStream(), true); out.println("GET /myWeb/demo.html HTTP/1.1");
out.println("Accept: */*");
out.println("Accept-Language: zh-CN,zh;q=0.8");
out.println("Host: 127.0.0.1:11000");
out.println("Connection: closed");
/*
* 注意:HTML头信息和主体信息之间一定要有一个空行
* 为了保证安全,所以索性加了两行
*/
out.println();
out.println(); BufferedReader bufr = new BufferedReader(
new InputStreamReader(s.getInputStream()));
String line = null;
while((line = bufr.readLine()) != null) {
System.out.println(line);
}
s.close();
} }
控制台会打印类似信息:
自定义图形界面浏览器:
代码如下(导包就不导了):
public class MyIEByGUI { private Frame f;
private TextField tf;
private Button but;
private TextArea ta; private Dialog d;
private Label lab;
private Button okBut; MyIEByGUI() {
init();
} public void init() {
f = new Frame("my window");
f.setBounds(300, 100, 600, 500);
f.setLayout(new FlowLayout()); tf = new TextField(60); but = new Button("转到"); ta = new TextArea(25, 70); /*
* 对话框,也是一个窗体,最好不要加到Frame里面去
*/
d = new Dialog(f, "提示信息-self", true);//true:对话框不处理掉,后面的窗体是无法操作的!!
d.setBounds(400, 200, 240, 150);
d.setLayout(new FlowLayout());
lab = new Label();
okBut = new Button("确定"); d.add(lab);
d.add(okBut); f.add(tf);
f.add(but);
f.add(ta); myEvent();
f.setVisible(true); } private void myEvent() {
//点击对话框中的确定按钮,对话框也不显示出来
okBut.addActionListener(new ActionListener() { @Override
public void actionPerformed(ActionEvent e) {
d.setVisible(false);
}
});
/*
* 关闭对话框,对话框不显示
*/
d.addWindowListener(new WindowAdapter() {
@Override
public void windowClosing(WindowEvent e) {
d.setVisible(false);
}
}); tf.addKeyListener(new KeyAdapter() { @Override
public void keyPressed(KeyEvent e) {
if(e.getKeyCode() == KeyEvent.VK_ENTER) {
try {
showDir();
} catch (IOException e1) {
e1.printStackTrace();
}
}
} }); /*
* 按钮是事件源
*/
but.addActionListener(new ActionListener() { @Override
public void actionPerformed(ActionEvent e) {
try {
showDir();
} catch (IOException e1) {
e1.printStackTrace();
}
}
}); f.addWindowListener(new WindowAdapter() { @Override
public void windowClosing(WindowEvent e) {
System.exit(0);
} });
} private void showDir() throws IOException {
ta.setText("");
/*
* http://127.0.0.1:8888/myWeb/demo.html
* 以下代码只不过是拆解url字符串路径而已
*/
String url = tf.getText(); int index1 = url.indexOf("//")+2; int index2 = url.indexOf("/", index1); String str = url.substring(index1, index2);
String[] arr = str.split(":");
String host = arr[0];
int port = Integer.parseInt(arr[1]); String path = url.substring(index2);
//ta.setText(str+"....."+path); Socket s = new Socket(host, port); PrintWriter out = new PrintWriter(s.getOutputStream(), true); out.println("GET "+path+" HTTP/1.1");
out.println("Accept: */*");
out.println("Accept-Language: zh-CN,zh;q=0.8");
out.println("Host: 127.0.0.1:11000");
out.println("Connection: closed");
out.println();
out.println(); BufferedReader bufr = new BufferedReader(
new InputStreamReader(s.getInputStream()));
String line = null;
while((line = bufr.readLine()) != null) {
ta.append(line+"\r\n");
}
s.close(); } public static void main(String[] args) {
new MyIEByGUI();
} }
运行效果图:
URL
示例代码:
/*
String getFile()
获取此 URL 的文件名。
String getHost()
获取此 URL 的主机名(如果适用)。
String getPath()
获取此 URL 的路径部分。
int getPort()
获取此 URL 的端口号。
String getProtocol()
获取此 URL 的协议名称。
String getQuery()
获取此 URL 的查询部分。
*/
import java.net.MalformedURLException;
import java.net.URL; public class URLDemo { public static void main(String[] args) throws MalformedURLException {
URL url = new URL("http://127.0.0.1/myWeb/demo.html?name=haha&age=30"); System.out.println("getProtocol():"+url.getProtocol());//http
System.out.println("getHost():"+url.getHost());//127.0.0.1
System.out.println("getPort():"+url.getPort());
System.out.println("getPath():"+url.getPath());//---/myWeb/demo.html
System.out.println("getFile():"+url.getFile());//---/myWeb/demo.html?name=haha&age=30
System.out.println("getQuery():"+url.getQuery());//name=haha&age=30 /*
不写端口时,给一个默认的端口号80
int port = getPort();
if(port == -1)
port = 80;
*/
} }
URLConnection(似乎封装了Socket)
示例代码:
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.net.URLConnection; public class URLConnectionDemo { public static void main(String[] args) throws IOException {
URL url = new URL("http://127.0.0.1:8888/myWeb/demo.html");
URLConnection conn = url.openConnection();
System.out.println(conn); InputStream in = conn.getInputStream(); byte[] buf = new byte[1024]; int len = in.read(buf); System.out.println(new String(buf, 0, len)); } }
所以自定义图形界面浏览器代码优化之后为:
public class MyIEByGUI2 { private Frame f;
private TextField tf;
private Button but;
private TextArea ta; private Dialog d;
private Label lab;
private Button okBut; MyIEByGUI2() {
init();
} public void init() {
f = new Frame("my window");
f.setBounds(300, 100, 600, 500);
f.setLayout(new FlowLayout()); tf = new TextField(60); but = new Button("转到"); ta = new TextArea(25, 70); /*
* 对话框,也是一个窗体,最好不要加到Frame里面去
*/
d = new Dialog(f, "提示信息-self", true);//true:对话框不处理掉,后面的窗体是无法操作的!!
d.setBounds(400, 200, 240, 150);
d.setLayout(new FlowLayout());
lab = new Label();
okBut = new Button("确定"); d.add(lab);
d.add(okBut); f.add(tf);
f.add(but);
f.add(ta); myEvent();
f.setVisible(true); } private void myEvent() {
//点击对话框中的确定按钮,对话框也不显示出来
okBut.addActionListener(new ActionListener() { @Override
public void actionPerformed(ActionEvent e) {
d.setVisible(false);
}
});
/*
* 关闭对话框,对话框不显示
*/
d.addWindowListener(new WindowAdapter() {
@Override
public void windowClosing(WindowEvent e) {
d.setVisible(false);
}
}); tf.addKeyListener(new KeyAdapter() { @Override
public void keyPressed(KeyEvent e) {
if(e.getKeyCode() == KeyEvent.VK_ENTER) {
try {
showDir();
} catch (IOException e1) {
e1.printStackTrace();
}
}
} }); /*
* 按钮是事件源
*/
but.addActionListener(new ActionListener() { @Override
public void actionPerformed(ActionEvent e) {
try {
showDir();
} catch (IOException e1) {
e1.printStackTrace();
}
}
}); f.addWindowListener(new WindowAdapter() { @Override
public void windowClosing(WindowEvent e) {
System.exit(0);
} });
} private void showDir() throws IOException {
ta.setText("");
/*
* http://127.0.0.1:8888/myWeb/demo.html
*/
String urlPath = tf.getText(); URL url = new URL(urlPath);
URLConnection conn = url.openConnection();
//System.out.println(conn); InputStream in = conn.getInputStream(); byte[] buf = new byte[1024]; int len = in.read(buf); ta.setText(new String(buf, 0, len)); } public static void main(String[] args) {
new MyIEByGUI2();
} }
图示原理:
域名解析
Java 网络编程(二)的更多相关文章
- Java网络编程二:Socket详解
Socket又称套接字,是连接运行在网络上两个程序间的双向通讯的端点. 一.使用Socket进行网络通信的过程 服务端:服务器程序将一个套接字绑定到一个特定的端口,并通过此套接字等待和监听客户端的连接 ...
- Java网络编程二
向web站点发送get post请求,并从web站点取得响应 通相应的set方法,在调用connect()方法前,可以设置请求的各种参数,实现同浏览器类似的访问,调用connect()后可以调用相应的 ...
- Java网络编程二--基于UDP的编程
DatagramSocket对象为基于UDP协议的Socket 构造器提供可以选择性绑定到指定端口和ip 创建完对象后调用:receive(DatagramPacket p) send(Dategra ...
- 如何使用socket进行java网络编程(二)
通过在如何使用socket进行java网络编程(一)中程序的编写,可以总结出一些常用的java socket编程的范例来. ServerSocket server = new ServerSocket ...
- Java 网络编程学习总结
新手一枚,Java学习中,把自己学习网络编程的知识总结一下,梳理下知识,方便日后查阅,高手莫进. 本文的主要内容: [1] 网络编程认识 [2] TCP/IP编程 ...
- 20145212 实验五《Java网络编程》
20145212 实验五<Java网络编程> 一.实验内容 1.运行下载的TCP代码,结对进行,一人服务器,一人客户端: 2.利用加解密代码包,编译运行代码,一人加密,一人解密: 3.集成 ...
- java 网络编程复习(转)
好久没有看过Java网络编程了,现在刚好公司有机会接触,顺便的拾起以前的东西 参照原博客:http://www.cnblogs.com/linzheng/archive/2011/01/23/1942 ...
- JAVA课程实验报告 实验五 Java网络编程及安全
北京电子科技学院(BESTI) 实 验 报 告 课程:Java程序设计 班级:1353 姓名:韩玉琪 学号:20135317 成绩: 指导教师:娄嘉 ...
- 20145225《Java程序设计》 实验五 Java网络编程及安全
20145225<Java程序设计> 实验五 Java网络编程及安全 实验报告 一.实验内容 基于Java Socket实现安全传输. 基于TCP实现客户端和服务器,结对编程一人负责客户端 ...
随机推荐
- linux下安装protobuf教程+示例(详细)
(.pb.h:9:42: fatal error: google/protobuf/stubs/common.h: No such file or directory 看这个就应该知道是没有找到头文件 ...
- vnextcn
Flag 标题 通过 提交 AC% 难度 E1 编写Hello World网站 9 17 52% 1 E2 编写Hello World控制台程序 11 13 84% 1 E3 控制器基础练 ...
- layer 弹出子页面然后给父页面赋值
//----赋值 并关闭当前页面 开始---- FunctionActionDeleteXZ = function (CompanyId, RelCompanyName) { parent.$(&qu ...
- Android实现传感器应用及位置服务
Android实现传感器应用及位置服务 开发工具:Andorid Studio 1.3 运行环境:Android 4.4 KitKat 代码实现 这里需用获取加速度传感器和地磁传感器,手机获取旋转的方 ...
- 一、换系统wince ---到 linux ubuntu 桌面
今天终于拿到了OK6410,还是比较满意.预装的为wince系统,现要修改为linux 按照 OK6410-A开发板LINUX3.0.1-2013-01用户手册 进行 1.遇到问题.无法写入SD卡( ...
- ios 唯一标示符
大家知道苹果每部 iOS 设备都有一个 UDID,它就像设备的身份证一样,记录着设备的名称.类型甚至一些关于用户的私人信息.通常情况下,UDID 的一个最大功能就是帮助广告发布商向特定用户推送定向广告 ...
- html readme
取html页面高度 document.documentElement.scrollHeight在IE和Chrome下,可以正常取到合适的全文高度,但是firefox下取到的则过高: 用document ...
- DNS劫持(网页打不开的解决方法)
我们上网,必不可少的就是DNS,在这里先介绍下DNS的相关知识. DNS 是域名系统 (Domain NameSystem) 的缩写,它是由解析器和域名服务器组成的.域名服务器是指保存有该网络中所有主 ...
- CentOS7下通过rpm方式安装MySQL及插入中文问题解决 [原创]
一 CentOS下通过rpm方式安装MySQL CentOS版本:CentOS-7 MySQL版本:MySQL-5.6.22 在网上搜了一下,Linux下安装MYSQL有三种方式: 1) 通过yum命 ...
- Careercup - Google面试题 - 5377673471721472
2014-05-08 22:42 题目链接 原题: How would you split a search query across multiple machines? 题目:如何把一个搜索que ...