gpt4 book ai didi

go - 尝试在 golang 中使用 io.CopyN() 读取时客户端卡住了

转载 作者:IT王子 更新时间:2023-10-29 02:36:52 25 4
gpt4 key购买 nike

我正在尝试制作用于传输文件的 TCP 服务器。我起诉 io.CopyN 阅读和写作。从服务器端,我正在向客户端发送文件,因此从服务器端,它发送完美的所有字节,但客户端在读取了几个 1000000 字节后卡住了。有时它工作正常,有时它会卡住。我正在使用 300 MB pdf 进行测试。任何帮助、代码和输出如下所示。

服务器

    package main

import (
"fmt"
"io"
"log"
"net"
"os"
"strconv"
"strings"
)

func main() {

ls, err := net.Listen("tcp", ":1234")

errFunc(err)

defer ls.Close()

conn, _ := ls.Accept()

defer conn.Close()

for {

file, err := os.Open(strings.TrimSpace("./" + "Mag" + ".pdf"))

errFunc(err)

defer file.Close()

fileInfo, err := file.Stat()

errFunc(err)

size := fileInfo.Size()

numberOfTime := size / 1000000

leftByte := size - numberOfTime*1000000

numberOfTimeString := strconv.Itoa(int(numberOfTime))
leftByteString := strconv.Itoa(int(leftByte))

fmt.Println("1000000 times : ", numberOfTimeString)

fmt.Println("Left Bytes : ", leftByteString)

_, err = fmt.Fprintf(conn, numberOfTimeString+"\n")

errFunc(err)

_, err = fmt.Fprintf(conn, leftByteString+"\n")

errFunc(err)

fileWriter := io.Writer(conn)

for i := 0; i < int(numberOfTime); i++ {

n, err := io.CopyN(conn, file, 1000000)

if i >= 30 {
fmt.Println(err, n)
}
}

n, err := io.CopyN(fileWriter, file, leftByte+1)

if err == io.EOF {
fmt.Println(err, n)
}

fmt.Printf("Succefully bytes sent : %v \n\n\n\n\n", n)

file.Close()

}

}

func errFunc(err error) {

if err != nil {
log.Fatal(err)
}

}

客户端

    package main

import (
"bufio"
"fmt"
"io"
"net"
"os"
"os/signal"
"strconv"
"strings"
"syscall"
)

func main() {

c := make(chan os.Signal, 15)
signal.Notify(c, syscall.SIGINT)

go func() {

for {
s := <-c

switch s {

case syscall.SIGINT:
os.Exit(1)
}

}

}()

conn, _ := net.Dial("tcp", ":1234")

defer conn.Close()

connReadWrite := bufio.NewReader(io.Reader(conn))

var i int
var filename string

for {

i++

nu := strconv.Itoa(i)

filename = "image" + nu + ".pdf"

file, err := os.Create(filename)

defer file.Close()

numberOfTimeString, err := connReadWrite.ReadString('\n')

if err != nil {

fmt.Println(err)
}

println("1000000 times :", numberOfTimeString)

numberOfTimeString = strings.TrimSuffix(numberOfTimeString, "\n")

numberOfTime, err := strconv.Atoi(numberOfTimeString)

if err != nil {

fmt.Println(err)
}

leftByteString, err := connReadWrite.ReadString('\n')

if err != nil {

println(err)
}

println("Left Bytes :", leftByteString)

leftByteString = strings.TrimSuffix(leftByteString, "\n")

leftByte, err := strconv.Atoi(leftByteString)

if err != nil {

panic(err)
}

fmt.Println("After convert in Num :", numberOfTime, leftByte)

newFileWriter := io.Writer(file)
newFileReader := io.Reader(conn)

for i := 0; i < numberOfTime; i++ {

n, err := io.CopyN(newFileWriter, newFileReader, 1000000)

if i >= 30 {
errFun(err, n)
}
}

n, err := io.CopyN(newFileWriter, newFileReader, int64(leftByte))

errFun(err, n)

fmt.Printf("sucessfully Transfered ---> \n\n\n\n\n\n")

}

}

func errFun(err error, n int64) {

if err == io.EOF {

fmt.Println("End of file : ", n)
return

} else if n == 0 {

fmt.Println("n is : ", n)
return

} else if err != nil {
fmt.Println(err)
return

}

fmt.Println(err, " : ", n)
}

输入/输出

首先我们从服务器端发送它需要读取的字节数,然后客户端它获得它需要读取的字节数,然后我发送文件,然后它读取。在图片中,我能够发送一次,第二次它卡住了,有时它第一次也卡住了。我也能够第二次从服务器端发送字节数,但正如你所看到的,它没有读取那个数字,它读取了一些“%PDF...”,它甚至不打印“100000 次:”,它正确地打印了“%???00 次:”我只是不明白这一点

enter image description here

最佳答案

我认为问题在于您在客户端中使用了 bytes.Buffer:

connReadWrite := bufio.NewReader(io.Reader(conn))

但是您稍后不会将它与 CopyN 一起使用:

newFileWriter := io.Writer(file)
newFileReader := io.Reader(conn)
for i := 0; i < numberOfTime; i++ {
_, err := io.CopyN(newFileWriter, newFileReader, 1000000)
if err != nil {
log.Fatalln(err)
}
}

使用:

newFileWriter := io.Writer(file)

for i := 0; i < numberOfTime; i++ {
_, err := io.CopyN(file, connReadWrite, 1000000)
if err != nil {
log.Fatalln(err)
}
}

可以修复它。

如果您可以控制用于发送文件的协议(protocol),我建议您做一些更简单的事情。例如使用 big-endian int64 长度前缀。

发送:

func sendFile(name string, conn net.Conn) error {
f, err := os.Open(name)
if err != nil {
return err
}
defer f.Close()

fi, err := f.Stat()
if err != nil {
return err
}
sz := fi.Size()

buf := bufio.NewWriter(conn)

err = binary.Write(buf, binary.BigEndian, sz)
if err != nil {
return err
}

_, err = io.CopyN(buf, f, sz)
if err != nil {
return err
}

return buf.Flush()
}

接收:

func recvFile(name string, conn net.Conn) error {
f, err := os.Create(name)
if err != nil {
return err
}
defer f.Close()

buf := bufio.NewReader(conn)
var sz int64
err = binary.Read(buf, binary.BigEndian, &sz)
if err != nil {
return err
}

_, err = io.CopyN(f, buf, sz)
if err != nil {
return err
}

return nil
}

关于go - 尝试在 golang 中使用 io.CopyN() 读取时客户端卡住了,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52233610/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com