java-通讯stocket插件mina实例

mina是对nio的具体实现。是目前比较高效和流行的nio(非阻塞式I/O)框架

mina主要包括:

其中服务端为:NioSocketAcceptor
客户端为:NioSocketConnector

使用mina进行一个简单的客户端上传文件的demo的实现:

demo实现的思想为:

客 户端跟服务端建立起来连接,客户端每次想服务端传输一定大小的文件内容。(byte的方式),然后服务端接收这些byte,将其output出来,形成文 件。客户端发送完毕后,传递一个完毕的标志,这里可以传个字符串”finish“,然后服务器收到这个结束标志,在写文件结束后,再传输个成功的标志给客 户端,(字符串”success“)然后客户端关闭连接。

服务端:

代码比较简单。

import java.net.InetSocketAddress;
import org.apache.mina.filter.codec.ProtocolCodecFilter;
import org.apache.mina.filter.codec.serialization.ObjectSerializationCodecFactory;
import org.apache.mina.filter.logging.LoggingFilter;
import org.apache.mina.transport.socket.nio.NioSocketAcceptor;

public class Main {
 private static final int PORT = 8080;

 public static void main(String[] args) throws Exception {
 //服务端的实例
 NioSocketAcceptor accept=new NioSocketAcceptor();
 //添加filter,codec为序列化方式。这里为对象序列化方式,即表示传递的是对象。
 accept.getFilterChain().addLast("codec",
 new ProtocolCodecFilter(new ObjectSerializationCodecFactory()));
                //添加filter,日志信息
 accept.getFilterChain().addLast("logging", new LoggingFilter());
 //设置服务端的handler
 accept.setHandler(new FileUploadHandler());
 //绑定ip
 accept.bind(new InetSocketAddress(PORT));

        System.out.println("upload  server started.");
 }
}

服务端的handler。

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.mina.core.service.IoHandlerAdapter;
import org.apache.mina.core.session.IoSession;

public class FileUploadHandler extends IoHandlerAdapter {

 private BufferedOutputStream out;

 private int count;

 private String fileName = "D:/log/test.jpg";

 private static final Log log = LogFactory.getLog(FileUploadHandler.class);

 public void sessionOpened(IoSession session) throws Exception {
 System.out.println("server open");
 }

 public void exceptionCaught(IoSession session, Throwable cause)
 throws Exception {
 System.out.println("exception");
 session.close(true);
 super.exceptionCaught(session, cause);
 }

 public void messageReceived(IoSession session, Object message) {
 System.out.println("server received");

 try {
 if (message instanceof FileUploadRequest) {
                 //FileUploadRequest 为传递过程中使用的DO。
 FileUploadRequest request = (FileUploadRequest) message;
 System.out.println(request.getFilename());
 if (out == null) {
 //新建一个文件输入对象BufferedOutputStream,随便定义新文件的位置
 out = new BufferedOutputStream(new FileOutputStream(
 "D:/log/" + request.getFilename()));
 out.write(request.getFileContent());
 } else {
 out.write(request.getFileContent());
 }
 count += request.getFileContent().length;

 } else if (message instanceof String) {
 if (((String)message).equals("finish")) {
 System.out.println("size is"+count);
 //这里是进行文件传输后,要进行flush和close否则传递的文件不完整。
 out.flush();
 out.close();
 //回执客户端信息,上传文件成功
 session.write("success");
 }
 }

 } catch (Exception e) {
 e.printStackTrace();
 }
 }

 public void sessionClosed(IoSession session) throws Exception {
 System.out.println("server session close");
 }
}

所有的handler都要继承IoHandlerAdapter,可以查看IoHandlerAdapter,其包括几个关于session状态的方法。按需进行重载即可。

然后就是公用的用来传输的DO:FileUploadRequest简单的POJO

import java.io.Serializable;

public class FileUploadRequest implements Serializable {
 private String hostname;
 private String filename;
 private byte[] fileContent;

 public String getHostname() {
 return hostname;
 }

 public void setHostname(String hostname) {
 this.hostname = hostname;
 }

 public String getFilename() {
 return filename;
 }

 public void setFilename(String filename) {
 this.filename = filename;
 }

 public byte[] getFileContent() {
 return fileContent;
 }

 public void setFileContent(byte[] fileContent) {
 this.fileContent = fileContent;
 }
}

接下来看下客户端的实现,也很简单:

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.net.InetSocketAddress;
import nio.upload.server.FileUploadRequest;
import org.apache.mina.core.future.ConnectFuture;
import org.apache.mina.core.session.IoSession;
import org.apache.mina.filter.codec.ProtocolCodecFilter;
import org.apache.mina.filter.codec.serialization.ObjectSerializationCodecFactory;
import org.apache.mina.filter.logging.LoggingFilter;
import org.apache.mina.transport.socket.nio.NioSocketConnector;

public class MainClient {

 private static final int PORT = 8080;

