好湿?好紧?好多水好爽自慰,久久久噜久噜久久综合,成人做爰A片免费看黄冈,机机对机机30分钟无遮挡

主頁 > 知識庫 > Go實現短url項目的方法示例

Go實現短url項目的方法示例

熱門標簽:建造者2地圖標注 阿里云ai電話機器人 鄭州亮點科技用的什么外呼系統 汕頭小型外呼系統 釘釘有地圖標注功能嗎 浙江高頻外呼系統多少錢一個月 黃岡人工智能電銷機器人哪個好 惠州電銷防封電話卡 濱州自動電銷機器人排名

首先說一下這種業務的應用場景:
1.把一個長url轉換為一個短url網址
2.主要用于微博,二維碼,等有字數限制的場景

主要實現的功能分析:
1.把長url的地址轉換為短url地址
2.通過短url獲取對應的原始長url地址
3.相同長url地址是否需要同樣的短url地址

這里實現的是一個api服務

數據庫設計

數據庫的設計其實也沒有非常復雜,如圖所示:

這里有個設置需要主要就是關于數據庫表中id的設計,需要設置為自增的

并且這里有個問題需要提前知道,我們的思路是根據id的值會轉換為62進制關于進制轉換的代碼為:

// 將十進制轉換為62進制  0-9a-zA-Z 六十二進制
func transTo62(id int64)string{
  // 1 -- > 1
  // 10-- > a
  // 61-- > Z
  charset := "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
  var shortUrl []byte
  for{
    var result byte
    number := id % 62
    result = charset[number]
    var tmp []byte
    tmp = append(tmp,result)
    shortUrl = append(tmp,shortUrl...)
    id = id / 62
    if id == 0{
      break
    }
  }
  fmt.Println(string(shortUrl))
  return string(shortUrl)
}

所以這里需要設置一下數據庫id的起始值,可以設置的大一點,這樣轉換為62進制之后不至于太短

代碼邏輯

項目完整的代碼git地址:https://github.com/pythonsite/go_simple_code/tree/master/short_url
當然這里的代碼還有待后面繼續做優化,但是這里通過golang內置的net/http 庫實現了一個簡單的api功能

代碼的目錄結構

|____logic
| |____logic.go
|____model
| |____data.go
|____api
| |____api.go
|____client
| |____client.go

logic目錄為主要的處理邏輯
model是定義了request和response結構體
api目錄為程序的入口程序
client 為測試請求,進行地址的轉換

model 代碼為:

package model


type Long2ShortRequest struct {
  OriginUrl string `json:"origin_url"`
}

type ResponseHeader struct {
  Code int `json:"code"`
  Message string `json:"message"`
}

type Long2ShortResponse struct {
  ResponseHeader
  ShortUrl string `json:"short_url"`
}

type Short2LongRequest struct {
  ShortUrl string `json:"short_url"`
}

type Short2LongResponse struct {
  ResponseHeader
  OriginUrl string `json:"origin_url"`
}

logic的代碼為:

package logic

import(
  "go_dev/11/short_url/model"
  "github.com/jmoiron/sqlx"
  "fmt"
  "crypto/md5"
  "database/sql"
)

var (
  Db *sqlx.DB
)

type ShortUrl struct {
  Id int64 `db:"id"`
  ShortUrl string `db:"short_url"`
  OriginUrl string `db:"origin_url"`
  HashCode string `db:"hash_code"`
}

func InitDb(dsn string)(err error) {
  // 數據庫初始化
  Db, err = sqlx.Open("mysql",dsn)
  if err != nil{
    fmt.Println("connect to mysql failed:",err)
    return
  }
  return
}

