java模擬客戶端向服務器上傳文件
本文實例為大家分享了java客戶端向服務器上傳文件的具體代碼,供大家參考,具體內容如下
先來了解一下客戶端與服務器Tcp通信的基本步驟:
- 服務器端先啟動,然后啟動客戶端向服務器端發(fā)送數(shù)據(jù)。
- 服務器端收到客戶端發(fā)送的數(shù)據(jù),服務器端會響應應客戶端,向客戶端發(fā)送響應結果。
- 客戶端讀取服務器發(fā)送的數(shù)據(jù)
文件上傳步驟:
客戶端使用本地字節(jié)輸入流,指定上傳數(shù)據(jù)的數(shù)據(jù)源。
客戶端使用網(wǎng)絡字節(jié)輸出流,把讀取的本地文件上傳到服務器。
服務器使用網(wǎng)絡字節(jié)輸入流,讀取客戶端上傳的文件。
服務器使用本地字節(jié)輸出流,把讀取到的文件保存到服務器硬盤上。
服務器使用網(wǎng)絡字節(jié)輸出流,給客戶端響應一個“上傳成功”。
客戶端使用網(wǎng)絡字節(jié)輸入流,讀取服務器響應的數(shù)據(jù)。
客戶端的代碼實現(xiàn)
public class fileClient {
public static void main(String[] args) throws IOException {
FileInputStream fis = new FileInputStream("D:\\1.jpg");//創(chuàng)建一個本地的輸入流,用于指定上傳數(shù)據(jù)的數(shù)據(jù)源
Socket socket = new Socket("127.0.0.1",8888);//創(chuàng)建一個客戶端對象,host是服務器名稱或Ip地址
OutputStream os = socket.getOutputStream();//使用socket中的方法,獲取網(wǎng)絡字節(jié)輸出流對象
byte[] bytes = new byte[1024];//把本地硬盤的數(shù)據(jù)通過網(wǎng)絡字節(jié)輸出流傳遞給客戶端
int len = 0;
while ((len = fis.read(bytes))!=-1){
os.write(bytes,0,len);
}
socket.shutdownOutput();//為了解決阻塞問題
InputStream is = socket.getInputStream();//使用socket中的方法,獲取網(wǎng)絡字節(jié)輸入流,用于讀取客戶端的數(shù)據(jù)
while((len = is.read(bytes))!=-1){//讀取客戶端的數(shù)據(jù)進行輸出
System.out.println(new String(bytes,0,len));
}
socket.close();//關閉流
fis.close();
}
}
服務器端代碼實現(xiàn)
public class fileServer {
public static void main(String[] args) throws IOException {
ServerSocket serverSocket = new ServerSocket(8888);//創(chuàng)建服務器端對象
//while(true){//讓服務器一直處于運行狀態(tài),保證客戶端可以隨時向服務器上傳文件
new Thread(new Runnable() {//為了提高文件上傳效率,來一個客戶端開一個線程
@Override
public void run() {
try{
Socket socket = serverSocket.accept();//使用accept方法接收客戶端的數(shù)據(jù)
InputStream is = socket.getInputStream();//創(chuàng)建網(wǎng)絡字節(jié)輸入流
File file = new File("D:\\upload");//判斷服務器硬盤中的文件夾是否存在,此文件夾用于存儲客戶端上傳的內容
if (!file.exists()){//判斷服務器端的文件夾是否存在
file.mkdirs();
}
String fileName = "\\picture"+System.currentTimeMillis()+new Random().nextInt(99999)+".jpg";//自己隨機生成文件名,防止重復
FileOutputStream fos = new FileOutputStream(file+fileName);//
int len;
byte[] bytes = new byte[1024];
while((len = is.read(bytes))!= -1 ){//將客戶端的數(shù)據(jù)寫入服務器硬盤中
fos.write(bytes,0,len);
}
OutputStream os = socket.getOutputStream();//向客戶端回應
os.write("上傳成功".getBytes());
socket.close();
fos.close();
}catch (IOException e){//在這里只能使用try catch解決異常,因為run方法不支持拋出異常
System.out.println(e);
}
}
}).start();
//}
}
}
注:在開啟多線程的時候,因為run方法的父類并不支持throws解決異常,所以run也不支持,只能使用try…catch解決異常
解釋一下為什么在上傳數(shù)據(jù)的時候客戶端和服務器會出現(xiàn)阻塞問題以及解決阻塞的方法
為什么會出現(xiàn)阻塞?
出現(xiàn)阻塞的根本問題是,客戶端從本地硬盤讀取文件給服務器的時候,因為用的是while循環(huán),所以文件的結束符-1并沒有被讀取到服務器,這時上傳到服務器的文件就沒有結束符,服務器把讀取到的文件保存到服務器硬盤上時就會一直執(zhí)行while循環(huán),導致阻塞。另外還有一個阻塞時服務器向客戶端響應文件時候的阻塞。
解決阻塞問題的方法?
阻塞問題的根本原因就是因為在讀取的時候沒有結束符,用 socket.shutdownOutput() 給上傳的文件一個中止序列。
API對 shutdownOutput() 的解釋:對于 TCP 套接字,任何以前寫入的數(shù)據(jù)都將被發(fā)送,并且后跟 TCP的正常連接終止序列。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
- Vue+Java 通過websocket實現(xiàn)服務器與客戶端雙向通信操作
- Java利用TCP協(xié)議實現(xiàn)客戶端與服務器通信(附通信源碼)
- Java實現(xiàn)UDP通信過程實例分析【服務器端與客戶端】
- java UDP通信客戶端與服務器端實例分析
- java Tcp通信客戶端與服務器端實例
- java模擬TCP通信實現(xiàn)客戶端上傳文件到服務器端
- java實現(xiàn)客戶端向服務器發(fā)送文件
- Java實現(xiàn)文件上傳服務器和客戶端
- java實現(xiàn)上傳文件到服務器和客戶端
- Java Socket編程服務器響應客戶端實例代碼
- Java thrift服務器和客戶端創(chuàng)建實例代碼
- java多線程實現(xiàn)服務器端與多客戶端之間的通信
- 教你怎么用java實現(xiàn)客戶端與服務器一問一答
相關文章
springboot2中HikariCP連接池的相關配置問題
這篇文章主要介紹了springboot2中HikariCP連接池的相關配置問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-12-12
SpringBoot+MyBatis-Plus實現(xiàn)分頁示例
本文介紹了SpringBoot+MyBatis-Plus實現(xiàn)分頁示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2024-12-12

