`

java学习笔记:TCP通讯,传送任意文件(服务端并发)

    博客分类:
  • java
阅读更多


 


 



 
 
 

 

 

/*
TCP通讯
需求:传送任意文件

tcp传输要定义2个端点,客户端和服务端
步骤
1.定义服务, 服务器ip和接收端口
2.读取客户端已有的图片文件
3.使用socket输出流将数据发给服务端
4.读取服务端的反馈信息
5.关闭资源

注意:
1.在while循环中的read方法最后读到文件结束标记-1后循环退出了
  而没有将该标记写入socket流中,那么服务端接收到的数据是不完整的,而且停不下来
  用shutdownOutput方法告诉服务端文件到了末尾结束发送了

[示例]:传送任意文件 (客户端)
*/

import java.net.*;
import java.io.*;
class Demo
{
  public static void main(String[] args) throws Exception
  {
    new FileClient("c:\\FoxitReader_CHS.rar"); //启动客户端,准备发送指定文件
  }
}

class FileClient  //客户端
{
  FileClient(String fileStr) throws Exception
  {
    s.op("客户端启动....");
    File file = new File(fileStr);  //关联一个文件c:\\FoxitReader_CHS.rar
    if(file.isFile())  //是一个标准文件吗?
    {
      client(file);    //启动连接
    }
    else
    {
      s.op("要发送的文件 "+fileStr+" 不是一个标准文件,请正确指定");
    }
  }
  
  public void client(File file)throws Exception
  {
    Socket sock= new Socket("192.168.1.3",10007); //指定服务端地址和端口

    FileInputStream fis = new FileInputStream(file); //读取本地文件
    OutputStream sockOut = sock.getOutputStream();   //定义socket输出流
    
    //先发送文件名.让服务端知道
    String fileName = file.getName();
    s.op("待发送文件:"+fileName);
    sockOut.write(fileName.getBytes());
    
    String serverInfo= servInfoBack(sock); //反馈的信息:服务端是否获取文件名并创建文件成功
    if(serverInfo.equals("FileSendNow"))   //服务端说已经准备接收文件,发吧
    {
      byte[] bufFile= new byte[1024];
      int len=0;
      while(true)
      {
        len=fis.read(bufFile);
        if(len!=-1)
        {
          sockOut.write(bufFile,0,len); //将从硬盘上读取的字节数据写入socket输出流
        }
        else
        {
          break;
        }
      }      
    }
    else
    {
      s.op("服务端返回信息:"+serverInfo);
    }
    sock.shutdownOutput();   //必须的,要告诉服务端该文件的数据已写完
    s.op("服务端最后一个返回信息:"+servInfoBack(sock));//显示服务端最后返回的信息
    
    fis.close();
    sock.close();
  }
  
  public String servInfoBack(Socket sock) throws Exception  //读取服务端的反馈信息
  {
    InputStream sockIn = sock.getInputStream(); //定义socket输入流
    byte[] bufIn =new byte[1024];
    int lenIn=sockIn.read(bufIn);            //将服务端返回的信息写入bufIn字节缓冲区
    String info=new String(bufIn,0,lenIn);
    return info;
  }
}

class s  
{
  public static void op(Object obj) //打印
  {
    System.out.println(obj);
  }
}

 

 

/*
服务端原先有很大的局限性
原先我们没有考虑客户端的文件名,和客户端判断是否有重名文件,我们是另外指定了文件名和路径
当客户端A连接服务端,被服务端获取到后,服务端执行具体流程,
这时如果B客户端尝试连接服务端,但只能等待
因为服务端还没有处理完A客户端的请求,不能回while起始位置执行accept方法,所以
暂时获取不到B客户端对象,那么为了可以让多个客户端同时被服务端并发接收,
那么服务端最好就是将每个客户端封装到一个个单独的线程中,这样,就可以同时处理多个客户端请求
如何定义线程呢?
只要明确每一个客户端要再服务端执行的代码即可,将该代码放到run方法中
如果服务端存在同名文件就返回信息并断开该次连接


如果要让客户端选择是否要覆盖文件,可以再加个反馈操作应该就可以了

[示例]:传送任意文件 (服务端并发执行)
*/
import java.net.*;
import java.io.*;
class Demo
{
  public static void main(String[] args) throws Exception
  {
    new FileServer();  //启动文件存储服务端
  }
}

class FileServer  //服务端
{
  FileServer() throws Exception
  { 
    s.op("服务端启动......");
    server();
  }

  public void server() throws Exception
  {
    ServerSocket serversock = new ServerSocket(10007);  //监听端口
    while(true)
    {   
      Socket sock = serversock.accept();            //循环等待客户端连接
      new Thread(new FileServThread(sock)).start(); //当成功连接客户端后开启新线程接收文件
    }
  }
}

class FileServThread implements Runnable   //服务端线程
{
  private Socket sock;
  FileServThread(Socket sock)
  {
    this.sock = sock;
  }
  
  public void run()
  {
    String ip = sock.getInetAddress().getHostAddress();   //获取客户端ip
    try
    {
      s.op("开启新线程接收来自客户端IP: "+ip+" 的文件");
      InputStream sockIn= sock.getInputStream();//定义socket输入流,接收客户端的信息
      File file =getClientFileName(sockIn);     //创建同名文件
      if(file==null)
      {
        writeOutInfo(sock,"存在同名文件或获取文件失败,服务端断开连接!");
        sock.close();              return;
      }
     
      FileOutputStream fos= new FileOutputStream(file); //用来写入硬盘
      byte[] bufFile = new byte[1024*1024];   //接收数据的缓存
      int len=0;
      while(true)
      {
        len=sockIn.read(bufFile); //接收数据
        if(len!=-1)
        {
          fos.write(bufFile,0,len); //写入硬盘文件
        }
        else
        {
          break;
        }
      }
      writeOutInfo(sock,"上传成功!");   //文件接收成功后给客户端反馈一个信息
      s.op("文件接收成功!"+System.getProperty("line.separator"));  //服务端打印一下
      fos.close();
      sock.close();     
    }
    catch(Exception ex)
    {
      throw new RuntimeException(ip+"异常!!!");
    }
  }
  
  public void writeOutInfo(Socket sock,String infoStr)throws Exception//将信息反馈给服务端
  {
    OutputStream sockOut = sock.getOutputStream();
    sockOut.write(infoStr.getBytes());
  }
  
  public File getClientFileName(InputStream sockIn) throws Exception //获取文件名并创建
  {  
    //获取客户端请求发送的文件名,并判断在D盘创建同名文件的情况
    byte[] bufName=new byte[1024];
    int lenInfo =0;
    lenInfo = sockIn.read(bufName);  //获取文件名
    String fileName = new String(bufName,0,lenInfo);
    
    File dir = new File("d:\\");    //存到D盘根目录
    File[] files=dir.listFiles();   //遍历d盘目录
    for(File f:files)
    {
      if(!f.isDirectory())  //如果遍历到的该文件不是一个目录的话
      { 
        if(f.getName().equals(fileName))  //判断是否是同名文件
        {
          s.op(f.getName()+"文件已存在,断开该ip连接."+System.getProperty("line.separator"));
          writeOutInfo(sock,"服务端已存在同名文件!"); //反馈给客户端的信息
          return null;       
        }
      }
    }
    s.op("将客户端发来的文件( "+fileName+" )存到"+dir.getAbsolutePath());     
    File file= new File(dir+fileName);  
    if(file.createNewFile())
    { 
      s.op("成功创建文件("+fileName+" )准备写入数据");
      writeOutInfo(sock,"FileSendNow");    //告诉客户端,开始传送数据吧  
      return file;
    }
    else
    {
      return null; //如果由于硬盘满了等原因创建文件失败的话
    }
  }
}

class s
{
  public static void op(Object obj) //打印
  {
    System.out.println(obj);
  }
}

 

  • 大小: 40.2 KB
  • 大小: 66 KB
  • 大小: 82.3 KB
2
0
分享到:
评论
2 楼 xouou_53320 2015-12-17  
lastSeries 写道
前辈,是不是在工作以后连System.out.println()这样的函数也得用一个类封装呀,是不是得从现在就养成这样的习惯

其实是我比较懒
1 楼 lastSeries 2015-11-23  
前辈,是不是在工作以后连System.out.println()这样的函数也得用一个类封装呀,是不是得从现在就养成这样的习惯

相关推荐

Global site tag (gtag.js) - Google Analytics