func Long2Short(req *model.Long2ShortRequest) (response *model.Long2ShortResponse, err error) {
  response = model.Long2ShortResponse{}
  urlMd5 := fmt.Sprintf("%x",md5.Sum([]byte(req.OriginUrl)))
  var short ShortUrl
  err = Db.Get(short,"select id,short_url,origin_url,hash_code from short_url where hash_code=?",urlMd5)
  if err == sql.ErrNoRows{
    err = nil
    // 數據庫中沒有記錄,重新生成一個新的短url
    shortUrl,errRet := generateShortUrl(req,urlMd5)
    if errRet != nil{
      err = errRet
      return
    }
    response.ShortUrl = shortUrl
    return
  }
  if err != nil{
    return
  }
  response.ShortUrl = short.ShortUrl
  return
}

func generateShortUrl(req *model.Long2ShortRequest,hashcode string)(shortUrl string,err error){
  result,err := Db.Exec("insert INTO short_url(origin_url,hash_code)VALUES (?,?)",req.OriginUrl,hashcode)
  if err != nil{
    return
  }
  // 0-9a-zA-Z 六十二進制
  insertId,_:= result.LastInsertId()
  shortUrl = transTo62(insertId)
  _,err = Db.Exec("update short_url set short_url=? where id=?",shortUrl,insertId)
  if err != nil{
    fmt.Println(err)
    return
  }
  return
}

// 將十進制轉換為62進制  0-9a-zA-Z 六十二進制
func transTo62(id int64)string{
  // 1 -- > 1
  // 10-- > a
  // 61-- > Z
  charset := "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
  var shortUrl []byte
  for{
    var result byte
    number := id % 62
    result = charset[number]
    var tmp []byte
    tmp = append(tmp,result)
    shortUrl = append(tmp,shortUrl...)
    id = id / 62
    if id == 0{
      break
    }
  }
  fmt.Println(string(shortUrl))
  return string(shortUrl)
}


func Short2Long(req *model.Short2LongRequest) (response *model.Short2LongResponse, err error) {
  response = model.Short2LongResponse{}
  var short ShortUrl
  err = Db.Get(short,"select id,short_url,origin_url,hash_code from short_url where short_url=?",req.ShortUrl)
  if err == sql.ErrNoRows{
    response.Code = 404
    return
  }
  if err != nil{
    response.Code = 500
    return
  }
  response.OriginUrl = short.OriginUrl
  return
}

api的代碼為:

package main

import (
  "io/ioutil"
  "net/http"
  "fmt"
  "encoding/json"
  "go_dev/11/short_url/logic"
  "go_dev/11/short_url/model"
  _ "github.com/go-sql-driver/mysql"
)

const (
  ErrSuccess = 0
  ErrInvalidParameter = 1001
  ErrServerBusy = 1002
)

func getMessage(code int) (msg string){
  switch code {
  case ErrSuccess:
    msg = "success"
  case ErrInvalidParameter:
    msg = "invalid parameter"
  case ErrServerBusy:
    msg = "server busy"
  default:
    msg = "unknown error"
  }

  return
}

// 用于將返回序列化數據,失敗的返回
func responseError(w http.ResponseWriter, code int) {
  var response model.ResponseHeader
  response.Code = code
  response.Message = getMessage(code)

  data, err := json.Marshal(response)
  if err != nil {
    w.Write([]byte("{\"code\":500, \"message\": \"server busy\"}"))
    return
  }

  w.Write(data)
}

// 用于將返回序列化數據,成功的返回
func responseSuccess(w http.ResponseWriter, data interface{}) {


  dataByte, err := json.Marshal(data)
  if err != nil {
    w.Write([]byte("{\"code\":500, \"message\": \"server busy\"}"))
    return
  }

  w.Write(dataByte)
}

// 長地址到短地址
func Long2Short(w http.ResponseWriter, r *http.Request) {
  // 這里需要說明的是發來的數據是通過post發過來一個json格式的數據
  data, err := ioutil.ReadAll(r.Body)
  if err != nil {
    fmt.Println("read all failded, ", err)
    responseError(w, 1001)
    return
  }

  var req model.Long2ShortRequest
  // 將反序列化的數據保存在結構體中
  err = json.Unmarshal(data, req)
  if err != nil {
    fmt.Println("Unmarshal failded, ", err)
    responseError(w, 1002)
    return
  }

  resp, err := logic.Long2Short(req)
  if err != nil {
    fmt.Println("Long2Short failded, ", err)
    responseError(w, 1003)
    return
  }

  responseSuccess(w, resp)
}