 /**
  * @param args
  * @throws IOException
  */
 public static void main(String[] args) throws Exception {
 //客户端的实现
 NioSocketConnector connector = new NioSocketConnector();
 connector.getFilterChain().addLast("codec",
 new ProtocolCodecFilter(new ObjectSerializationCodecFactory()));
 connector.getFilterChain().addLast("logging", new LoggingFilter());
 FileUploadClientHandler h = new FileUploadClientHandler();
 connector.setHandler(h);
 //本句需要加上,否则无法调用下面的readFuture来从session中读取到服务端返回的信息。
 connector.getSessionConfig().setUseReadOperation(true);

 ConnectFuture cf = connector.connect(new InetSocketAddress("localhost",
 PORT));

 IoSession session;
 //等待连接成功
 cf.awaitUninterruptibly();
 session = cf.getSession();

 System.out.println("client send begin");

 //传递文件开始
 String fileName = "test.jpg";
 FileInputStream fis = new FileInputStream(new File(fileName));
 byte[] a = new byte[1024 * 4];
 FileUploadRequest request = new FileUploadRequest();
 request.setFilename(fileName);
 request.setHostname("localhost");
 while (fis.read(a, 0, a.length) != -1) {
 request.setFileContent(a);
 //像session中写入信息供服务端获得
 session.write(request);
 }
 //发送完成的标志
 session.write(new String("finish"));

 System.out.println("client send finished and wait success");
 //接上面来取得服务端的信息
 Object result = session.read().awaitUninterruptibly().getMessage();
 if (result.equals("success")) {
 System.out.println("success!");
 //关闭客户端
 connector.dispose();
 }
 }
}

客户端handler的实现。

import org.apache.mina.core.service.IoHandlerAdapter;
import org.apache.mina.core.session.IoSession;

public class FileUploadClientHandler extends IoHandlerAdapter {

 public void sessionOpened(IoSession session) throws Exception {
 System.out.println("client open");
 }

 public void sessionClosed(IoSession session) throws Exception {
 System.out.println("client session close");
 }

 public void messageReceived(IoSession session, Object message)
 throws Exception {
 System.out.println("thr result is" + message);
 }
}

原文转载

时间: 2024-08-01 06:23:39

java-通讯stocket插件mina实例的相关文章

Java程序只运行一个实例[转]

如果希望你的Java程序只能存在一个实例,可以参考下面的用法. 原文链接:http://blog.csdn.net/yaerfeng/article/details/7264729 Java没有提供这样的机制.从操作系统的观点来看,一个启动的Java Application仅仅是一个JVM的运行实例.运行相同Application的两个实例,仅仅是运行两个无关的JVM.       只有让多个运行实例之间有一个既定的通讯机制就可以保证只有一个实例运行. 方案1:使用Java的加锁文件机制,ide

Java中间件之RMI及实例介绍 · zijian‘s blog

RMI介绍 ??远程方法调用(Remote Method Invocation)是Sun公司规定的允许在不同的JAVA虚拟机之间进行对象间通信的一种规范.在RMI中,JVM可以位于一个或多个计算机上,其中一个JVM可以调用存储在另一个JVM中的对象方法.这就使得应用程序可以远程调用其他对象方法,从而达到分布式计算的目的,以共享各个系统的资源和处理能力. ??除了RMI外,基于JAVA的实现不同JAVA虚拟机上的应用程序之间通信技术主要有两种:套接字和JAVA消息服务(JMS). ??使用套接字是

Java的Swing插件的使用

为了完成java实验5,我下载了Java的Swing插件,下载安装的过程就不再赘述,这有一个教程点这里 经过摸索,我学会了插件的基本使用 1. 在Eclipse中创建一个Java Project,建立一个包,如图所示 2.如图所示,点击other选项 3.如图选择Application Window,再点next 4.进入这个页面,Source folder选择之前建好的工程的src文件夹,Package选择工程中的包,Name自己定义,点finish 5.这时在包中就会有一个叫做NewWind

java网络编程之UDP实例

package Socket; import java.net.DatagramPacket; import java.net.InetAddress; public class Dgram { public static DatagramPacket toDatagram(String s, InetAddress destIA, int destPort) { byte[] buf = new byte[s.length() + 1]; s.getBytes(0, s.length(), b

java网络编程之TCP实例

Dgram类 package Socket; import java.net.DatagramPacket; import java.net.InetAddress; public class Dgram { public static DatagramPacket toDatagram(String s, InetAddress destIA, int destPort) { byte[] buf = new byte[s.length() + 1]; s.getBytes(0, s.leng

java中定时器的应用实例

1 package com.xiangshang.listener; 2 3 import java.util.Timer; 4 import java.util.TimerTask; 5 6 import javax.servlet.ServletContextEvent; 7 import javax.servlet.ServletContextListener; 8 9 public class MyTimerListener implements ServletContextListen

Eclipse下的Java反编译插件 查看源代码不再困难

Eclipse下的Java反编译插件:Eclipse Class Decompiler,整合了目前最好的2个Java反编译工具Jad和JD-Core,并且和Eclipse Class Viewer无缝集成,能够很方便的使用本插件查看类库源码,以及采用本插件进行Debug调试. 转载自:http://bbs.csdn.net/topics/390263414 Eclipse Class Decompiler插件: http://download.csdn.net/detail/ibm_hoojo/

Gradle 1.12用户指南翻译——第46章. Java 库发布插件

文由CSDN博客貌似掉线翻译,其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Github上的地址: https://github.com/msdx/gradledoc 本文翻译所在分支: https://github.com/msdx/gradledoc/tree/1.12. 直接浏览双语版的文档请访问: http://gradledoc.qiniudn.com/1.12/usergu

Java学习-012-文件删除实例及源代码

此文源码主要为应用 Java 创建文件的源代码.若有不足之处,敬请大神指正,不胜感激! 文件删除源代码如下: 1 /** 2 * @function 文件操作:删除文件.若文件存在且未被占用,则删除文件:若文件存在且被占用,则记录文件占用信息:若文件不存在,则对应记录日志信息 3 * 4 * @author Aaron.ffp 5 * @version V1.0.0: autoUISelenium main.java.aaron.java.tools FileUtils.java deleteF