java服务器端线程体会
一个完整的项目包括服务器和客服端
服务器端初步编写:
(1)
服务器端应用窗口的编写 (服务器类Server):
包括窗口和组件的一些设置, 添加一些客服端的元素,如客服端在线用户表(Vector), 服务端接口ServerSocket, 服务端线程ServerThread
其次就是组件的事件监听以及相应的事件方法
(2)
服务器线程的编写 (服务器线程类 ServerThread):
服务器线程起分配服务器代理线程作用,接受客服端的连接同时为其分配一个专用的服务器代理线程.
编写相对简单,给出服务器(Server) 通过其服务端接口(ServerSocket)监听客服端的连接(Socket),同时为其分配代理线程(ServerAgentThread)
public class ServerThread extends Thread
{
Server father; //声明Server的引用
ServerSocket ss;//声明ServerSocket的引用
boolean flag=true;
public ServerThread(Server father)
{//构造器
this.father=father;
ss=father.ss;
}
public void run()
{
while(flag)
{
try
{
Socket sc=ss.accept();//等待客户端连接
ServerAgentThread sat=new ServerAgentThread(father,sc);
sat.start();//创建并启动服务器代理线程
}
catch(Exception e)
{
e.printStackTrace();
}
}
}
}
(3)
服务器代理线程(ServerAgentThread) :代理线程对应相应的客服端 起到接受发送信息作用 一个服务器代理线程与一个客服端唯一对应
服务器代理线程类是服务器主要类,
其中包括一些基本的代理线程内容 如服务器(Server) 客服端连接(Socket) 数据输入输出(DateInputStream DateOutputStream)
还有与客服端的数据的一些交流方法, 线程run方法起判断客服端信息作用
public class ServerAgentThread extends Thread
{ Server father;//声明Server(服务器Server)的引用
Socket sc;//声明Socke(客服端接口Socket)t的引用
DataInputStream din;//声明数据输入流与输出流
DataOutputStream dout;
boolean flag=true;//控制线程的生命周期
public ServerAgentThread(Server father,Socket sc)
{
//构造方法一般初始化 服务器 客服端接口 还有应用端接口数据输入输出
this.father=father;
this.sc=sc;
try
{
din=new DataInputStream(sc.getInputStream());//创建数据输入流
dout=new DataOutputStream(sc.getOutputStream());//创建数据输出流
}
catch(Exception e)
{
e.printStackTrace();
}
}
public void run()
{
while(flag)
{
try
{
String msg=din.readUTF().trim();
//接收客户端传来的信息判断并作出相应的回应
//其中单个回应一般包含对服务器端的操作数据回应 以及对客服端的操作数据回应
if(msg.startsWith("c"))//收到新用户的信息
{
this.nick_name(msg);
}
else if(msg.startsWith("<#CLIENT_LEAVE#>")){
this.client_leave(msg);
}
else if(msg.startsWith("<#TIAO_ZHAN#>")){
this.tiao_zhan(msg);
}
else if(msg.startsWith("<#TONG_YI#>")){
this.tong_yi(msg);
}
else if(msg.startsWith("<#BUTONG_YI#>")){
this.butong_yi(msg);
}
else if(msg.startsWith("<#BUSY#>")){
this.busy(msg);
}
else if(msg.startsWith("<#MOVE#>")){
this.move(msg);
}
else if(msg.startsWith("<#RENSHU#>")){
this.renshu(msg);
}
}
catch(Exception e)
{
e.printStackTrace();
}
}
} //单个操作方法的书写
//此处省略几个
//新用户
public void nick_name(String msg)
{
try
{
String name=msg.substring(13);//获得用户的昵称
this.setName(name);//用该昵称给该线程取名
Vector v=father.onlineList;//获得在线用户列表
boolean isChongMing=false;//重名flag
int size=v.size();//获得用户列表的大小
for(int i=0;i<size;i++)
{
//遍历列表,查看是否已经有该用户名
ServerAgentThread tempSat=(ServerAgentThread)v.get(i);
if(tempSat.getName().equals(name))
{
isChongMing=true;//有重名,将标志位设为true
break;
}
}
if(isChongMing==true)//如果重名
{
dout.writeUTF("<#NAME_CHONGMING#>");//将重名信息发送给客户端
din.close();//关闭数据输入流
dout.close();//关闭数据输出流 //重点部分 重名需要终止分配的该服务器代理线程以免产生安全问题
sc.close();//关闭Socket
flag=false;//终止该服务器代理线程
}
else//如果不重名
{
v.add(this);//将该线程添加到在线列表
father.refreshList();//刷新服务器在线信息列表
String nickListMsg="";
size=v.size();//获得在线列表大小
//(实时更新)
for(int i=0;i<size;i++)
{
ServerAgentThread tempSat=(ServerAgentThread)v.get(i);
nickListMsg=nickListMsg+"|"+tempSat.getName();
}
nickListMsg="<#NICK_LIST#>"+nickListMsg;
//将在线用户组成字符串 <#NICK_LIST#>|1|2|3|4|5|6|7
Vector tempv=father.onlineList;
size=tempv.size();
for(int i=0;i<size;i++)
{//遍历在线列表
ServerAgentThread satTemp=(ServerAgentThread)tempv.get(i);
satTemp.dout.writeUTF(nickListMsg);//将最新的列表信息发送到各个客户端
if(satTemp!=this)
{//给其他客户端发送新用户上线的信息
satTemp.dout.writeUTF("<#MSG#>"+this.getName()+"上线了...");
}
}
}
}
catch(IOException e)
{
e.printStackTrace();
} } } //用户下线
public void client_leave(String msg){
try{
Vector tempv=father.onlineList;//获得在线列表
tempv.remove(this);//移除该用户代理线程
int size=tempv.size();
String nl="<#NICK_LIST#>";
for(int i=0;i<size;i++){//遍历在线列表
ServerAgentThread satTemp=(ServerAgentThread)tempv.get(i);
//向各个客户端发送用户离线信息
satTemp.dout.writeUTF("<#MSG#>"+this.getName()+"离线了...");
//组织信息的在线用户列表
nl=nl+"|"+satTemp.getName();
}
for(int i=0;i<size;i++){//将最新的列表信息发送到各个客户端
ServerAgentThread satTemp=(ServerAgentThread)tempv.get(i);
satTemp.dout.writeUTF(nl);
} (重点部分 线程下线需要关闭其相关代理)
this.flag=false;//终止该服务器代理线程
father.refreshList();//更新服务器在线用户列表
}
catch(IOException e){e.printStackTrace();}
}
java服务器端线程体会的更多相关文章
- Java Socket线程的设计原理介绍
转自:http://developer.51cto.com/art/201003/190001.htm Java Socket线程我们经常会用到的技术,但是有很多程序员还是有不少的使用问题,下面我们就 ...
- java之线程
java之线程 一:线程: 线程是什么呢?线程,有时被称为轻量级进程是程序执行流的最小单元.一个标准的线程由线程ID,当前指令指针(PC),寄存器集合和堆栈组成.另外,线程是进程中的一个实体,是被系统 ...
- Java 使用线程方式Thread和Runnable,以及Thread与Runnable的区别
一. java中实现线程的方式有Thread和Runnable Thread: public class Thread1 extends Thread{ @Override public void r ...
- Java的线程安全
线程安全 我们这里讨论的线程安全,就限定于多个线程之间存在共享数据访问这个前提,因为如果一段代码根本不会与其他线程共享数据,那么从线程安全的角度来看,程序是串行执行还是多线程执行对它来说是完全没有区别 ...
- 深入理解Java之线程池
原作者:海子 出处:http://www.cnblogs.com/dolphin0520/ 本文归作者海子和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则 ...
- java中线程分两种,守护线程和用户线程。
java中线程分为两种类型:用户线程和守护线程. 通过Thread.setDaemon(false)设置为用户线程: 通过Thread.setDaemon(true)设置为守护线程. 如果不设置次属性 ...
- java 多线程—— 线程让步
java 多线程 目录: Java 多线程——基础知识 Java 多线程 —— synchronized关键字 java 多线程——一个定时调度的例子 java 多线程——quartz 定时调度的例子 ...
- java 多线程—— 线程等待与唤醒
java 多线程 目录: Java 多线程——基础知识 Java 多线程 —— synchronized关键字 java 多线程——一个定时调度的例子 java 多线程——quartz 定时调度的例子 ...
- Java的线程模型
并发不一定要依赖多线程(如PHP中很常见的多进程并发),但是在Java里面谈论并发,大多数都与线程脱不开关系. 线程是比进程更轻量级的调度执行单位,线程的引入,可以把一个进程的资源分配和执行调度分开, ...
随机推荐
- Winform在控件内实现简单画笔功能
using System.Drawing; using System.Windows.Forms; namespace ZhuoHuiSchoolroom.ZhuoHuiClass { /// < ...
- wireshark使用过程中,卡死未响应
原因 : 未知 处理办法:1.重装wireshark ----------无效果 2.卸载有道词典-----------成功,wireshark正常运行,重装有道词典该问题必现,所以问题由有道词典引起 ...
- 小程序中的pick
picker:从底部弹起的滚动选择器. 属性:model string类型 说明:选择器类型 : selector 普通选择器 multiSelector 多列选择器 time 时间选择 ...
- vue vuex开发中遇到的问题及解决小技巧
1.在vue的开发中,如果使用了vuex,数据的组装,修改时在mutations中,页面是建议修改变量值的,如果强制修改,控制台就会出现错误.如下: 这种错误虽然不会影响结果,但是是vuex不提倡的方 ...
- SpringCloud之Ribbon(四)
一:Ribbon是什么? Ribbon是Netflix发布的开源项目,主要功能是提供客户端的软件负载均衡算法,将Netflix的中间层服务连接在一起.Ribbon客户端组件提供一系列完善的配置项如连接 ...
- angularjs路由菜单强制刷新
在开发过程中遇到使用路由控制单页加载页面时,点击菜单页面不重新刷新的情况,angularjs认为路由没有变化,而不会去刷新页面,解决办法: angular.module('myApp').direct ...
- (转) XSS Attacks – Exploiting XSS Filter
中文翻译: from wooyun'drops 0x00 前言 这又是一篇来自全职赏金猎人Masato kinugawa的神作.一次双杀,用一篇报告拿下了两个CVE,分别是CVE-2015-6144和 ...
- 玩转Django2.0---Django笔记建站基础四(视图)
第四章 视图 4.1 探究视图 一.视图说明 视图(View)是Django的MTV架构模式的V部分,主要负责处理用户请求和生成相应的相应部分,然后在页面或其它类型文档中显示.也可以理解为视图是MVC ...
- Java多态之Father f=new Son();
成员变量静态方法看左边,非静态方法编译看左边,运行看右边. 左边Father f其实是定义了一个Father类的对象,而右边new Son()可以只理解为是一个重写了Father类方法的对象. 因此, ...
- Xmind8 Pro 破解教程(序列号|破解文件)
最近需要打开文件后缀名为.xmind的文件,所以下载了Xmind8 .打开以后想要导出,奈何普通版本只能导出.txt文本文档,所以只好动手pj.话不多说看下边.一.下载XMindCrack.jar文件 ...