Java小程序之服务器的UI实现


Java小程序之服务器的UI实现
一、前言:
前面我们做的服务器都是没有界面的,只是利用简单的输入输出语句在控制台中打印语句;今天,我们要让服务器有界面,可视化操作;

二、UI功能需求:
Java小程序之服务器的UI实现_第1张图片
三、 知识点分析

JTable使用:TableModelJScrollPaneVector

JTable:可以理解为表示数据的展示组件

TableModel:用于封装数据组件

Vector: 队列

代码框架

Java小程序之服务器的UI实现_第2张图片
四、具体实现思路
1、先实现整个界面UI(功能暂时不着急)这里主要是JTable的使用,刚开始启动按钮和关闭按钮都不能操作
2、最上方功能按钮的实现,当界面初始化的时候,用一个线程开始不断检测输入框中是否有内容,有内容则启动按钮可操作;启动按钮启动后,创建服务器对象,等待客户端的连接,并将启动按钮再次设置为不可操作,而关闭按钮设置为可操作;
3、获取客户端输入的账户和密码,服务器校验成功后,将该账号的相关信息进行封装,将封装好的信息加到Table中显示
4、显示聊天功能,消息内容昨天实在控制台中打印的,这次,我们这要将从客户端接收到的消息放在服务器界面UI中显示即可;

五、源代码:
源代码结构图:
Java小程序之服务器的UI实现_第3张图片
com.huaxin.server包:
MyServer类:(什么时候启动服务器?当点击启动按钮时,执行者下面的代码)
package com.huaxin.server;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.BindException;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketException;
import java.util.ArrayList;

import javax.swing.JOptionPane;

import com.huaxin.UI.InfoPanel;
import com.huaxin.UI.TablePanel;

public class MyServer {
	
	public static ArrayList serverList = new ArrayList();
	public int port;
	public ServerSocket server;
	private TablePanel tablePanel;
	private InfoPanel infoPanel;
	public MyServer(int port, TablePanel tablePanel, InfoPanel infoPanel) {
		this.port=port;
		this.tablePanel=tablePanel;
		this.infoPanel=infoPanel;
	}
	
	//启动服务器
	public void startServer(){
		try {
			//创建服务器对象
			server = new ServerSocket(port);
			System.out.println("服务器已经启动......");
			
			while(true){
				//服务器连接客户端,阻塞方法
				Socket socket=server.accept();
				System.out.println("有客户端连进来了......");
				
				ServerThread st = new ServerThread(socket,tablePanel,infoPanel);
				st.start();
				serverList.add(st);
				
				
			}
		} 
		catch (BindException e) {
			JOptionPane.showMessageDialog(null, "端口号已经占用!请重新启动服务器并输入新的端口号!");
			System.exit(0);
		}
		catch (SocketException e) {
			JOptionPane.showMessageDialog(null, "服务器已关闭.....");
		}
		catch (Exception e) {
			e.printStackTrace();
		}
	}
	
