• golang文件上传和下载


    【代码】golang 实现的文件服务(包括上传,下载的server端和client端)

     
    //下载(支持断电续传)(client)
    
    package main
    
    import (
     
    
       "http"
        "os"
        "io"
        "strconv"
    )
    
    const (
        UA = "Golang Downloader from Kejibo.com"
    )
    
    func main() {
        f, err := os.OpenFile("./file.exe", os.O_RDWR, 0666)  //其实这里的 O_RDWR应该是 O_RDWR|O_CREATE,也就是文件不存在的情况下就建一个空文件,但是因为windows下还有BUG,如果使用这个O_CREATE,就会直接清空文件,所以这里就不用了这个标志,你自己事先建立好文件。
        if err != nil { panic(err) }
        stat, err := f.Stat()    //获取文件状态
        if err != nil { panic(err) }
        f.Seek(stat.Size, 0)    //把文件指针指到文件末,当然你说为何不直接用 O_APPEND 模式打开,没错是可以。我这里只是试验。
        url := "http://dl.google.com/chrome/install/696.57/chrome_installer.exe"
        var req http.Request
        req.Method = "GET"
        req.UserAgent = UA
        req.Close = true
        req.URL, err = http.ParseURL(url)
        if err != nil { panic(err) }
        header := http.Header{}
        header.Set("Range", "bytes=" + strconv.Itoa64(stat.Size) + "-")
        req.Header = header
        resp, err := http.DefaultClient.Do(&req)
        if err != nil { panic(err) }
        written, err := io.Copy(f, resp.Body)
        if err != nil { panic(err) }
        println("written: ", written)
    }
    
    
    //下载(server)
    package main
    
    import (
        "flag"
        "fmt"
        "io"
        "log"
        "net/http"
        "os"
        "path"
        "strconv"
    )
    
    var dir string
    var port int
    var staticHandler http.Handler
    
    // 初始化参数
    func init() {
        dir = path.Dir(os.Args[0])
        flag.IntVar(&port, "port", 800, "服务器端口")
        flag.Parse()
        fmt.Println("dir:", http.Dir(dir))
        staticHandler = http.FileServer(http.Dir(dir))
    }
    
    func main() {
        http.HandleFunc("/", StaticServer)
        err := http.ListenAndServe(":"+strconv.Itoa(port), nil)
        if err != nil {
            log.Fatal("ListenAndServe: ", err)
        }
    }
    
    // 静态文件处理
    func StaticServer(w http.ResponseWriter, req *http.Request) {
        fmt.Println("path:" + req.URL.Path)
        if req.URL.Path != "/down/" {
            staticHandler.ServeHTTP(w, req)
            return
        }
    
        io.WriteString(w, "hello, world!
    ")
    }
    
    //上传(client)
    
    package main
    
    import (
        "bytes"
        "fmt"
        "io"
        "io/ioutil"
        "mime/multipart"
        "net/http"
        "os"
    )
    
    func postFile(filename string, targetUrl string) error {
        bodyBuf := &bytes.Buffer{}
        bodyWriter := multipart.NewWriter(bodyBuf)
        //关键的一步操作
        fileWriter, err := bodyWriter.CreateFormFile("uploadfile", filename)
        if err != nil {
            fmt.Println("error writing to buffer")
            return err
        }
        //打开文件句柄操作
        fh, err := os.Open(filename)
        if err != nil {
            fmt.Println("error opening file")
            return err
        }
    
        //iocopy
        _, err = io.Copy(fileWriter, fh)
        if err != nil {
            return err
        }
        contentType := bodyWriter.FormDataContentType()
        bodyWriter.Close()
        resp, err := http.Post(targetUrl, contentType, bodyBuf)
        if err != nil {
            return err
        }
        defer resp.Body.Close()
        resp_body, err := ioutil.ReadAll(resp.Body)
        if err != nil {
            return err
        }
        fmt.Println(resp.Status)
        fmt.Println(string(resp_body))
    
        return nil
    }
    
    // sample usage
    func main() {
        target_url := "http://localhost:9090/upload"
        filename := "./astaxie.pdf"
        postFile(filename, target_url)
    }
    
    //上传(server)
    package main
    
    import (
        "crypto/md5"
        "flag"
        "fmt"
        "html/template"
        "io"
        "log"
        "net/http"
        "os"
        "path"
        "strconv"
        "time"
    )
    
    var dir string
    var port int
    
    // 初始化参数
    func init() {
        dir = path.Dir(os.Args[0])
        flag.IntVar(&port, "port", 800, "服务器端口")
        flag.Parse()
        fmt.Println("dir:", http.Dir(dir))
    }
    
    func main() {
        http.HandleFunc("/upload", upload)
        err := http.ListenAndServe(":"+strconv.Itoa(port), nil)
        if err != nil {
            log.Fatal("ListenAndServe: ", err)
        }
    }
    
    // 处理/upload 逻辑
    func upload(w http.ResponseWriter, r *http.Request) {
        fmt.Println("method:", r.Method) //获取请求的方法
        if r.Method == "GET" {
            crutime := time.Now().Unix()
            h := md5.New()
            io.WriteString(h, strconv.FormatInt(crutime, 10))
            token := fmt.Sprintf("%x", h.Sum(nil))
            t, _ := template.ParseFiles("upload.gtpl")
            t.Execute(w, token)
        } else {
            r.ParseMultipartForm(32 << 20)
            file, handler, err := r.FormFile("uploadfile")
            if err != nil {
                fmt.Println(err)
                return
            }
            defer file.Close()
            fmt.Fprintf(w, "%v", handler.Header)
            f, err := os.OpenFile("./upload/"+handler.Filename, os.O_WRONLY|os.O_CREATE, 0666)
            if err != nil {
                fmt.Println(err)
                return
            }
            defer f.Close()
            io.Copy(f, file)
        }
    }
    

      

  • 相关阅读:
    Android中的Prelink技术
    Android 性能分析工具之 ARM Streamline
    Android之Systrace
    学习资源收藏夹
    Linux利器之perf(火焰图)
    Linux程序Segmentation fault (core dumped)
    C++编译器、链接器工作原理
    react使用redux
    Mac更新node版本和npm版本
    Nuxt引入axios;AXIOS的模块化封装
  • 原文地址:https://www.cnblogs.com/zhangym/p/5841732.html
Copyright © 2020-2023  润新知