NIO实现TCP文件传输

最近这段时间学习了一下NIO,就想把FileChannel和SocketChannel方面的知识结合一下,于是就练习了这个基于NIO的TCP文件传输的例程。主要使用了SocketChannel,ServerSocketChannel,Selector,FileChannel等实现。当服务器端和客户端都启动后,服务端接受客户端的连接,然后通过TCP将本地的一个文件发送至客户端,客户端接收到可读事件后,读取该文件,并存储,完成所有工作。

以下是服务器端代码:

/**  
* Title: Server.java  
* Description: 
* @author:wh
* @date 2019年7月9日
* @version 1.0
* Company: itiis
*/  

import java.io.FileInputStream;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Iterator;
/**
*@class_name:Server
*@comments:nio传输文件(发送)
*@param:
*@return: 
*@author:wh
*@createtime:2019年7月9日
*/
public class Server {
 //通道管理器
 private Selector selector;
 
 //获取一个ServerSocket通道,并初始化通道
 public Server init(int port) throws IOException{
     //1.获取一个ServerSocket通道
     ServerSocketChannel serverChannel = ServerSocketChannel.open();
     System.out.println(serverChannel.isBlocking());
     serverChannel.configureBlocking(false);////设置为非阻塞
     System.out.println(serverChannel.isBlocking());
     //2.绑定监听,配置TCP参数,例如backlog大小
     serverChannel.socket().bind(new InetSocketAddress(port));
     //3.获取通道管理器
     selector=Selector.open();
     //将通道管理器与通道绑定,并为该通道注册SelectionKey.OP_ACCEPT事件,
     //只有当该事件到达时,Selector.select()会返回,否则一直阻塞。
     serverChannel.register(selector, SelectionKey.OP_ACCEPT);//注册channel到selector,监测接受此通道套接字的连接
     return this;
 }
 public void listen() throws IOException{
     System.out.println("服务器启动成功");
     boolean isRun = true;
     while(isRun){
          //当有注册的事件到达时,方法返回,否则阻塞。
         selector.select();
         //获取selector中的迭代器,选中项为注册的事件
         Iterator<SelectionKey> ite=selector.selectedKeys().iterator();
         while(ite.hasNext()){
             SelectionKey key = ite.next();
             //删除已选的key,防止重复处理
             ite.remove();
             if(key.isAcceptable()){
                 ServerSocketChannel server = (ServerSocketChannel)key.channel();
                 //获得客户端连接通道
                 SocketChannel channel = server.accept();
                 channel.configureBlocking(false);//可以在任意位置调用这个方法,新的阻塞模式只会影响下面的i/o操作
                 //在与客户端连接成功后,为客户端通道注册SelectionKey.OP_WRITE事件。
                 channel.register(selector, SelectionKey.OP_WRITE);
                 System.out.println("客户端请求连接事件");
             }else if(key.isReadable()){
                 
             }else if(key.isWritable()){
                 SocketChannel socketChannel = (SocketChannel)key.channel();
                 FileInputStream file = new FileInputStream("C:\\Users\\WH\\Desktop\\实验室各种账号密码.txt");
                 FileChannel fileChannel = file.getChannel();
                 //500M  堆外内存
                 ByteBuffer byteBuffer = ByteBuffer.allocateDirect(524288000);
                 while(fileChannel.position()<fileChannel.size()){
                     fileChannel.read(byteBuffer);//从文件通道读取到byteBuffer
                     byteBuffer.flip();
                     while(byteBuffer.hasRemaining()){
                         socketChannel.write(byteBuffer);//写入通道
                     }
                     byteBuffer.clear();//清理byteBuffer
                     System.out.println(fileChannel.position()+" "+fileChannel.size());
                 }
                 System.out.println("结束写操作");
                 socketChannel.close();
             }
         }
     }
 }
 public static void main(String[] args) throws IOException {
     new Server().init(9981).listen();
 }

}

以下是客户端代码:

/**  
* Title: Client.java  
* Description: 
* @author:wh
* @date 2019年7月9日
* @version 1.0
* Company: itiis
*/  
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.SocketChannel;
import java.util.Iterator;



