服务器端发送文件:
package test;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
public class TCPServer {
static int size=10*1024;
static byte[] buf=new byte[size];
static int len=-1;
public static void main(String[] args) throws Exception{
ServerSocket ss=null;
OutputStream out=null;
Socket s=null;
try {
ss=new ServerSocket(1024);
System.out.println("等待客户端连接");
s=ss.accept();
System.out.println("连接成功");
// InputStream in=s.getInputStream();
out=s.getOutputStream();
long start=System.currentTimeMillis();
System.out.println("开始发送文件");
String fileName=args[0];//"I:"+File.separator;
if(!fileName.endsWith(File.separator)){
fileName+=File.separator;
}
File file=new File(fileName);
String parent=file.getParent();
if(parent==null){
File[] fs=file.listFiles();
for(int i=0;i<fs.length;i++){
if(fs[i].isHidden())
{
System.out.println("隐藏文件"+fs[i].getAbsolutePath()+",不会被发送");
continue;
}
test(fs[i],out,fs[i].getParent());
}
}else{
test(file,out,parent);
}
System.out.println("文件发送成功"+(System.currentTimeMillis()-start)+"ms");
// out.write("接受成功".getBytes());
out.flush();
} finally {
if(out!=null)out.close();
if(s!=null)s.close();
if(ss!=null)ss.close();
}
}
private static void test(File file,OutputStream out,String sendFileName)throws Exception{
FileInputStream in=null;
String fname=file.getAbsolutePath();
String name=fname.replace(sendFileName, "");
if(file.isDirectory()){
File[] fs=file.listFiles();
out.write(new byte[]{(byte)2}, 0, 1);//2:文件夹名
int fileLength=name.getBytes().length;
out.write(intToBytes(fileLength),0,4);//文件名的长度
out.write(name.getBytes(),0,name.getBytes().length);//文件名
System.out.println("文件夹:"+name+" "+name.length());
out.flush();
for(int i=0;i<fs.length;i++){
if(fs[i].isHidden())
{
System.out.println("隐藏文件"+fs[i].getAbsolutePath()+",不会被发送");
continue;
}
test(fs[i],out,sendFileName);
}
}else{
out.write(new byte[]{(byte)1}, 0, 1);//1:文件名
int fileLength=name.getBytes().length;
out.write(intToBytes(fileLength),0,4);//文件夹名的长度
out.write(name.getBytes(),0,name.getBytes().length);//文件夹名
System.out.println("文件:"+name+" "+name.length()+" "+file.length());
out.flush();
in=new FileInputStream(file);
out.write(new byte[]{(byte)0}, 0, 1);//0表示文件数据
out.write(longToBytes(file.length()),0,8);//文件的长度
out.flush();
while((len=in.read(buf,0,size))!=-1){
out.write(buf,0,len);
out.flush();
}
in.close();
}
}
private static byte[] intToBytes(int i){
byte[] b=new byte[4];
b[0]=(byte)((i>>>24)&255);
b[1]=(byte)((i>>>16)&255);
b[2]=(byte)((i>>>8)&255);
b[3]=(byte)(i&255);
return b;
}
private static byte[] longToBytes(long i){
byte[] b=new byte[8];
b[0]=(byte)((i>>>56)&255);
b[1]=(byte)((i>>>48)&255);
b[2]=(byte)((i>>>40)&255);
b[3]=(byte)((i>>>32)&255);
b[4]=(byte)((i>>>24)&255);
b[5]=(byte)((i>>>16)&255);
b[6]=(byte)((i>>>8)&255);
b[7]=(byte)(i&255);
return b;
}
}
客户端代码:
package test;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.PrintStream;
import java.net.Socket;
public class TCPClient {
public static void main(String[] args) throws Exception{
System.setOut(new PrintStream(new FileOutputStream("D:\\log.txt")));
Socket s=null;
InputStream in=null;
try {
s=new Socket("192.168.0.122", 1024);
String savePath=args[0];//"D:"+File.separator+"test"+File.separator;
in=s.getInputStream();
int size=10*1024;
byte[] buf=new byte[size];
int len=-1;
byte[] bs=new byte[size];
while((len=in.read(buf, 0, size))!=-1){
writeData(buf,0,len,savePath,bs);
}
if(out!=null){
out.flush();
out.close();
}
} finally {
if(in!=null)
in.close();
if(s!=null)
s.close();
}
}
static FileOutputStream out=null;
static long count=0;//计算是否完成数据的读取,开始下一条命令
static int cmd=-1;
static int bsl=0;
private static void writeData(byte[] buf,int off, int len,String savePath,byte[] bs) throws Exception{
if(len-off==0)return;
System.out.println("偏移量:"+off+"命令"+cmd+"数量:"+count);
int i=off;
if(count==0l){//如果一条命令的数据已经读完就开始读取下一条命令
cmd=buf[i++];
System.out.println("获取命令:"+cmd);
count=-1l;
if(len-i==0)return;
writeData(buf,i,len,savePath,bs);
}else if(count==-1l){//读取文件(夹)名称的长度或文件的大小
System.out.println("获取长度");
switch (cmd){
case 0:
if(len-i+bsl<8){
System.arraycopy(buf, i, bs, bsl, len-i);
System.out.println("读取长度1:"+(len-i)+" 未读取完");
bsl=len-i;
i=0;
return;
}
System.arraycopy(buf, i, bs, bsl, 8-bsl);
System.out.println("读取长度1:"+(8-bsl)+" 读取完");
count=bytesToLong(bs, 0);
i+=8-bsl;
bsl=0;
writeData(buf,i,len,savePath,bs);
break;
case 1:
case 2:
if(len-i+bsl<4){
System.arraycopy(buf, i, bs, bsl, len-i);
System.out.println("读取长度2:"+(len-i)+" 未读取完");
bsl=len-i;
i=0;
return;
}
System.arraycopy(buf, i, bs, bsl, 4-bsl);
System.out.println("读取长度2:"+(4-bsl)+" 读取完");
count=bytesToInt(bs, 0);
i+=4-bsl;
bsl=0;
writeData(buf,i,len,savePath,bs);
break;
}
}else{//写入文件或创建文件夹、创建文件输出流
System.out.println("3");
switch (cmd){
case 0:
System.out.println("写入文件");
if(len-i-count>0){
try{
System.out.println("写入文件 长度:"+count+"文件写入完成");
out.write(buf, i, (int)count);
i+=count;
count=0;
out.flush();
}finally{
if(out!=null)out.close();
}
writeData(buf,i,len,savePath,bs);
}else{
System.out.println("写入文件 长度:"+(len-i)+"文件写入没有完成");
out.write(buf,i,len-i);
count-=len-i;
i=0;
}break;
case 1:
if(len-i-count<0){
System.out.println("获取文件名字:"+(len-i)+"写入没有完成 剩余长度"+count);
System.arraycopy(buf, i, bs, bsl, len-i);
bsl+=len-i;
count-=bsl;
i=0;
return;
}else{
System.out.println("获取文件名字:"+(count-bsl)+"写入完成 剩余长度");
System.arraycopy(buf, i, bs, bsl, (int)count);
String name=new String(bs,0,(int)count+bsl);
System.out.println("文件:"+savePath+name);
out=new FileOutputStream(savePath+name);
bsl=0;
i+=count;
count=0;
writeData(buf,i,len,savePath,bs);
}
break;
case 2:
if(len-i-count<0){
System.out.println("获取文件夹名字:"+(len-i)+"写入没有完成 剩余长度"+count);
System.arraycopy(buf, i, bs, bsl, len-i);
bsl+=len-i;
count-=bsl;
i=0;
return;
}else{
System.out.println(len+" "+count+" "+bsl+" ");
System.out.println("获取文件夹名字:"+(count-bsl)+"写入完成 剩余长度");
System.arraycopy(buf, i, bs, bsl, (int)count);
String name=new String(bs,0,bsl+(int)count);
File file=new File(savePath+name);
bsl=0;
i+=count;
count=0;
if(!file.exists()){
file.mkdirs();
}
System.out.println("文件夹:"+savePath+name);
writeData(buf,i,len,savePath,bs);
}
break;
}
}
}
private static int bytesToInt(byte[] buf,int off){
int i=0;
i=i|((buf[off]&255)<<24);
i=i|((buf[off+1]&255)<<16);
i=i|((buf[off+2]&255)<<8);
i=i|(buf[off+3]&255);
return i;
}
private static long bytesToLong(byte[] buf,int off){
long i=0;
i=i|(((long)buf[off]&255)<<56)
|(((long)buf[off+1]&255)<<48)
|(((long)buf[off+2]&255)<<40)
|(((long)buf[off+3]&255)<<32)
|(((long)buf[off+4]&255)<<24)
|(((long)buf[off+5]&255)<<16)
|(((long)buf[off+6]&255)<<8)
|((long)buf[off+7]&255);
return i;
}
}
分享到:
相关推荐
需要注意的是,确保在发送和接收端保持一致的字符编码,以避免乱码问题。 5. **实现细节**: - **服务器端**:创建ServerSocket监听特定端口,等待客户端连接。接收到连接后,通过Socket的输入流接收文件数据,并...
在这个场景中,我们讨论的是如何使用Java的Socket来实现文件上传功能,即从客户端将文件发送到服务器,然后保存到服务器的数据库中。这个过程涉及到多个关键知识点,下面我们将详细探讨。 1. **Java Socket基础**:...
本篇文章将详细讲解如何利用Java的Socket和多线程技术来实现整个文件夹的传输。 首先,理解Socket通信的基本原理。Socket是网络通信中的端点,它允许两个程序通过网络连接进行数据交换。在Java中,`java.net.Socket...
在`Socket-master`这个文件夹中,可能包含了一个Java Socket编程的示例项目,用于演示如何在Nginx TCP转发场景下获取用户真实IP。该项目可能包括了服务器端和客户端的代码,通过分析和运行这些代码,我们可以更深入...
Java局域网聊天室和文件夹发送是一款基于UDP和TCP协议开发的应用,它允许用户在同一个局域网内的设备之间进行实时的文本聊天和文件交换。这个应用的主要特点包括群聊、私聊功能以及能够发送单个文件或整个文件夹。在...
源代码分析可以帮助我们深入理解客户端如何建立连接,如何发送和接收数据,以及如何解析邮件格式。这对于学习网络编程、邮件协议和Java Socket编程都是宝贵的学习材料。同时,也可以通过源代码学习到如何实现安全的...
Java基于Socket实现局域网文件传输,此文件包含了Eclipse项目源码和已经打包好了的.jar文件(文件发送端和文件接收端)。 处在同一局域网内的两台主机,一台运行sender.jar文件(发送端),另一台运行receiver.jar...
文件传输是计算机网络中一个常见的功能,它允许用户在网络中的不同设备之间发送和接收文件。Java作为一种广泛使用的编程语言,提供了多种方式来实现文件传输。其中一种常用的方法是通过Socket编程进行文件夹的传输。...
在项目中,"wuziqi"这个文件夹很可能包含了所有源代码,包括服务器端和客户端的Java类。这些类可能包括棋盘逻辑处理、网络通信、用户界面以及游戏规则的实现。阅读和理解这些源码,可以帮助深入学习Java Socket网络...
在Java中,`java.net.Socket`类和`java.net.ServerSocket`类是实现Socket编程的基础。 2. **TCP/IP协议**:Java Socket基于TCP/IP协议栈工作,TCP(传输控制协议)提供面向连接、可靠的数据传输服务。在这个P2P聊天...
在这个示例中,我们将探讨如何使用Java的Socket API来实现在服务器和客户端之间传输文件。文件传输是一个常见的需求,特别是在分布式系统或者网络应用中,理解如何通过Socket实现文件传输对于Java开发者来说至关重要...
在给定的`javaSocket` 文件夹中,可能包含了一个简单的Java Socket客户端和服务器端的源代码示例。通过运行这些示例,你可以直观地理解如何在实际项目中使用Java Socket进行网络通信。记住,实际应用中可能需要更...
Java Socket AMF3 Flash Game Framework 是一个专门为Flash游戏设计的后端框架,它结合了Java技术和AMF3编码,使得游戏开发者能够构建交互性强、安全可靠的Web游戏。在这个框架中,Java作为服务器端语言,提供了稳定...
- `client`:可能包含了客户端的源代码,包括Socket连接创建、心跳发送和接收的逻辑。 - `messages`:可能存储了用于心跳检测的协议定义和数据包封装的类。 - `test`:包含单元测试或集成测试代码,用于验证心跳...
此外,客户端也可能使用多线程,例如,一个线程用于接收服务器发送的试卷,另一个线程用于处理用户输入和保存作答记录。 3. **随机生成试卷**: 试卷的随机生成是系统的重要组成部分,可能涉及到数据结构和算法的...
- "MyChat6"可能是这个聊天室项目的源代码文件夹,包含服务器端和客户端的Java源码,可能还包括配置文件、测试数据等。 - 源码可能包括Server类、Client类以及可能的Util类,用于处理网络通信、线程管理和数据解析...
- "room"可能是项目文件夹的名字,包含了服务器端和客户端的Java源代码,以及可能的配置文件或其他资源。 - 学习案例时,可以查看这些源代码,了解具体的实现细节。 总的来说,这个"java中socket在线编程聊天系统...
`Socket`类在`java.net`包下,提供了建立连接、发送数据和接收数据的方法。 在多线程环境中,每个Socket连接通常由一个独立的线程来处理,这样可以同时处理多个客户端的请求,提高系统的并发性。这里,"Server...
- **读写数据**:服务端通常使用`Socket`对象的`getInputStream()`和`getOutputStream()`方法获取输入流和输出流,以发送和接收数据。 - **文件传输**:在这个例子中,服务端可能实现了将文件夹内容分解为单个文件...