婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > golang socket斷點續傳大文件的實現方法

golang socket斷點續傳大文件的實現方法

熱門標簽:中國地圖標注省會高清 西部云谷一期地圖標注 高德地圖標注口訣 學海導航地圖標注 地圖標注的汽車標 南通如皋申請開通400電話 江西轉化率高的羿智云外呼系統 浙江高速公路地圖標注 廣州呼叫中心外呼系統

在日常編程中,我們肯定會遇到用socket傳送文件內容,如果是大文件的,總不能傳送到一半因某原因斷掉了,又從新傳送文件內容吧。對,我們需要續傳,也就是接著上次傳送的位置繼續發送文件內容。

續傳的話,其實并不難,我理解的思路大概如下:

客戶端發送消息詢問服務端,你上次接收到的文件內容位置

服務端告訴客戶端上次接收到的文件內容位置

客戶端就從上次斷點的位置繼續發送文件內容

客戶端發送文件內容完畢后通知服務端,然后斷開連接

下面我們看看代碼的實現

服務端

// file name: server.go

package main

import (
 "os"
 "io"
 "net"
 "log"
 "strconv"
 // "time"
)

// 把接收到的內容append到文件
func writeFile(content []byte) {
 if len(content) != 0 {
  fp, err := os.OpenFile("test_1.txt", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0755)
  defer fp.Close()
  if err != nil {
   log.Fatalf("open file faild: %s\n", err)
  }
  _, err = fp.Write(content)
  if err != nil {
   log.Fatalf("append content to file faild: %s\n", err)
  }
  log.Printf("append content: 【%s】 success\n", string(content))
 }
}

// 獲取已接收內容的大小
// (斷點續傳需要把已接收內容大下通知客戶端從哪里開始發送文件內容)
func getFileStat() int64 {
 fileinfo, err := os.Stat("test_1.txt")
 if err != nil {
  // 如果首次沒有創建test_1.txt文件,則直接返回0
  // 告訴客戶端從頭開始發送文件內容
  if os.IsNotExist(err) {
   log.Printf("file size: %d\n", 0)
   return int64(0)
  }
  log.Fatalf("get file stat faild: %s\n", err)
 }
 log.Printf("file size: %d\n", fileinfo.Size())
 return fileinfo.Size() 
}

func serverConn(conn net.Conn) {
 defer conn.Close()
 for {
  var buf = make([]byte, 10)
  n, err := conn.Read(buf)
  if err != nil {
   if err == io.EOF {
    log.Println("server io EOF\n")
    return
   }
   log.Fatalf("server read faild: %s\n", err)
  }
  log.Printf("recevice %d bytes, content is 【%s】\n", n, string(buf[:n]))
  // 判斷客戶端發送過來的消息
  // 如果是'start-->‘則表示需要告訴客戶端從哪里開始讀取文件數據發送
  switch string(buf[:n]) {
  case "start-->":
   off := getFileStat()
   // int conver string
   stringoff := strconv.FormatInt(off, 10)
   _, err = conn.Write([]byte(stringoff))
   if err != nil {
    log.Fatalf("server write faild: %s\n", err)
   }
   continue
  case "--end":
   // 如果接收到客戶端通知所有文件內容發送完畢消息則退出
   log.Fatalf("receive over\n")
   return
  // default:
  //  time.Sleep(time.Second * 1)
  }
  // 把客戶端發送的內容保存到文件
  writeFile(buf[:n])
 }
}

func main() {
 // 建立監聽
 l, err := net.Listen("tcp", ":8888")
 if err != nil {
  log.Fatalf("error listen: %s\n", err)
 }
 defer l.Close()

 log.Println("waiting accept.")
 // 允許客戶端連接,在沒有客戶端連接時,會一直阻塞
 conn, err := l.Accept()
 if err != nil {
  log.Fatalf("accept faild: %s\n", err)
 }
 serverConn(conn)
}

客戶端

// file name: client.go

package main

import (
 "os"
 "io"
 "net"
 "log"
 "time"
 "strconv"
)

// 獲取服務端發送的消息
func clientRead(conn net.Conn) int {
 buf := make([]byte, 5)
 n, err := conn.Read(buf)
 if err != nil {
  log.Fatalf("receive server info faild: %s\n", err)
 }
 // string conver int
 off, err := strconv.Atoi(string(buf[:n]))
 if err != nil {
  log.Fatalf("string conver int faild: %s\n", err)
 }
 return off
}

