大家好,今天给大家介绍基于java+Socket+sqlserver的网络通讯系统设计与实现,文章末尾附有本毕业设计的论文和源码下载地址哦。
文章目录:
项目难度:中等难度
适用场景:相关题目的毕业设计
配套论文字数:13375个字36页
包含内容:整套源码+完整毕业论文+答辩PPT
提示:以下为毕业论文的简略介绍,项目源码及完整毕业论文下载地址见文末。
第1章 绪论
1.1 课题选择的背景
信息时代的到来引发了一场知识和信息革命,计算机和现代网络技术的广泛应用极大地促进了社会经济的发展,同时也带来了一些新兴产业的发展,比如聊天软件。随着计算机网络日新月异的发展,人们的交流方式越来越多,传统的交流方式已经难以满足人们的交流的需求了,网络聊天已经慢慢成为人们生活的一部分,呈现出良好的势头和广阔的发展前景。在互联网上即时的和好友取得联系,已经成为当今社会人们主流的联系方式。同时,即时通信系统对现代企业也有着重大意义,它能实现快速人际交流、数据共享,从而提高效率和生产力。实时通信为诸多企业开拓了网络应用的新领域。自从它诞生以来,以实时交互、资费低廉等优点,受到广大个人用户的喜爱,已经成为网络生活中不可或缺的一部分。越来越多的企业已开始认识到即时通信工具能够带来极高的生产力,并借助它的应用,来提高业务协同性及反馈的敏感度和快捷度。网上聊天系统是为人们之间进行交流和联系提供的一个平台。通过提供完善的网上聊天系统服务,可以达到增进彼此之间的了解,增进人与人之间的感情交流。因此,两台计算机之间进行即时通讯、发送文件等交流方式已经成为必然潮流。于是出现了QQ、UC 等聊天工具,然而QQ、UC 等聊天工具虽然方便、实用,但是,娱乐功能太多,有很多吸引人的娱乐功能,从一定程度上来说,是一种娱乐工具,不能作为用于即时通讯的专用工具。目前,用于实现单一的即时通讯的软件实在太少,为此,我们决定开发一个专用于实现多台计算机之间即时通讯的软件,以方便多台计算机之间信息的快速交流。
1.2 课题选择意义
省略
1.3 系统开发技术简介
1.3.1 Java简介
Java是由Sun Microsystems公司于1995年5月推出的Java程序设计语言(以下简称Java语言)和Java平台的总称。它以C++为基础,但是却是一个全新的软件开发语言。Java是一个简单,面象对象,分布式,解释性,强壮,安全,与系统无关,可移植,高性能,多线程和动态的语言-------这是 Sun给Java的定义。
Java平台由Java虚拟机(Java Virtual Machine)和Java 应用编程接口(Application Programming Interface、简称API)构成。Java 应用编程接口为Java应用提供了一个独立于操作系统的标准接口,可分为基本部分和扩展部分。在硬件或操作系统平台上安装一个Java平台之后,Java应用程序就可运行。现在Java平台已经嵌入了几乎所有的操作系统。
Java是一种简单的,面向对象的,分布式的,解释型的,健壮安全的,结构中立的,可移植的,性能优异、多线程的动态语言。Java 对开发者具有不可估量的价值,使他们可以:
• 在一个平台上编写软件,然后即可在几乎所有其他平台上运行
• 创建可在 Web 浏览器和 Web 服务中运行的程序
• 开发适用于在线论坛、存储、投票、HTML 格式处理以及其他用途的服务器端应用程序
• 将采用 Java 语言的应用程序或服务组合在一起,形成高度定制的应用程序或服务
• 为移动电话、远程处理器、低成本的消费产品以及其他任何具有数字核心的设备编写强大而高效的应用程序
Sun公司的口号就是"网络就是计算机",Java能使所有东西从桌面计算平稳的转变为基于网络的计算,它是专门为此而建立的,并显然是为了完成这个任务而来的。使用Java,我们可以相对轻松的一天编写一个有条理的网络程序。今天,Java的网络功能正在飞跃发展,不断有新的特性增加到这个有价值的基础上,JavaSoft实验室正在不断努力使Java更加完善。
1.3.2 JDBC技术简介
JDBC(Java Data Base Connectivity,java数据库连接)是一种用于执行SQL语句的Java API,可以为多种关系数据库提供统一访问,JDBC是Java的开发者Sun的Javasoft公司制定的。它由一组用Java语言编写的类和接口组成。JDBC为工具/数据库开发人员提供了一个标准的API,据此可以构建更高级的工具和接口,使数据库开发人员能够用纯 Java API 编写数据库应用程序。
有了JDBC,向各种关系数据发送SQL语句就是一件很容易的事。换言之,有了JDBC API,就不必为访问Sybase数据库专门写一个程序,为访问Oracle数据库又专门写一个程序,或为访问Informix数据库又编写另一个程序等等,程序员只需用JDBC API写一个程序就够了,它可向相应数据库发送SQL调用。同时,将Java语言和JDBC结合起来使程序员不必为不同的平台编写不同的应用程序,只须写一遍程序就可以让它在任何平台上运行,这也是Java语言“编写一次,处处运行”的优势。
Java数据库连接体系结构是用于Java应用程序连接数据库的标准方法。JDBC对Java程序员而言是API,对实现与数据库连接的服务提供商而言是接口模型。作为API,JDBC为程序开发提供标准的接口,并为数据库厂商及第三方中间件厂商实现与数据库的连接提供了标准方法。JDBC使用已有的SQL标准并支持与其它数据库连接标准,如ODBC之间的桥接。JDBC实现了所有这些面向标准的目标并且具有简单、严格类型定义且高性能实现的接口。
Java 具有坚固、安全、易于使用、易于理解和可从网络上自动下载等特性,是编写数据库应用程序的杰出语言。所需要的只是 Java应用程序与各种不同数据库之间进行对话的方法。而 JDBC 正是作为此种用途的机制。
ODBC在Windows系列平台应用程序中和JDBC在Web和Internet应用程序中的作用的作用类似。ODBC(OpenData Base Connectivity),称为开放式数据库互联技术,是由Microsoft公司倡导并得到业界普遍响应的一门数据库连接技术。 JDBC现在可以连接的数据库包括:xbase、Oracle、Sybase、Aceess以及Paradox等。
JDBC的工作机制 :JDBC定义了Java语言同SQL数据之间的程序设计接口。JDBC有一个非常独特的动态连接结构,它使得系统模块化。使用JDBC来完成对数据库的访问包括以下四个主要组件:Java的应用程序、JDBC驱动器管理器、驱动器和数据源。
JavaSoft公司开发了JDBC API,JDBC API是一个标准统一的SQL数据存取接口。 JDBC在Internet中的作用与ODBC在Windows系列中的作用类似。它为Java程序提供了一个统一缝地操作各种数据库的接口,程序员编程时,可以不关心它所要操作的数据库是哪个厂家的产品,从而提高了软件的通用性,而且在Internet上确实无法预料你的用户想访问什么类型的数据库。只要系统上安装了正确的驱动器组,JDBC应用程序就可以访问其相关的数据库。
1.3.3 数据库Microsoft SQL-Server 简介
省略
存储过程是存储在服务器上的预先编译好的SQL语句,可以在服务器上的SQL Server环境下运行[12]。使用存储过程有如下的好处:执行时间短;客户/服务器开发分离;安全性高。SQL-Server 2000工具界面与标准Windows 界面非常类似,具有菜单栏、图标、树状视图控件、单选按钮等。这些熟悉的界面使以前即使从没有见到过这些应用程序的人也能够很快地了解一些基本的功能。
1.3.4 多线程介绍
Java 支持内置多线程编程。多线程程序包括两条或两条以上并发运行的部分。程序中每个这样的部分都叫一个线程,每个线程都是独立的执行路径。因此,多线程是多任务处理的一种特殊形式。
多任务处理有两种截然不同的类型:基于进程的和基于线程的。进程本质上一个正在执行的程序。因此,基于进程的多任务处理的特点是允许你的计算机同时运行两个或更多的程序。举例来说,基于进程的读任务处理使你在运行文本编辑器的时候可以同时运行Java 编译器。在基于进程的多任务处理中,程序是调度程序多分派的最小代码。也就是说,进程是资源申请、调度和独立运行的单位,因此,它使用系统中的运行资源。
在基于线程的多任务处理环境中,线程是最小的执行单位。线程是进程中的一个单一的连续控制流程。一个进程可以拥有多个线程。这意味着一个程序可以同时执行两个或多个任务的功能。例如,一个文本编辑器可以在打印的同时格式化文本。所以,多进程程序处理大局问题,而多线程程序处理细节问题。
多线程程序比多进程程序需要更少的管理开销。进程是重量级的任务,需要分配它们自己独立的地址空间。进程间通信是昂贵和受限的。进程间的转换也是很需要花费的。另一方面,线程是轻量级的选手,它和进程一样拥有独立的执行控制,由操作系统负责调度。区别在于线程没有独立的存储空间,而是和所属进程的其他线程共享一个存储空间,因此,线程间通信是容易的,线程间的转换也是低开销的。当Java 程序使用多进程的任务处理环境时,多进程的程序不受Java 的控制,而多线程则受控于Java。
多线程可以帮会你写出CPU 最大利用率的高效程序,因为空闲时间保持最低。这对Java 运行的交互式的网络互联环境至关重要,因为空闲时间是公共的,举个例子来说,网络的数据处理传输速率远低于计算机处理能力,本地文件系统资源的读写速度远低于CPU 的处理能力,当然,用户输入也比计算机慢的多。在传统的单线程环境中,你的程序必须等待每一个这样的任务完成以后才能执行下一步—尽管CPU 有很多空闲时间。多线程是你能够获得并充分利用这些空闲时间。
·多线程编程简单,效率高(能直接共享数据和资源,而多进程却不能)
·适合于开发多种交互接口的程序
·减轻编写交互频繁,涉及面多的程序的困难
在本系统中,设置后台线程处理服务器和客户程序的通信。服务器资源以轻量级的线程提供,和多个客户线程并发连接通信。这样可以减轻服务器端的开销,同时保证了多个用户端连接能够得到快速而高效的服务。
第2章 系统分析
2.1 系统需求
随着网络的迅速发展,通过即时通讯工具进行相互交流,沟通会对用户带来很多的方便,像现在很流行的QQ、MSN、UC等,它们为用户们之间的沟通带来很多的方便的,但是这些即时工具在企业或者学校禁止用户访问外网的时候,这些即时的通信工具就不能使用了,用户之间就不能很方便的沟通交流了,可以通过自己设计一个即时通讯工具,自己建立服务器,放在内部网络,在用户不能访问外部网络的时候,可以很方便用户之间进行沟通交流。
2.1.1 系统的特点分析
基于JAVA的网络聊天系统基于C/S 模式实现,采用Java 语言编写。所有用户都通过服务器端中转消息,这也是现有大多数聊天系统所采用的模式。本系统真正地实现了多点聊天的功能,可以在多点间快速传递信息。同时,它体积小,功能单一,界面简洁美观,容易上手,只要简单阅读服务器端和客户端的帮助文档即可完全掌握它的使用方法,同时该系统是网络聊天,特别是局域网聊天不可多得的好帮手。
2.1.2 网络聊天系统需要解决的问题
首先,网络聊天系统是一个比较复杂的系统,良好的架构设计非常的重要:C/S 模式是较为成熟的软件运行模式,基于局域网设计,运行速度快,安全性高,为当今主要的软件所采用的模式。不好的架构会在系统实现或者维护阶段系统的扩展带来很多麻烦。
其次,服务器的承载能力,如何处理多个用户的同时登录和请求,好的设计会使系统在很多并发用户登录的时候不至于系统崩溃。
最后,服务器和客户端的通信协议制定比较困难,如何制定一套简单,并且易于客户端和服务器之间理解的通信规则非常重要,否则会在服务器和客户端通信设计的时候带来很多的问题。
2.2 系统设计思想
本系统的设计思路如下:
(1) 系统采用模块化设计,针对不同的应用设计不同的模块,并对这些模块都进行集中式管理,作为应用程序有较强的可操作性和扩展性。
(2) 合理的数据流设计,在应用系统设计中,相对独立的模块间以数据流相互连接,使各模块间的耦合性较低,方便系统运行,提高系统安全性,尽量保证低耦合,高内聚。
(3) 采用封装的思想,根据模块的不同情况进行类的封装,并重用各个模块,并合理利用各种设计模式,为以后维护带来方便。
2.3 系统实现功能分析
本次设计主要实现了以下几个功能:
● 客户端功能设计如下:
(1)身份认证,该模块主要实现用户的登陆认证,以及服务器代理IP和端口的设置,并检查数据的合法性。
(2)聊天功能,该模块主要用Socket实现不同用户之间消息的的发送,并且不仅仅传文本格式的数据,也可以传输图片,需要自己封装并自定义消息流,并能够识别显示。
(3)文件传输,该模块主要实现不同用户之间进行文件的传输。
● 服务器端功能设计如下:
(1)在线用户统计,该模块主要实现在线用户的查看显示,并可以控制其在线状态,强迫特定用户下线。
(2)日志记录,该模块主要记录客户端用户上线、下线情况,并存储下来。
(3)发布信息,该模块主要服务器可以发布系统信息,使客户端可以接收。
第3章 系统总体设计
3.1 总体构架
基于Java的网络聊天系统设计与实现主要考虑三个个方面的设计,即服务器模块设计,服务器客与客户端通信规则设计和客户端模块设计,本系统整体构架如图3.1所示。
在这里插入图片描述
图3.1 系统整体构架图
3.2 系统整体功能
基于JAVA的网络聊天系统分为服务器端和客服端,其中:
服务器端包括:在线用户管理模块,用户管理模块,部门管理模块,日志管理模块;
客服端包括:用户认证模块,主界面模块,聊天模块,文件传输模块。
系统整体功能设计如下图2.2所示
图3.2 系统整体功能设计图
3.3 系统模块功能设计
整个系统分为3个模块,其中主要功能如下:
(一)服务器端模块
服务器端模块首先设计多线程来处理客户端的连接,当受到客户端请求的时候,建立一个新的线程来处理客户端的连接。并且在一个注册中心中登记该线程,并存储客户端用户的一些信息,方便服务器统计在线用户,以便与这些在线用户进行通信。除此之外,为了方便用户更好的操作和掌握服务器端,系统还设计四个管理服务器的模块,分别是在线用户管理模块,用户管理模块块,部门管理管理模块和日志管理模块。各个模块功能设计如下:
(二)协议规则模块
该模块主要实现服务器与客户端之间通信规则的制定。其具体思路是:在服务器使用Socket通信的时候,把服务器与客户端通信的信息封装为一个类对象,然后通过自己手动将这些类对象转化为输入流,并在另一端输出流,根据自己制定的规则逆序解析流,把流转化为类对象。实现服务器与客户端之间的通信。其中具体的表示信息要自己制定,比如登陆成功标示,失败标示,修改密码标示,聊天标示等,这些都需要自己通过一些数字标记。
(三)客户端模块
客户端模块主要实现,用户账号到服务器的认证,以及登陆之后与其他用户通信或者部门通信,除了简单文本聊天之外,也考虑用户之间文件传输的功能。其中客户端之间的通信,是在客户端之间搭建服务器,也即客户端自己建立SocketServer,并用多线程来处理对不同用户的聊天。用户之间文件传输也是通过搭建客户端之间的服务器来实现文件出传的,不需要通过服务器来实现信息的传输,可以减少延迟,也减少了服务器端的流量损失。但是在客户之间搭建连接的时候仍然需要通过服务器来实现一些简单的通知操作,实现客户端与客户端之间搭建其连接。因此客户端大体分四个模块,分别是用户认证模块,主界面模块,聊天模块和文件传输模块。各个功能模块设计如下:
4.1 系统E-R图
(1) E-R模型
基于网络聊天系统数据库E-R模型,如图 所示:
图4.1 聊天系统E-R图
4.2 数据字典
(1) 用户表 (用户ID(主键),用户名,密码,用户密码,性别,头像ID,部门ID,年龄,电话号码,地址,注册时间,状态)。
(2) 分组信息表(分组ID(主键),分组名字,标记)。
4.3 各个数据表的创建
基于网络聊天系统数据库包括部门信息以及用户信息,部门信息表格和用户信息表格的设计结果如表4.1,表4.2所示:每个表格表示在数据库中的一个表。
第5章 系统实现与编码
5.1 服务器功能模块的设计
5.1.1 Socket和ServerSocket介绍
Socket,简称套接字,用于实现网络上客户和服务器之间的连接。也就是说网络上两个或两个以上双工方式通信的进程之间总有一个连接,这个连接的端点成为套接字,套接字是在比较低的层次上通信的。
具体的说:一个服务器应用程序一般侦听一个特定的端口等待客户端的连接请求,当一个连接请求到达时,客户端和服武器端建立一个通信连接,在连接过程中,客户端被分配一个本地端口与一个socket建立连接,客户端通过写socket来通知服务器,以读socket中的信息,类似的服务器也获得一个本地端口,它需要一个新的端口号来侦听原始端口上的其他连接请求。服务器也通过它的本地端口连接一个socket,通过读写和客户端通信。
Socket程序的工作过程:
1、建立Socket连接:在通信开始之前由通信双方确认身份,建立一条专用的虚拟连接通道。
2、数据通信:利用虚拟连接通道传送数据信息进行通道。
3、关闭:通信结束时,再将所建的虚拟连接拆除。
实现套接字的服务端,需要使用ServerSocket 类。ServerSocket类是服务器程序的运行基础,它允许程序绑定一个端口号来监听客户端的请求,一旦产生客户端请求,它将接受这一请求,同时产生一个完整的Socket 连接对象。服务器绑定的端口必须公开,以便让客户端程序知道如何连接这个服务器。同时,作为服务器,它必须能够接收多个客户的请求,这就需要为服务器设置一个请求队列,如果服务器不能马上响应客户端的请求,要将这个请求放进请求队列中,等服务器将当前的请求处理完,会自动到请求队列中按照先后顺序取出请求进行处理。服务器的资源是有限的,这就导致它的最大连接数是有限的,通过ServerSocket 的构造函数可以指定这个最大连接数。如果不明
确指定这个连接数,默认最大连接数为50,也就是说,客户端的请求队列最大可以容纳50 个请求,当超过这个最大连接数时,用户的请求将不再会被响应。 利用SocketServer 也提供了一些方法,它们主要有:
accept() 返回一个“已连接”的Socket 对象
getInetAddress() 得到该服务器的IP 地址
getLocalPort() 得到服务器所侦听的端口号
setSoTimeout() 设置服务器超时时间
getSoTimeout() 得到服务器超时时间
服务器和客户端通过Socket简单通信框架下图所示。
图 5.1 服务器和客户端通过Socket简单通信框架
5.2 服务器功能模块的实现
5.2.1 多线程服务器模块实现
服务器模块核心功能是使用Socket Server实现多线程的服务器,针对每一个客户建立一个单独的线程处理客户端的请求。其具体思路是,首先建立Socket Server,并绑定服务器IP和某个未使用的端口。然后监听该端口,如果有客户端的Socket连接则建立一个客户端线程ClientThread类对象,有该对象处理该客户的一些请求,并在ClientThread中保存User Bean对象,该对象保存了用户的一些基本信息,比如账号、密码、年龄、是否在线等。如果服务器没有关闭Socket Server则服务器则一直处理客户端连接。其核心代码如下。
public class MessageServer extends Thread {
//服务器端SocketServer
private ServerSocket server;
private boolean isStop = false;
public MessageServer(int port) throws IOException {
server = new ServerSocket(port);
}
/**
* 消息服务
*/
public void run() {
Socket client;
try {
while (!isStop) {
//建立处理客户端连接的线程
client = server.accept();
(new ClientThread(client)).start();
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (server != null) {
try {
server.close();
} catch (IOException e1) {
// e1.printStackTrace();
}
}
isStop = true;
}
}
}
5.2.2 在线用户管理模块实现
该模块主要实现在线用户统计、多线程服务器的启动关闭、以及服务器发送系统公告等功能。其中在线用户统计是在客户端每次连接的时候服务器建立了ClientThread来单独处理某个客户请求,如果该客户登陆成功的话则在工具类PubValue中一个HashMap添加该用户信息,查看在线用户信息的时候,可以通过工具类PubValue来的HashMap读取在线用户列表。系统公告则是通过服务器根据在线用户列表,广播系统公告到每个客户端,实现信息的公布。其实核心代码如下:
private void login(Message message) throws IOException {
if (message.getType().equals(PackOper.LOGIN)) {
User user = (User) message;
List list = UserDaoFactory.getUserDao().selectId(user.getId());
if (list.size() > 0) {
User tempUser = (User) list.get(0);
if (tempUser.getPassword().equals(user.getPassword())) {
if (tempUser.getIsOnline() == Parameter.ONLINE) { message.setType(PackOper.LOGIN_ONLINED);
myObjectOut.writeMessage(message);
this.setStop();
return;
}
ServerUI.getInstance().getLogUI().addLog(
LogOper.getInstance().insertOnLineLog(tempUser));
//登录成功后
//先通知所有人
//再通知此用户
//再把自己加到线程组中
//接着更新数据库
//下载树给此用户
tempUser.setType(PackOper.LOGIN_SUCCEED);
tempUser.setIsOnline(pub.Parameter.ONLINE);
this.chatCompany(tempUser);
this.user = tempUser;
this.myObjectOut.writeMessage(tempUser);
PubValue.addUserThread(this.user.getId(), this);
UserDaoFactory.getUserDao().setOnline(tempUser.getId(),
Parameter.ONLINE);
ServerUI.getInstance().getOnLineUI().updateOnLine();
try {
Thread.sleep(10);
} catch (InterruptedException e1) {
// e1.printStackTrace();
}
// 登录成功后发送树
PubValue.company.setType(PackOper.ADD_COMPANY);
myObjectOut.writeMessage(PubValue.company);
list = DepartmentDaoFactory.
getDepartmentDao().select(null);
Iterator it = list.iterator();
while (it.hasNext()) {
message = (Message) it.next();
message.setType(PackOper.ADD_DEPARTMENT);
myObjectOut.writeMessage(message);
}
list = UserDaoFactory.getUserDao().select(null);
it = list.iterator();
while (it.hasNext()) {
message = (Message) it.next();
message.setType(PackOper.ADD_USER);
myObjectOut.writeMessage(message);
}
return;
}
}
message.setType(PackOper.LOGIN_DEFEATED);
myObjectOut.writeMessage(message);
this.setStop();
}
}
其实现效果如图所示:
图5.2 聊天服务器端界面
5.2.3 部门管理模块实现
该模块主要实现对部门进行管理,根据情况添加,删除或者修改部门等。其设计主要是通过SQL语句来操作数据库数据的。跟普通的管理系统设计思路一样。没有太多的算法,主要是SQL语句的构造,以及数据库操作的API函数的使用。其实现效果如图所示.
图5.3 部门管理模块界面
5.2.4 用户管理模块实现
该模块类似于部门管理模块的设计,主要是针对用户的一些信息进行管理,根据需要添加、删除或者修改用户信息。其实现主要也是SQL语句的构造以及JDBC驱动提供函数的使用。比较复杂的部分设计主要体现界面的设计,如何使用Swing构造友好并且易于交互的界面需要花费一些时间。以及实现组合条件的用户查询,可以根据用户账号查询,或者所在部门查询,或者编号查询,或者根据这三个条件组合查询,这一功能比较复杂,如何根据不同的情况构造合适的SQL语句进行数据查询,显得很重要,否则粗略的设计会增加很多冗余代码。
5.2.5日志管理模块实现
日志管理模块主要是监控服务器的一些状态,以及客户端用户登陆的情况。这些日志信息记录是通过工具类LogOper实现日志的记录。该类使用了设计模式中单例模式,保证服务器端在进行日志操作的时候只有一个实例在内存中进行日志的记录。其日志的存放路径主要是在系统目录下,其文件名为server.log。通过服务器UI界面查询日志信息也是通过读取该文件来查询日志记录的。其实现效果如下图5.5:
图5.5 日志管理模块界面
5.3协议设计与实现
5.3.1 协议规则
协议规则里主要用一些整数来表示服务器与客户端通信的时候包的类型。服务器可以按照这个规则来解析包,并根据包中的信息做出相应的操作。其标示的具体含义和定义规则下表5.1所示。
5.3.2 协议实现
协议的指定主要是考虑在进行通信的过程中,保证服务器和客户端的通信消息能够被双方正确的理解,并作出相应的处理,比如登陆失败消息,登陆成功消息,文件传输消息,聊天消息等,服务器和客户端必须要按照指定的规则去打包和解析消息。其中协议的设计是通过对Socket流进行自定义的封装,把已经定义好的对象的每一个属性按照unicode编码中的一些特殊的编码\u0000(空格)进行分隔封装,并发送输入和输出流,另一端安装封装的逆序把输出流封装为对象,并识别该对象随对应是什么样的消息,客户端或者服务器作出相应的处理。而通过继承输入输出流实现了服务器器信息的发送,但是具体的包类型是什么样的消息是通过标示来标记。通过标示来解析这些包是什么样类型的包,实现服务器和客户端之间可理解的通信。其核心类PackOper类主要代码如下:
/**
* 解包类
* @author Administrator
*/
public class PackOper implements PackInterface {/**
* 创建一个包,根据object类型
* @param object
* @return
*/
public byte[] createPackage(Object object) {
if (object == null) {
return null;
}
if (object instanceof MessagePack) {
return createMessagePack((MessagePack) object);
}
if (object instanceof User) {
return createUser((User) object);
}
if (object instanceof Department) {
return createDepartment((Department) object);
}
if (object instanceof Company) {
return createCompany((Company) object);
}
return null;
}
/**
* 创建消息包
* @param pack
* @return
*/
private byte[] createMessagePack(MessagePack pack) {
return createByte(STARTSEPARATOR + pack.getType() + SEAS
+ pack.getFrom() + SEAS + pack.getFromIP() + SEAS
+ pack.getFromPort() + SEAS + pack.getTo() + SEAS
+ pack.getMessage() + ENDSEPARATOR);
}
/**
* 解一般消息包
* @param strs
* @return
*/
private Message unbindMessagePack(String[] strs) {
MessagePack messagePack = null;
if (strs != null) {
try {
messagePack = new MessagePack();
messagePack.setType(strs[0]);
messagePack.setFrom(strs[1]);
messagePack.setFromIP(strs[2]);
messagePack.setFromPort(Integer.parseInt(strs[3]));
messagePack.setTo(strs[4]);
messagePack.setMessage(strs[5]);
} catch (Exception e) {
return null;
}
}
return
}
}
MyObjectInputStream.java,该文件继承输出流,并自定义解析输出流的规则。Unit包里封装了所有消息类封装Bean。MyObjectInputStream.java,该文件继承输入流,并自定义解析输入流的规则。PackOper.java该文件主要实现了,Socket发送字节流的解包规则,并转化字节流为对象。也即服务器实现客户端不同消息识别,服务器端并作出相应处理。
5.4客户端功能模块的设计与实现
5.4.1 登陆认证模块
该模块主要实现用户的登陆认证,以及服务器代理IP和端口的设置,并检查数据的合法性。主要是参考现在通用即时聊天工具QQ界面进行设计,没有采用具体的算法实现漂亮的算法,只是截取QQ界面的图片作为自己设计界面的背景。同时用户登录身份认证使用简单模拟的QQ协议通过发送登陆验证包消息给服务器,服务器解包之后查询数据库检查用户身份的合法性,并发送给客户端认证成功或者失败包,客户端解包之后如果认证成功则登陆成功,显示主界面,否则提示用户登陆失败,提示用户重新登陆。并且可以根据服务器IP和端口的变化进行动态设置服务器地址,具有很好的灵活性。具体实现效果如下图5.6所示。
图5.6 用户登陆界面
5.4.2 主界面模块
该模块是参考QQ设计,截取QQ图片作为自己界面设计背景素材。其中分组信息也是通过向服务器发送查询该客户端用户的分组以及好友信息包,服务器解包之后查询数据库,并将查询结果封装成对象之后,转化为Socket流发送给客户端,客户端将包信息解析封装为对象,并显示数据库到客户端主界面上。该模块设计难点主要在如何有效的使用Swing提供的现有组建去扩展,实现漂亮的交互界面。如图5.7:
图5.7 用户登陆后主界面
5.4.3 聊天模块
聊天模块设计思路如下:
该模块是现在比较流行的QQ作为参考,设计的聊天的界面,特别是表情的发送,表情的窗口是FaceUI类封装的,它调用相应的辅助类来实现模拟QQ表情的实现,同时,自定义封装了聊天文本框,来实现文字和图片混合的文本框,在发送的时候获取用户发送的内容,自己封装为预定义好的对象,并转化为Socket流发送,经过服务器,有服务器找到对应客户端并转发消息,另一个客户端运行专门的客户端线程检测消息,接收到服务器的消息之后,自动弹出聊天窗体,并把好友发过来的消息显示在聊天界面上。其中,ChatUI类封装了聊天主界面,实现聊天的功能。FaceUI类实现聊天的时候可以添加表情。其实现效果如下图5.8:
图5.8聊天界面
5.4.4 文件传输模块
该模块主要实现了用户之间的聊天功能,可以传输表情和文字。而且也可以实现不同用户之间多线程的文件传输功能,可以在聊天的同时不影响文件传输并行进行,而且多线程传输文件也提高了文件的传输速度。实现技术组要采用多线程文件传输模块设计,主要实现不同用户之间进行发送文件,SendStreamThread类,该类主要采用多线程的思想,发送Socket流,实现客户端和服务器的通信,并通过服务器实现客户端与另一个客户端通信的功能。ReceiveStreamThread类,该类主要采用多线程的思想,接收Socket流,实现客户端和服务器的通信,接收Socket流并根据QQ协议规则将Socket流转化为相应的消息对象,并通知本地主线程,作出相应的处理。该类无论是在服务器或者是客户端都在一直运行,只要发现有Socket流到来就接收,并交给主线程进行处理。其核心代码如下,以ReceiveStreamThread类为例。.
public class ReceiveStreamThread extends Thread {
private Socket socket;
private TransTask transTask;
public ReceiveStreamThread(Socket socket) {
this.socket = socket;
}
public void run() {
BufferedInputStream bufInStream = null;
BufferedOutputStream bufOutStream = null;
MyObjectInputStream myObjectIn = null;
MyObjectOutputStream myObjectOut = null;
try {
myObjectIn = new MyObjectInputStream(socket.getInputStream());
myObjectOut = new MyObjectOutputStream(socket.getOutputStream());
MessagePack msgPack = (MessagePack) myObjectIn.readMessage();
if (!msgPack.getType().equals(PackOper.UPFILE)) {
socket.close();
return;
}
// 寻找是否有此任务
this.transTask = TransFileStock.getReceiveFileUI(msgPack.getMessage());
if (transTask == null) {
socket.close();
return;
}
msgPack.setType(PackOper.UPFILE_FIAT);
myObjectOut.writeMessage(msgPack);
bufInStream = new BufferedInputStream(socket.getInputStream());
bufOutStream = new BufferedOutputStream(new FileOutputStream(
transTask.getSendFile()));
byte[] bufByte = new byte[1024];
int readLen = -1;
int readSum = 0;
while ((readLen = bufInStream.read(bufByte, 0, bufByte.length)) > 0
&& !transTask.getFileUI().isCancel()) {
bufOutStream.write(bufByte, 0, readLen);
readSum += readLen;
transTask.getFileUI().setPlan(readSum);
}
bufOutStream.flush();
if(readSum < transTask.getFileLength()){
transTask.getFileUI().setClose();
}else{
//传送完毕
transTask.getFileUI().setTransFinish();
}
TransFileStock.removeReceiveTrans(transTask.toString());
} catch (UnsupportedEncodingException e) {
// TODO 自动生成 catch 块
e.printStackTrace();
} catch (IOException e) {
// TODO 自动生成 catch 块
e.printStackTrace();
} }
}
其实现效果如下图5.9所示。
本项目源码及完整论文如下,有需要的朋友可以点击进行下载。如果链接失效可点击下方卡片扫码自助下载。
序号 | 毕业设计全套资源(点击下载) |
---|---|
本项目源码 | 基于java+Socket+sqlserver的网络通讯系统设计与实现(源码+文档)_java_Socket_CS架构_网络通讯系统.zip |