/**
 *@class_name:Client
 *@comments:nio传输文件客户端(接收)
 *@param:
 *@return: 
 *@author:wh
 *@createtime:2019年7月9日
 */
public class Client {
    //管道管理器
    private Selector selector;
    
    public Client init(String serverIp, int port) throws IOException, InterruptedException{
        //获取socket通道
        SocketChannel channel = SocketChannel.open();
        
        channel.configureBlocking(false);
        //获得通道管理器
        selector=Selector.open();
        
        //客户端连接服务器,需要调用channel.finishConnect();才能实际完成连接。
        boolean connectResult = channel.connect(new InetSocketAddress(serverIp, port));

        //为该通道注册SelectionKey.OP_CONNECT事件
        channel.register(selector, SelectionKey.OP_CONNECT);
        return this;
    }
    
    public void listen() throws IOException{
        System.out.println("客户端启动");
        //轮询访问selector
        while(true){
            //选择注册过的io操作的事件(第一次为SelectionKey.OP_CONNECT)
            selector.select();
            Iterator<SelectionKey> ite = selector.selectedKeys().iterator();
            while(ite.hasNext()){
                SelectionKey key = ite.next();
                //删除已选的key,防止重复处理
                ite.remove();
                if(key.isConnectable()){
                    SocketChannel channel=(SocketChannel)key.channel();
                    //如果正在连接,则完成连接
                    if(channel.isConnectionPending()){
                        channel.finishConnect();
                    }
                    channel.configureBlocking(false);
                    //向服务器发送消息
                    channel.write(ByteBuffer.wrap(new String("send message to server.").getBytes()));
                    //连接成功后,注册接收服务器消息的事件
                    channel.register(selector, SelectionKey.OP_READ);//订阅读取事件
                    System.out.println("客户端连接成功");
                }else if(key.isReadable()){ //有可读数据事件。
                    SocketChannel channel = (SocketChannel)key.channel();
                    ByteBuffer byteBuffer = ByteBuffer.allocate(10);
                    int readLength = channel.read(byteBuffer);
                    byteBuffer.flip();
                    int count = 0;
                    File file = new File("C:\\Users\\WH\\Desktop\\实验室各种账号密码_copy.txt");
                    if(!file.exists()) file.createNewFile();
                    FileOutputStream fe =new FileOutputStream(file,true);//可追加写
                    FileChannel outFileChannel = fe.getChannel();
                    while(readLength != -1){ //分多次读取
                        count = count+readLength;
                        System.out.println("count="+count+" readLength="+readLength);
                        readLength = channel.read(byteBuffer);//将socketChannel数据读到byteBuffer
                        byteBuffer.flip();
                        outFileChannel.write(byteBuffer);//byteBuffer数据写到FileChannel
                        fe.flush();
                        byteBuffer.clear();
                    }
                    outFileChannel.close();
                    
                    fe.close();
                    System.out.println("读取结束");
                    channel.close();
                }
            }
        }
    }
    
    public static void main(String[] args) throws IOException, InterruptedException {
        new Client().init("127.0.0.1", 9981).listen();
    }

}

运行服务器和客户端,可以看到桌面出现一个“实验室各种账号密码_copy.txt”文件,打开后和原始文件一样。

客户端运行结果.png

使用了Selector的NIO可以使用一个线程来对多个通道中就绪通道进行选择,大大节省了CPU资源,省去了多个线程来回切换资源浪费。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容

  • Java NIO(New IO)是从Java 1.4版本开始引入的一个新的IO API,可以替代标准的Java I...
    JackChen1024阅读 7,554评论 1 143
  • JAVA NIO基础 ...
    文思li阅读 613评论 0 3
  • 前言: 之前的文章《Java文件IO常用归纳》主要写了Java 标准IO要注意的细节和技巧,由于网上各种学习途径,...
    androidjp阅读 2,900评论 0 22
  • 通道(Channel)是java.nio的第二个主要创新。它们既不是一个扩展也不是一项增强,而是全新、极好的Jav...
    桥头放牛娃阅读 3,051评论 0 9
  • # Java NIO # Java NIO属于非阻塞IO,这是与传统IO最本质的区别。传统IO包括socket和文...
    Teddy_b阅读 586评论 0 0