// 短地址到長地址
func Short2Long(w http.ResponseWriter, r *http.Request) {
  // 這里需要說明的是發來的數據是通過post發過來一個json格式的數據
  data, err := ioutil.ReadAll(r.Body)
  if err != nil {
    fmt.Println("read all failded, ", err)
    responseError(w, 1001)
    return
  }

  var req model.Short2LongRequest
  // 將反序列化的數據保存在結構體中
  err = json.Unmarshal(data, req)
  if err != nil {
    fmt.Println("Unmarshal failded, ", err)
    responseError(w, 1002)
    return
  }

  resp, err := logic.Short2Long(req)
  if err != nil {
    fmt.Println("Long2Short failded, ", err)
    responseError(w, 1003)
    return
  }
  responseSuccess(w, resp)
}

func main(){
  err := logic.InitDb("root:123456@tcp(192.168.50.145:3306)/short_url?parseTime=true")
  if err != nil{
    fmt.Printf("init db failed,err:%v\n",err)
    return
  }
  http.HandleFunc("/trans/long2short", Long2Short)
  http.HandleFunc("/trans/short2long", Short2Long)
  http.ListenAndServe(":18888", nil)
}

小結

這次通過這個小代碼對go也有了一個初步的認識和使用,同時也通過net/http 包實現了api的功能,也對其基本使用有了大致了解

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • Django靜態資源URL STATIC_ROOT的配置方法
  • Django中URLconf和include()的協同工作方法
  • Django URL傳遞參數的方法總結
  • python3使用urllib示例取googletranslate(谷歌翻譯)
  • go語言實現一個簡單的http客戶端抓取遠程url的方法
  • Django中URL視圖函數的一些高級概念介紹
  • Django中傳遞參數到URLconf的視圖函數中的方法
  • Python中Django框架利用url來控制登錄的方法
  • 在Django的URLconf中使用命名組的方法

標簽:昭通 晉中 瀘州 東營 駐馬店 滄州 泰安 阿壩

巨人網絡通訊聲明:本文標題《Go實現短url項目的方法示例》,本文關鍵詞  實現,短,url,項,目的,方法,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Go實現短url項目的方法示例》相關的同類信息!
  • 本頁收集關于Go實現短url項目的方法示例的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 欧美综合一区二区三区| 免费看黄网址| 偷玩男生大jiji秘?免费网站| 一边做饭一边躁狂暴躁视频大全| 劳拉的性放荡bd中文版| 亚洲乱码一区二区三区香蕉| 黑人成人影院| 亚洲激情在线播放| 色就是se94se人妻无码| 天天躁夜夜躁av| 少妇搡BBBB搡BBB搡爱恋| 久久久久久久久久免免费精品| 将进酒小说免费阅读全文无删减| 无码av秘?一区二区三区狠狠| 察隅县| 用力粗大啊水揉捏湿透h| 国产伦一区二区三区四区久久| 太爽了我要丢了| 欧美ed2k| 毛片69| 欧美最婬乱婬爆婬A片| 天天操天天草| 欧美黑人巨大xxxxx视频| 亚洲人成网站在线观看播放青青| 日本精品在线观看| 超级大乳BBWBBW乃BW| 非洲黑人高清一级毛片| 一级二级三级毛片| 午夜一区二区三区在线观看| 理伦电影网站在线观看| 憋着晨尿做h失禁| 叔叔不约| 天堂漫画在线观看| 受不了?太大太粗太满床戏| 调教jk黑色丝袜超薄过膝袜| 久久综合一区二区三区| 鬼灭之刃第三季在线观看免费动漫 | 亚洲精品第一国产麻豆| 日本a√在线| 亚洲性天堂| 18禁免费吃奶摸下激烈视频|