java在线聊天项目0.5版 解决客户端向服务器端发送信息时只能发送一次问题 OutputStreamWriter DataOutputStream socket.getOutputStream()
没有解决问题之前客户端代码:
package com.swift; import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.io.BufferedWriter;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.net.Socket;
import java.net.UnknownHostException; import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.JTextField; public class ChatClientFrame2 extends JFrame { private static final long serialVersionUID = -118470059355655240L;
Socket s=null;
JLabel label_shang=new JLabel();
JLabel label_xia=new JLabel();
JTextField tf = new JTextField(38);
JTextArea ta = new JTextArea(15, 50);
JButton button=new JButton(); public ChatClientFrame2() {
setBounds(200, 200, 500, 400);
setTitle("客户端聊天工具 —— 0.4");
//对窗口进行大的布局,分为三行一列,在pBasic面板上添加三个面板shang zhong xia
JPanel pBasic=new JPanel();
pBasic.setLayout(new BorderLayout());//不设置默认也是这种布局模式
setContentPane(pBasic);//把面板放在窗口上,不记得用this.关键字
JPanel shang=new JPanel();
JPanel zhong=new JPanel();
JPanel xia=new JPanel();
//设置JPanel面板的大小
shang.setSize(470, 25);
zhong.setSize(470, 180);
xia.setSize(470, 40);
pBasic.add(shang,BorderLayout.NORTH);
pBasic.add(zhong,BorderLayout.CENTER);
pBasic.add(xia,BorderLayout.SOUTH);
shang.setBackground(Color.red);
zhong.setBackground(Color.yellow);
xia.setBackground(Color.blue);
/*
* 三个面板,上边放一个标签“聊天记录”,中间放一个文本域,
* 下边分为左中右——分别放标签“输入信息“,文本框和”发送“按钮
*/
label_shang.setText("聊天记录");
shang.add(label_shang);
ta.setLineWrap(true);// 自动换行
JScrollPane scroll=new JScrollPane(ta);// 增加滚动条,以便不增加行数
zhong.add(scroll);
label_xia.setText("输入信息");
xia.add(label_xia,BorderLayout.WEST);
/*
* 增加功能,窗口监听事件,窗口打开时设置光标焦点在tf文本域中
*/
this.addWindowListener(new WindowAdapter() {
@Override
public void windowOpened(WindowEvent e) {
tf.requestFocus();
}
});
xia.add(tf,BorderLayout.CENTER);
button.setText("发送");
xia.add(button,BorderLayout.EAST);
/*
* 增加“发送”按钮的功能,增加回车的功能,监听相同,
* 则使用内部类实现,以提高代码复用性
*/
final class ShareListener implements ActionListener{ @Override
public void actionPerformed(ActionEvent e) {
String taText=ta.getText();
String tfText=tf.getText()+"\r\n";
ta.setText(taText+tfText);
tf.setText("");
//当回车或发送按钮时,tfText发送到服务器
try {
//可以尝试用DataOutputStream里的writeUTF方法
// DataOutputStream ds=new DataOutputStream(s.getOutputStream());
// ds.writeUTF(tfText);
// ds.flush();
// ds.close();
BufferedWriter bw=new BufferedWriter(new OutputStreamWriter(s.getOutputStream(),"utf-8"));
bw.write(tfText);
bw.close();
} catch (IOException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
} }
}
button.addActionListener(new ShareListener());
tf.addActionListener(new ShareListener());
//通过压缩自动调整各个面板
pack();
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);// 点关闭按钮同时退出程序
setVisible(true);
//创建窗体直接调用连接服务器
connect();
} /*
* 增加一个连接到服务器的方法
*/
public void connect() {
try {
s=new Socket("127.0.0.1",8888);
System.out.println("connected!");
} catch (UnknownHostException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} public static void main(String[] args) {
// 别忘了创建窗体对象,也可以用生成对象调用其他的方法如launchFrame()
new ChatClientFrame2();
} }
没有解决问题之前服务端代码:
package com.swift; import java.io.BufferedReader;
import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.ServerSocket;
import java.net.Socket; public class ChatServer { public static void main(String[] args) {
try {
ServerSocket ss = new ServerSocket(8888);
for (;;) {
Socket s = ss.accept();
//当有连接,则显示,用于测试
System.out.println("a client connected success");
//可以尝试用DataInputStream中readUTF方法
// DataInputStream dis=new DataInputStream(s.getInputStream());
// String str=dis.readUTF();
// dis.close(); BufferedReader br=new BufferedReader(new InputStreamReader(s.getInputStream(),"utf-8"));
String str=br.readLine();
System.out.println(str);
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} }
演示过程
出现问题:
java.net.SocketException: Socket is closed
如果不关闭流
把这句删掉bw.close();
是可以的,程序不会出错,但总要关闭,可以在客户端窗口关闭的时候执行这句。
继续运行发现还是只能发送一条信息,但没有出错提示。这时为什么呢?
查看程序发现,当每次点击回车或者发送按钮,都会有一条信息发送,但服务器就是没有显示,因为客户端发送了多次,服务端只读取了一次,所以显示一条,
下边代码是读取多次,显示多次:
package com.swift; import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.ServerSocket;
import java.net.Socket; public class ChatServer { public static void main(String[] args) {
try {
ServerSocket ss = new ServerSocket(8888);
for (;;) {
Socket s = ss.accept();
//当有连接,则显示,用于测试
System.out.println("a client connected success"); BufferedReader br=new BufferedReader(new InputStreamReader(s.getInputStream(),"utf-8"));
//问题:客户端多次发送信息,而服务器端只读取了一次,所以只输出一次,多次输出下面的办法也不好
String str=br.readLine();
System.out.println(str);
String str1=br.readLine();
System.out.println(str1);
String str2=br.readLine();
System.out.println(str2);
String str3=br.readLine();
System.out.println(str3);
}
} catch (IOException e) {
e.printStackTrace();
}
} }
接收了四次
可以增加一个死循环来无限接收
成功解决接收次数问题,可后边的关闭不能执行了
修改服务端代码如下:
0.5版效果图如下:
0.5版服务端代码:
package com.swift; import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.ServerSocket;
import java.net.Socket; public class ChatServer { public static void main(String[] args) {
boolean started=false;
try {
ServerSocket ss = new ServerSocket(8888);
started=true;
while(started) {
boolean connected=false;
Socket s = ss.accept();
System.out.println("a client connected success");
connected=true;
BufferedReader br=new BufferedReader(new InputStreamReader(s.getInputStream(),"utf-8"));
while(connected) {
String str=br.readLine();
System.out.println(str);
}
br.close();
}
} catch (IOException e) {
e.printStackTrace();
}
} }
0.5版客户端代码:(增加disconnect()方法,放入输出流和端口的关闭,优化输出流创建次数)
package com.swift; import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.io.BufferedWriter;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.net.Socket;
import java.net.UnknownHostException; import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.JTextField; public class ChatClientFrame extends JFrame { private static final long serialVersionUID = -118470059355655240L;
Socket s=null;
BufferedWriter bw=null;
JLabel label_shang=new JLabel();
JLabel label_xia=new JLabel();
JTextField tf = new JTextField(38);
JTextArea ta = new JTextArea(15, 50);
JButton button=new JButton(); public ChatClientFrame() {
setBounds(200, 200, 500, 400);
setTitle("客户端聊天工具 —— 0.5");
//对窗口进行大的布局,分为三行一列,在pBasic面板上添加三个面板shang zhong xia
JPanel pBasic=new JPanel();
pBasic.setLayout(new BorderLayout());//不设置默认也是这种布局模式
setContentPane(pBasic);//把面板放在窗口上,不记得用this.关键字
JPanel shang=new JPanel();
JPanel zhong=new JPanel();
JPanel xia=new JPanel();
//设置JPanel面板的大小
shang.setSize(470, 25);
zhong.setSize(470, 180);
xia.setSize(470, 40);
pBasic.add(shang,BorderLayout.NORTH);
pBasic.add(zhong,BorderLayout.CENTER);
pBasic.add(xia,BorderLayout.SOUTH);
shang.setBackground(Color.red);
zhong.setBackground(Color.yellow);
xia.setBackground(Color.blue); label_shang.setText("聊天记录");
shang.add(label_shang);
ta.setLineWrap(true);// 自动换行
JScrollPane scroll=new JScrollPane(ta);// 增加滚动条,以便不增加行数
zhong.add(scroll);
label_xia.setText("输入信息");
xia.add(label_xia,BorderLayout.WEST);
/*
* 增加功能,窗口监听事件,窗口打开时设置光标焦点在tf文本域中
*/
this.addWindowListener(new WindowAdapter() {
@Override
public void windowOpened(WindowEvent e) {
tf.requestFocus();
}
});
xia.add(tf,BorderLayout.CENTER);
button.setText("发送");
xia.add(button,BorderLayout.EAST); final class ShareListener implements ActionListener{ @Override
public void actionPerformed(ActionEvent e) {
String taText=ta.getText();
String tfText=tf.getText()+"\r\n";
ta.setText(taText+tfText);
tf.setText("");
//当回车或发送按钮时,tfText发送到服务器
try {
bw.write(tfText);
bw.flush();
} catch (IOException e1) {
e1.printStackTrace();
} }
}
button.addActionListener(new ShareListener());
tf.addActionListener(new ShareListener());
//通过压缩自动调整各个面板
pack();
this.addWindowListener(new WindowAdapter() {
@Override
public void windowClosing(WindowEvent e) {
//disconnect();//出错
System.exit(0);
}
});
setVisible(true);
//创建窗体直接调用连接服务器
connect();
} public void connect() {
try {
s=new Socket("127.0.0.1",8888);
System.out.println("connected!");
bw=new BufferedWriter(new OutputStreamWriter(s.getOutputStream(),"utf-8")); } catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
public void disconnect() {
try {
s.close();
bw.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} public static void main(String[] args) {
new ChatClientFrame();
} }
java在线聊天项目0.5版 解决客户端向服务器端发送信息时只能发送一次问题 OutputStreamWriter DataOutputStream socket.getOutputStream()的更多相关文章
- java在线聊天项目0.6版 解决客户端关闭后异常问题 dis.readUTF()循环读取已关闭的socket
服务端对try catch finally重新进行了定义,当发生异常,主动提示,或关闭出现异常的socket 服务器端代码修改如下: package com.swift; import java.io ...
- java在线聊天项目0.7版 连接多个客户端问题,开启多个客户端后服务器端只接收到一个 对各种异常的补充处理
问题的原因是 while(connected) { String str=dis.readUTF(); System.out.println(str); } 不断循环执行,一直在死循环获取socket ...
- java在线聊天项目0.9版 实现把服务端接收到的信息返回给每一个客户端窗口中显示功能之客户端接收
客户端要不断接收服务端发来的信息 与服务端不断接收客户端发来信息相同,使用线程的方法,在线程中循环接收 客户端修改后代码如下: package com.swift; import java.awt.B ...
- java在线聊天项目0.8版 实现把服务端接收到的信息返回给每一个客户端窗口中显示功能
迭代器的方式会产生锁定 服务器端增加发送给每个客户端已收到信息的功能 所以当获取到一个socket,并打开它的线程进行循环接收客户端发来信息时,我们把这个内部类的线程Client保存到集合List&l ...
- java在线聊天项目0.3版本 制作客户端窗体,实现发送按钮和回车发送信息功能,使用ActionListener监听事件中actionPerformed方法(用内部类和匿名内部类两种方法)
方法一,使用匿名内部类的监听方法,因方法一致代码稍冗余 package com.swift; import java.awt.BorderLayout; import java.awt.Color; ...
- java在线聊天项目0.2版本 制作客户端窗体,使用swing(用户界面开发工具包)和awt(抽象窗口工具包) BorderLayout布局与GridLayout布局不同之处 JPanel设置大小
代码如下: package com.swift; import java.awt.BorderLayout; import java.awt.Color; import javax.swing.JBu ...
- java在线聊天项目0.1版本 制作客户端窗体,使用swing(用户界面开发工具包)和awt(抽象窗口工具包)
建立Chat项目,并在项目中创建窗口类 package com.swift; import java.awt.BorderLayout; import javax.swing.JFrame; impo ...
- java在线聊天项目1.3版 ——设计好友列表框功能
设计好友列表框功能,思路—— 1.当客户端成功登陆后,则客户端把成功登陆信息发送给服务端, 2.由服务端将接收到来自各个成功登陆的客户端的用户信息添加进好友列表, 3.每当有成功登陆的用户就向各个客户 ...
- java在线聊天项目1.1版 ——开启多个客户端,分别实现注册和登录功能,使用客户端与服务端信息request机制,重构线程,将单独的登录和注册线程合并
实现效果图: eclipse项目中初步整合之前的各个客户端和服务端的窗口与工具类,效果如下图: 已将注册服务器线程RegServer功能放到LoginServer中,使用客户端与服务端的request ...
随机推荐
- java多线程中用到的方法详细解析
在多线程学习的过程中涉及的方法和接口特别多,本文就详细讲解下经常使用方法的作用和使用场景. 1.sleep()方法. 当线程对象调用sleep(time)方法后,当前线程会等待指定的时间(t ...
- 黑马方法引用学习 Stream流 函数式接口 Lambda表达式 方法引用
- SVN有任何胜过git的地方吗?
SVN有任何胜过git的地方吗? 好的技术问题通常会引出技术专家们依据经验得出的深层次的观点.但对于这样的问题的答案也很容易演变成完全基于个人喜好的情绪倾泄,而不是根据事实.标准和具体的专业知识.就比 ...
- 17.TLB
我们只想读4个字节,但我们要经过如下的步骤 读取 字节的PDE 读取 字节的 PTE 读取 字节(int 占用4字节)的物理内存 在 10-10-12 分页模式下,CPU 每次要访问额外的访问 8 字 ...
- ZJOI2017 day2 T2 线段树 想法题
考完D2发现自己简直zz了...花式扔基本分 首先这道题有个显然的套路:树上一些点到一个定点的距离和=这些点深度和+点数*定点深度和-2*lca深度和 ——上一次见这个套路是LNOI2014,上次做的 ...
- python之模块random,time,os,sys,序列化模块(json,pickle),collection
引入:什么是模块: 一个模块就是一个包含了python定义和声明的文件,文件名就是模块名字加上.py的后缀. 但其实import加载的模块分为四个通用类型. 1.使用python编写的代码(.py ...
- jQuery val()方法及valHooks源码解读
val: function( value ) { var hooks, ret, isFunction, elem = this[0]; if ( !arguments.length ) {//无参数 ...
- Unix高级环境编程之fcntl函数
#include <fcntl.h> int fcntl(int fd, int cmd, ...) fcntl功能 复制一个现有的描述符 (cmd = F_DUPFD) ##### 返回 ...
- Unity Shader入门精要学习笔记 - 第6章 开始 Unity 中的基础光照
转自冯乐乐的<Unity Shader入门精要> 通常来讲,我们要模拟真实的光照环境来生成一张图像,需要考虑3种物理现象. 首先,光线从光源中被发射出来. 然后,光线和场景中的一些物体相交 ...
- SVN状态图标消失的解决方法
有些时候我们会发现我们的SVN状态图标会突然消失,所有的文件夹都变成了普通的文件夹格式,这点很不利于我们进行管理. 进入任意一个文件夹,鼠标右键,依次进入TortoiseSVN---Settings ...