	public void sendMsgToClient(String s) {

		for (int i = 0; i 

ServerThread类:
package com.huaxin.server;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Vector;

import com.huaxin.UI.InfoPanel;
import com.huaxin.UI.TablePanel;

import javafx.scene.control.TabPane;

public class ServerThread extends Thread {

	public Socket socket;
	public InputStream ins;
	public OutputStream os;
	public String str;
	public String name;
	public String pwd;
	private TablePanel tablePanel;
	private InfoPanel infoPanel;
	
	public ServerThread(Socket socket, TablePanel tablePanel, InfoPanel infoPanel) {
		this.socket = socket;
		this.tablePanel=tablePanel;
		this.infoPanel=infoPanel;
	}

	public void run() {
		
		Vector vs=null;
		
		try {

			// 获取输入输出流
			ins = socket.getInputStream();
			os = socket.getOutputStream();

			 str = "welcome to zhou's Server";
			// 向客户端输出信息
			sendMsg(os, str);
			os.flush();
			
			/************客户端登录以及验证****************/
			
			//获取客户端输入的账户和密码
			getMsg();
			//账号和密码校验
			boolean falg =loginCheck();
			//校验不通过时,循环校验
			while(!falg){
				str="Fail to connect server......";
				sendMsg(os, str);
				os.flush();
				
				str="please check your name and password and login again.....";
				sendMsg(os, str);
				os.flush();
				getMsg();
				falg =loginCheck();
			}
			
			//校验成功后:开始聊天
				str="successful connected..... you can chat with your friends now ......";
//				str="yes";
				sendMsg(os, str);
				os.flush();
				//登录成功后,将用户名,ip地址以及登录时间加到TabelPanel中的Table中
				//先获取到TabelPanel中的Table对象,一层一层传递过来
				
				String ip =socket.getRemoteSocketAddress().toString();
				Date date = new Date();
				SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
				String currentTime =sdf.format(date);
				
				//封装用户名,ip地址以及时间
				vs = new Vector();
				vs.add(name);
				vs.add(ip);
				vs.add(currentTime);
				
				//将封装好的vs加入到tablePanel面板的table中,并刷新面板
				tablePanel.data.add(vs);
				tablePanel.table.updateUI();
				
				// 接受客户端的信息
				String s = readMsg(ins);
				
				
				while (!s.equals("bye")) {
					
					//在消息框中显示发送的信息
					infoPanel.jta.append(name+" say:"+s+"\r\n");
					
					for (int i = 0; i < MyServer.serverList.size(); i++) {
						
						//群发消息
						ServerThread st = MyServer.serverList.get(i);
						//不给自己发消息
						if (st == this) continue;
						
						OutputStream ous =st.socket.getOutputStream();
						sendMsg(ous,name+" say:"+s);
						os.flush();
					    } 
					  //循环读取信息 
					s = readMsg(ins);
				}
			
			// server.close();
		} catch (Exception e) {
//			e.printStackTrace();
			System.out.println("客户端不正常退出......");
		}

		//关闭客户端后,将该用户的记录从向量队里中移除,并刷新面板
		if(vs!=null){
			this.tablePanel.data.remove(vs);
			this.tablePanel.table.updateUI();
		}
		// 关闭流、服务器、套接字
		try {
			ins.close();
			os.close();
			socket.close();
			
			MyServer.serverList.remove(this);
		} catch (IOException e) {
//			System.out.println("这个地方有错误......");
			e.printStackTrace();
		}
	}

	//发送消息的函数
	public void sendMsg(OutputStream os, String s) throws IOException {

		// 向客户端输出信息
		//
		byte[] bytes = s.getBytes();
		os.write(bytes);
		os.write(13);
		os.write(10);
		os.flush();

	}

	//读取客户端输入数据的函数
	public String readMsg(InputStream ins) throws Exception {
		// 读取客户端的信息
		int value = ins.read();
		// 读取整行 读取到回车(13)换行(10)时停止读
		String str = "";
		while (value != 10) {
			//点击关闭客户端时会返回-1值
			if(value ==-1){
				throw new Exception();
			}
			str = str + ((char) value);
			value = ins.read();
		}
		str = str.trim();
		return str;
	}
	
	//获取客户端账号和密码的函数
	public void getMsg() throws Exception {
		str="please enter your name :";
		sendMsg(os, str);
		os.flush();
		name =readMsg(ins);
		
		str="please enter your password :";
		sendMsg(os, str);
		os.flush();
		pwd =readMsg(ins);

	}
	
	//校验客户端输入的账号和密码的函数
	public boolean loginCheck() throws Exception{
		if(name.equals("zhou") && pwd.equals("zhou") 
				|| name.equals("user") && pwd.equals("pwd") 
				|| name.equals("huaxinjiaoyu") && pwd.equals("huaxinjiaoyu")){
			
			return true;
	   }
		return false;
 }
}

com.huaxin.UI包:
(该包分为将整个面板的三个区域分出了三个类,在UIFrame中创建该三个类,并整合三个类,达到上图的整体效果)

StartPanel类:(最上面的面板类)
package com.huaxin.UI;

import java.awt.Color;
import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.IOException;

import javax.swing.JButton;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JTextField;

import com.huaxin.server.MyServer;
import com.huaxin.server.ServerThread;

public class StartPanel extends JPanel{
	
	public JTextField jtf;
	public JButton startBtn;
	public JButton stoptBtn;
	public boolean flag=true;
	public MyServer ms ;
	public TablePanel tablePanel;
	public InfoPanel infoPanel;
	
	/*
	 * 最上面的面板类
	 */
	public StartPanel(TablePanel tablePanel, InfoPanel infoPanel) {
		
		this.tablePanel=tablePanel;
		this.infoPanel=infoPanel;
	
		//创建相关组件
		JLabel label =new JLabel("端口号");
		jtf = new JTextField(10);
		startBtn = new JButton("启动服务");
		//设置按钮不可操作
		startBtn.setEnabled(false);
		stoptBtn = new JButton("关闭服务");
		stoptBtn.setEnabled(false);
		//设置背景颜色
		this.setBackground(Color.green);
		//设置面板大小
		this.setPreferredSize(new Dimension(0,50));
		this.setLayout(new FlowLayout());
		//组件添加
		this.add(label);
		this.add(jtf);
		this.add(startBtn);
		this.add(stoptBtn);
		//给两个按钮添加监听器
		startBtn.addActionListener(al);
		stoptBtn.addActionListener(al);
		
		//检测输入框是否为空
		checkText();
	}
	
	//按钮监听器的具体实现
	ActionListener al = new ActionListener() {
		public void actionPerformed(ActionEvent e) {
			String command =e.getActionCommand();
			if(command.equals("启动服务")){
				//启动服务的相关逻辑操作
				flag=false;
				int port =Integer.parseInt(jtf.getText());
				ms = new MyServer(port,tablePanel,infoPanel);
				infoPanel.ms=ms;
				stoptBtn.setEnabled(true);
				startBtn.setEnabled(false);
				//用线程处理,避免阻塞
				new Thread(){
					public void run() {
						ms.startServer();
					};
				}.start();
				
			}else if(command.equals("关闭服务")){
				try {
//					flag=true;
					//关闭服务逻辑实现
					//移除所有的客户端
					while(ms.serverList.size()!=0) {
						ms.serverList.get(0).socket.close();
						ms.serverList.remove(0);
					}
					//服务器关闭
					ms.server.close();
					//设置按钮可操作属性
					stoptBtn.setEnabled(false);
					startBtn.setEnabled(true);
				} catch (IOException e1) {
					e1.printStackTrace();
				}
				
			}
		}
	};
	
	//线程检测文本输入框是否有输入
	public void checkText(){
		new Thread(){
			public void run() {
				while(flag){
					//获取文本输入框的信息
					String info=jtf.getText();
					//信息不为空或者"",则可以启动服务器
					if(!(info==null || info.equals(""))){
						startBtn.setEnabled(true);
					}
					
					try {
						this.sleep(500);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			};
		}.start();
	}
	

}


TablePanel类:登录用户信息显示面板
package com.huaxin.UI;

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.util.Vector;

import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.table.DefaultTableModel;

public class TablePanel extends JPanel {
	
	public JTable table;
	public Vector> data;
	
	public TablePanel() {
		//设置面板属性
		this.setPreferredSize(new Dimension(450,0));
		this.setBackground(Color.white);
		this.setLayout(new BorderLayout());
		
		//创建表格
		Vector colNames = new Vector();
		colNames.add("用户名");
		colNames.add("IP地址");
		colNames.add("登录时间");
		data =new Vector>();
		Vector vs=new Vector();
		
		//创建表格模型
		DefaultTableModel tm =new DefaultTableModel(data, colNames);
		table = new JTable(tm);
		//创建滚动条
		JScrollPane jsp =new JScrollPane(table);
		this.add(jsp);
	}
	
	

}


InfoPanel类:聊天信息显示面板
package com.huaxin.UI;

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JButton;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.JTextField;

import com.huaxin.server.MyServer;

public class InfoPanel extends JPanel{
	
	
	public MyServer ms ;
	public JTextArea jta;
	public JTextField jtf;
	
	
	public InfoPanel() {
		//设置信息面板的属性
		this.setPreferredSize(new Dimension(350,0));
		this.setBackground(Color.blue);
		this.setLayout(new BorderLayout());
		//文本输入域
		jta = new JTextArea();
		//设置自动换行
		jta.setLineWrap(true);
		JScrollPane jsp = new JScrollPane(jta);
		this.add(jsp);
		//面板底部的发送面板
		jtf = new JTextField(25);
		JButton btn = new JButton("发送");
		//发送按钮添加监听器
		btn.addActionListener(al);
		JPanel panel=new JPanel();
		panel.setLayout(new FlowLayout());
		this.add(panel,BorderLayout.SOUTH);
		panel.add(jtf);
		panel.add(btn);
		
	}
	
	//监听具体实现
	ActionListener al = new ActionListener() {
		
		public void actionPerformed(ActionEvent e) {
			//获取内容
			String s=jtf.getText();
			//为空的提示信息
			if( s==null || "".equals(s)){
				JOptionPane.showMessageDialog(null, "输入框不能为空!");
			}
			else if(ms.server.isClosed()){
				JOptionPane.showMessageDialog(null, "服务器已经关闭,不能发送消息!");
			}else{
				try {
					//不为空,则将消息发送给服务器,服务器转发给消息给每个客户端
					ms.sendMsgToClient(s);
					//清空文本
					jtf.setText("");
				} catch (Exception e1) {
					e1.printStackTrace();
				}
			}
		}
	};
}



UIFrame类:程序入口以及面板整合类:
package com.huaxin.UI;
import java.awt.BorderLayout;
import javax.swing.JFrame;

/*
 * 服务器后台界面类
 */
public class UIFrame extends JFrame {
	
	public TablePanel tablePanel;
	public InfoPanel infoPanel ;

	public static void main(String[] args) {
		UIFrame ui = new UIFrame();
		ui.initFrame();
	}

	public void initFrame() {
		// 设置窗体的相关属性
		this.setSize(800, 600);
		this.setTitle("服务器后台界面");
		this.setDefaultCloseOperation(3);
		this.setLocationRelativeTo(null);
		this.setLayout(new BorderLayout());
		tablePanel = new TablePanel();
		infoPanel = new InfoPanel();
		StartPanel startPanel = new StartPanel(tablePanel,infoPanel);
		
		//在主页面中添加三个面板
		this.add(startPanel,BorderLayout.NORTH);
		this.add(tablePanel, BorderLayout.CENTER);
		this.add(infoPanel, BorderLayout.EAST);
		this.setVisible(true);
	}	

}



六、运行结果:

先启动界面UI类,输入9090点击启动服务,开启服务器,服务器一定要先打开,否则客户端无法连接

Java小程序之服务器的UI实现_第4张图片



利用系统自带的telnet客户端,启动客户端
Java小程序之服务器的UI实现_第5张图片




输入正确的密码进行登录:

Java小程序之服务器的UI实现_第6张图片



当有用户登录进来后的界面显示效果:
Java小程序之服务器的UI实现_第7张图片



以同样的方法开启另外客户端
Java小程序之服务器的UI实现_第8张图片


界面UI小时效果:

Java小程序之服务器的UI实现_第9张图片


在三个客户端分别输入聊天消息:

Java小程序之服务器的UI实现_第10张图片


界面UI显示效果:
Java小程序之服务器的UI实现_第11张图片


测试系统发送功能:(在信息面板中的发送按钮添加监听)

Java小程序之服务器的UI实现_第12张图片



Java小程序之服务器的UI实现_第13张图片


七、总结:
1、不要着急,一步一步实现
2、服务那边的代码可以参看昨天的博客;
3、把界面的单个部分分成三个类出去实现,最后在一个类中创建三个类的对象,并将三个对象在该类中整合
4、一定按照思路来,越急躁的人越回一通乱来
个人经验分享:以前我也尝试照着书中的代码一句一句去敲,先把一个类的代码敲完,再去敲另一个了类的代码,其实这样是非常不好的,我们写代码的时候也不是一口气就把一个类的代码全部写完,同时,定义的变量也不是一下就能想到要那么多的变量,只是在我们需要的时候去定义某个变量;所以,写程序,思路真的非常非常重要;我把源代码都贴上来,也不是让某些人直接copy拿去交作业(当然,有部分人肯定是直接copy的),更重要的是帮助那些自学的朋友在某个地方卡壳的时候,参看一下的代码,能顺时顿悟;而浪费太多不必要的时间;最最重要的是能和大家一起学习讨论和交流;嗯,相信我的意思大家都明白!
共勉!

你可能感兴趣的:(Java小程序,Java小程序开发)