// 發送消息到服務端
func clientWrite(conn net.Conn, data []byte) {
 _, err := conn.Write(data)
 if err != nil {
  log.Fatalf("send 【%s】 content faild: %s\n", string(data), err)
 }
 log.Printf("send 【%s】 content success\n", string(data))
}

// client conn
func clientConn(conn net.Conn) {
 defer conn.Close()

 // 發送"start-->"消息通知服務端,我要開始發送文件內容了
 // 你趕緊告訴我你那邊已經接收了多少內容,我從你已經接收的內容處開始繼續發送
 clientWrite(conn, []byte("start-->"))
 off := clientRead(conn)

 // send file content
 fp, err := os.OpenFile("test.txt", os.O_RDONLY, 0755)
 if err != nil {
  log.Fatalf("open file faild: %s\n", err)
 }
 defer fp.Close()

 // set file seek
 // 設置從哪里開始讀取文件內容
 _, err = fp.Seek(int64(off), 0)
 if err != nil {
  log.Fatalf("set file seek faild: %s\n", err)
 }
 log.Printf("read file at seek: %d\n", off)

 for {
  // 每次發送10個字節大小的內容
  data := make([]byte, 10)
  n, err := fp.Read(data)
  if err != nil {
   if err == io.EOF {
    // 如果已經讀取完文件內容
    // 就發送'--end'消息通知服務端,文件內容發送完了
    time.Sleep(time.Second * 1)
    clientWrite(conn, []byte("--end"))
    log.Println("send all content, now quit")
    break
   }
   log.Fatalf("read file err: %s\n", err)
  }
  // 發送文件內容到服務端
  clientWrite(conn, data[:n])
 }
}

func main() {
 // connect timeout 10s
 conn, err := net.DialTimeout("tcp", ":8888", time.Second * 10)
 if err != nil {
  log.Fatalf("client dial faild: %s\n", err)
 }
 clientConn(conn)
 }

客戶端讀取文件test.txt內容發送到服務端,服務端把接收到的文件內容保存在test_1.txt文件中。我們模擬斷點續傳的方式是:

第一次先發送test.txt文件內容到服務端

修改test.txt文件,加一些內容

再次運行server socket以及client socket,觀察客戶端是不是只發送新增的文件內容到服務端

# 假設我的test.txt文件有以下內容
$ cat test.txt
hello golang.

# 先運行server socket再運行client socket(分別在兩個終端窗口運行)
$ go run server.go
$ go run client.go

# 服務端會輸出以下內容
2018/04/05 23:37:13 waiting accept.
2018/04/05 23:37:15 recevice 8 bytes, content is 【start-->】
2018/04/05 23:37:15 file size: 0
2018/04/05 23:37:15 recevice 10 bytes, content is 【hello gola】
2018/04/05 23:37:15 append content: 【hello gola】 success
2018/04/05 23:37:15 recevice 2 bytes, content is 【n.】
2018/04/05 23:37:15 append content: 【n.】 success
2018/04/05 23:37:16 recevice 6 bytes, content is 【--end】
2018/04/05 23:37:16 receive over
exit status 1

# 客戶端會輸出如下內容
2018/04/05 23:37:15 send 【start-->】 content success
2018/04/05 23:37:15 read file at seek: 0
2018/04/05 23:37:15 send 【hello gola】 content success
2018/04/05 23:37:15 send 【n.】 content success
2018/04/05 23:37:16 send 【--end】 content success
2018/04/05 23:37:16 send all content, now quit

# 這時候我們看看test_1.txt內容跟test.txt完全一致
$ cat test_1.txt
hello golan.

# ------- 模擬斷點續傳 ----------
# 現在我們往test.txt追加內容: hello python.
$ cat test.txt
hello golang.
hello python.

# 我們再一次運行server socket 和 client socket(分別在兩個終端窗口運行)
$ go run server.go
$ go run client.go

