Golang 一些基本功能使用記錄

(TOC)mysql

命令行參數

import "flag"
func main() {
	ports := flag.String("ports", "10086", "list the port ....")
	var version_check bool
	flag.BoolVar(&version_check, "v", false, "version")
	flag.Parse()
	if version_check {
		do()
	}
	portsList := strings.Split(*ports, ",")
	if len(portsList) > 5 {
		fmt.Println("no more than 5 ports")
		return
	}
}

配置文件

package main
import (
	"fmt"
	"github.com/BurntSushi/toml"
)
type tomlConfig struct {
	Title	string
	Mysql	mysqlInfo
}
type mysqlInfo struct {
	Host	string
	Port	int
	User	string
	Passwd	string
	Db	string
}
func main() {
	var config tomlConfig
	if _, err := toml.DecodeFile("config.toml", &config); err != nil {
		fmt.Println(err)
		return
	}
	fmt.Println(config.Title)
	fmt.Println(config.Mysql.Host)
}

配置文件以下,須要名稱和tomlconfig中的相同,而且代碼中首字母大寫,配置中小寫git

Title = "test"
[mysql]
host = "11.22.33.44"
port = 3306
user = "smart"
passwd = "smart"
db = "smart"

MySQL

package main
import (
	"database/sql"
	"fmt"
	_ "github.com/go-sql-driver/mysql"
)
func fetchdata() {
	db, err := sql.Open("mysql", "user:passwd@tcp(host:port)/database")
	if err != nil {
		fmt.Println(err)
	}
	err = db.Ping()
	if err != nil {
		fmt.Println(err)
	}
	rows, err := db.Query("select * from table")
	if err != nil {
		fmt.Println(err)
	}
	for rows.Next() {
		var counts string
		var indexs int
		if err := rows.Scan(&counts, &indexs); err != nil {
			fmt.Println(err)
		}
		fmt.Println(counts, indexs)
	}
	defer db.Close()
}
func main() {
	fetchdata()
}

Socket

func udpReceiver(port string) {
	defer workExitLock.Done()
	var addr *net.UDPAddr
	var server *net.UDPConn
	var err error
	if addr, err = net.ResolveUDPAddr("udp", port); err != nil {
		Error.Printf("Udp listener error: %s", err)
		return
	}
	if server, err = net.ListenUDP("udp", addr); err != nil {
		Error.Printf("Udp listener error: %s", err)
		return
	}
	if err = server.SetReadBuffer(UDP_READ_BUFF); err != nil {
		Error.Printf("Udp listener error: %s", err)
		return
	}
	Info.Printf("Listen Udp Sucessfully, port: %s", port)
	var buf []byte
	for {
		if len(buf) < UDP_PACK_SIZE {
			buf = make([]byte, PACK_BUF_SIZE, PACK_BUF_SIZE)
		}
		nbytes, addr, err := server.ReadFromUDP(buf)
		if err != nil {
			Error.Printf("Receive udp data error: %s", err)
			continue
		}
		msg := buf[:nbytes]
		buf = buf[nbytes:]
		udpChan <- UdpMessage{addr, msg}
	}
}
 
// 可選udp或者tcp
func netSender(typename string, addr string, retry int) {
	for {
		conn, err := net.Dial(typename, addr)
		if err != nil {
			Error.Printf("make conn error: ", err)
			time.Sleep(time.Duration(retry) * time.Second)
			continue
		}
		for one := range resChan {
			_, err = conn.Write([]byte(one))
			if err != nil {
				Error.Printf("send data error: %s", err)
				conn.Close()
				break
			}
		}
	}
}

在實驗環境中,若是對端沒有處理指定端口的數據,會每隔一秒種返回一個ICMP包提示 udp port distinct unreachable 此時發送程序會有對應的發送失敗提示,也是每秒鐘一個github

log

package main
import (
	"config"
	"io"
	"log"
	"os"
)
var (
	Info    *log.Logger
	Error   *log.Logger
)
func Init(
	infoHandle io.Writer,
	errorHandle io.Writer) {
	Info = log.New(infoHandle,
		"INFO: ",
		log.Ldate|log.Ltime|log.Lshortfile)
	Error = log.New(errorHandle,
		"ERROR: ",
		log.Ldate|log.Ltime|log.Lshortfile)
}
func logInit(conf config.ProcessInfo) {
	pinfo := conf.Logdir + "/" + "log.info"
	perro := conf.Logdir + "/" + "log.err"
	finfo, err := os.OpenFile(pinfo, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0666)
	if err != nil {
		log.Fatalf("file open error : %v", err)
	}
	ferro, err := os.OpenFile(perro, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0666)
	if err != nil {
		log.Fatalf("file open error : %v", err)
	}
	Init(finfo, ferro)
}
 
func main() {
	gomaxprocs := runtime.NumCPU() - 1
	runtime.GOMAXPROCS(gomaxprocs)
	Info.Printf("Program start. GOMAXPROCS: %d", gomaxprocs)
}

同步統計

使用sync/atomic,利用其實現的函數能夠達到無衝突的統計golang

import "sync/atomic"
type count64 uint64
func (c *count64) increment(incr int) count64 {
	return count64(atomic.AddUint64((*uint64)(c), uint64(incr)))
}
func (c *count64) get() count64 {
	return count64(atomic.LoadUint64((*uint64)(c)))
}

其餘

要注意golang特有的一些格式要求,好比:sql

  1. 不能有沒用的變量,沒用到的import
  2. 對外提供的函數,首字母必須大寫,不然不能被其餘package使用。同一個package不一樣文件的函數和全局變量直接使用
  3. 儘可能少使用異常,經過返回err進行判斷和處理,panic在嚴重時才使用
相關文章
相關標籤/搜索