# 服務端會輸出以下內容
2018/04/05 23:44:31 waiting accept.
2018/04/05 23:44:34 recevice 8 bytes, content is 【start-->】
2018/04/05 23:44:34 file size: 12
2018/04/05 23:44:34 recevice 10 bytes, content is 【
hello pyt】
2018/04/05 23:44:34 append content: 【
hello pyt】 success
2018/04/05 23:44:34 recevice 4 bytes, content is 【hon.】
2018/04/05 23:44:34 append content: 【hon.】 success
2018/04/05 23:44:35 recevice 6 bytes, content is 【--end】
2018/04/05 23:44:35 receive over
exit status 1
# 服務端在接收到客戶端發送的 start--> 信息后會獲取上次接收到文件內容位置,并通知客戶端(這里file size 是12)

# 客戶端會輸出以下內容
2018/04/05 23:44:34 send 【start-->】 content success
2018/04/05 23:44:34 read file at seek: 12
2018/04/05 23:44:34 send 【
hello pyt】 content success
2018/04/05 23:44:34 send 【hon.】 content success
2018/04/05 23:44:35 send 【--end】 content success
2018/04/05 23:44:35 send all content, now quit
# 我們客戶端獲取到了服務端返回的文件位置,通過 Seek 來指定從哪里開始讀取文件
# 通過日志可以看到我們客戶端只發送了后面追加的內容: hello python. 到服務端

# 我們看看此時test_1.txt文件的內容是否跟test.txt一致
$ cat test_1.txt
hello golang.
hello python.

以上這篇golang socket斷點續傳大文件的實現方法就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • 解決Goland中利用HTTPClient發送請求超時返回EOF錯誤DEBUG
  • Golang中Delve版本太低無法Debug的問題
  • Golang命令行進行debug調試操作
  • Goland 斷點調試Debug的操作

標簽:德宏 曲靖 保定 許昌 貴州 東營 常州 吐魯番

巨人網絡通訊聲明:本文標題《golang socket斷點續傳大文件的實現方法》,本文關鍵詞  golang,socket,斷點,續傳,大,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《golang socket斷點續傳大文件的實現方法》相關的同類信息!
  • 本頁收集關于golang socket斷點續傳大文件的實現方法的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    亚洲国产精品二十页| 中文字幕在线一区| 777色狠狠一区二区三区| 欧美成人一区二区三区片免费| 中文字幕av在线一区二区三区| 久久电影国产免费久久电影 | 一区二区视频在线| 成人手机在线视频| 亚洲特黄一级片| 欧美视频你懂的| 亚洲激情在线播放| 制服丝袜亚洲精品中文字幕| 天堂资源在线中文精品| 日韩一区二区在线看| 国产一区二区福利| 日韩一区在线播放| 日韩欧美亚洲一区二区| 99热99精品| 精品一区二区日韩| 国产精品色在线观看| 欧美日韩一区不卡| 国产成人精品免费网站| 亚洲一区在线视频| 日韩精品专区在线影院观看| 成人动漫精品一区二区| 另类综合日韩欧美亚洲| 亚洲国产电影在线观看| 欧洲亚洲国产日韩| 国产精品一品视频| 天天免费综合色| 午夜精品福利一区二区蜜股av| 国产婷婷精品av在线| 精品免费视频.| 日韩亚洲欧美一区二区三区| 欧美剧情电影在线观看完整版免费励志电影| 老色鬼精品视频在线观看播放| 五月激情六月综合| 日韩美女久久久| 中文字幕一区日韩精品欧美| 久久久国产综合精品女国产盗摄| 日韩欧美高清在线| 欧美精品色综合| 日韩欧美中文字幕公布| 欧美综合久久久| 欧美精三区欧美精三区| 91精品欧美久久久久久动漫| 91精品在线麻豆| 91精品国产91久久综合桃花| 日韩三级视频中文字幕| 日韩精品在线网站| 国产亚洲欧洲一区高清在线观看| 久久九九久精品国产免费直播| 精品欧美乱码久久久久久1区2区| 久久精品视频网| 亚洲伊人伊色伊影伊综合网| 亚洲一二三四在线| 蜜臀av性久久久久蜜臀av麻豆| 成人av在线资源| 欧美日韩免费不卡视频一区二区三区| 欧美在线观看视频一区二区三区| 欧美日韩午夜影院| 国产亚洲精品bt天堂精选| 欧美激情在线观看视频免费| 性久久久久久久| 夫妻av一区二区| 欧美图片一区二区三区| 精品国产91乱码一区二区三区 | 韩国精品免费视频| 色偷偷久久人人79超碰人人澡| 日韩免费观看高清完整版在线观看| 欧美激情在线看| 成人精品免费看| 日韩一级大片在线观看| 亚洲免费观看高清完整版在线观看| 激情偷乱视频一区二区三区| 色综合天天综合网国产成人综合天 | 91猫先生在线| 亚洲国产wwwccc36天堂| 欧美丝袜自拍制服另类| 亚洲免费资源在线播放| 亚洲一区二区在线免费看| 懂色av噜噜一区二区三区av| 美腿丝袜在线亚洲一区| 亚洲成人免费视| 精品毛片乱码1区2区3区| 日韩精品亚洲一区| 精品嫩草影院久久| 国产成人鲁色资源国产91色综 | 三级欧美韩日大片在线看| 欧美在线|欧美| 亚洲日本在线观看| 欧美性xxxxx极品少妇| 亚洲欧美aⅴ...| 国产精品网站在线| 91丨porny丨首页| 成人av电影在线| 国产丶欧美丶日本不卡视频| 精品午夜久久福利影院| 天天操天天综合网| 一区二区三区四区中文字幕| 精品国产在天天线2019| 91日韩在线专区| av午夜精品一区二区三区| 精品一区二区三区蜜桃| 天使萌一区二区三区免费观看| 亚洲天堂精品视频| 久久天天做天天爱综合色| 欧美精品九九99久久| 欧美日韩国产首页| 欧美视频在线一区| 欧美丰满一区二区免费视频| 欧美影视一区在线| 欧美精品免费视频| 欧美一级夜夜爽| 美女www一区二区| 欧美大片顶级少妇| 日韩欧美国产高清| 日韩和欧美的一区| 一本色道久久加勒比精品| 欧美tickling网站挠脚心| 中文字幕巨乱亚洲| 一区二区在线观看视频在线观看| 亚洲免费av高清| 日本欧美加勒比视频| 精品在线播放免费| 91丨九色porny丨蝌蚪| 色欧美片视频在线观看在线视频| 色婷婷久久99综合精品jk白丝| 欧美一区二区三区视频在线 | 色综合激情久久| 91福利在线看| 亚洲精品一区二区三区四区高清| 蜜臀99久久精品久久久久久软件| av一区二区三区在线| 91视视频在线观看入口直接观看www | 国产人久久人人人人爽| 国产精品久久久久久久久图文区| 中文字幕日韩av资源站| 蜜桃一区二区三区在线观看| 波多野结衣中文字幕一区二区三区 | 97久久超碰国产精品| 精品国产123| 国产美女视频一区| 国产精品色哟哟| 色偷偷一区二区三区| 日韩国产欧美一区二区三区| 久久久久高清精品| 91美女片黄在线观看91美女| 色综合久久九月婷婷色综合| 另类小说综合欧美亚洲| 51精品国自产在线| 日本va欧美va精品发布| www.欧美亚洲| 国产精品免费av| 国产精品一区二区不卡| 欧美一区二区大片| 男女男精品视频网| 久久女同性恋中文字幕| 精品一区二区在线看| 制服丝袜成人动漫| 九九在线精品视频| 欧美精品123区| 欧美精品一区二区三区一线天视频| 亚洲与欧洲av电影| 欧美另类高清zo欧美| 国产一区二区三区最好精华液| 久久一区二区视频| 成人av午夜电影| 日韩电影在线观看电影| 亚洲精品伦理在线| 欧美另类高清zo欧美| 国产aⅴ综合色| 欧美国产成人在线| 国产精品一品二品| 亚洲成人精品一区| 亚洲欧洲精品成人久久奇米网| 欧美性视频一区二区三区| 看片网站欧美日韩| 亚洲一区二区视频| 日本一区二区在线不卡| 一本色道久久加勒比精品| 狠狠网亚洲精品| 亚洲一区二区三区美女| 黄色日韩三级电影| 中文字幕亚洲视频| 日韩三级免费观看| 欧美三区免费完整视频在线观看| 日日摸夜夜添夜夜添精品视频| 国产精品另类一区| 国产精品欧美一级免费| 中文字幕av资源一区| 中文字幕乱码亚洲精品一区| 精品综合久久久久久8888| 免费观看久久久4p| 肉丝袜脚交视频一区二区| 亚洲午夜在线电影| 图片区日韩欧美亚洲| 午夜视频在线观看一区二区| 婷婷久久综合九色国产成人| 一区二区三区欧美|