Golang Sqlx Gorm 对比

Sqlx Gorm 对比

170W数据下,sqlx和gorm查询不同条数数据性能对比,机器配置 8核/16G,mysql分别最大连接数为1 和 最大连接数500最大空闲连接数100的情况下测试。

package tesst

import (
	"fmt"
	"testing"
	"time"

	"github.com/jmoiron/sqlx"
	"gorm.io/driver/mysql"
	"gorm.io/gorm"
)

type ShardingDB struct {
	ID        uint64    `db:"id" gorm:"column:id"`
	DBID      string    `db:"db_id" gorm:"column:db_id"`
	Host      string    `db:"host" gorm:"column:host"`
	Port      int32     `db:"port" gorm:"column:port"`
	User      string    `db:"user" gorm:"column:user"`
	Password  string    `db:"password" gorm:"column:password"`
	Memo      string    `db:"memo" gorm:"column:memo"`
	State     int32     `db:"state" gorm:"column:state"`
	CreatedAt time.Time `db:"create_time" gorm:"column:create_time"`
	UpdatedAt time.Time `db:"update_time" gorm:"column:update_time"`
}

// TableName returns table name of sharding_db.
func (s *ShardingDB) TableName() string {
	return "sharding_db"
}

var (
	dbhostsip  = "127.0.0.1"  // IP地址
	dbport     = 3306         // Port
	dbusername = "root"       // 用户名
	dbpassword = "admin"      // 密码
	dbname     = "test" // 表名
)

func Benchmark(b *testing.B) {
	dsn := fmt.Sprintf("%s:%s@(%s:%d)/%s?charset=utf8&parseTime=True&loc=UTC",
		dbusername, dbpassword, dbhostsip, dbport, dbname)

	limits := []int{
		5,
		50,
		500,
		10000,
	}

	sqlxDB, _ := sqlx.Connect("mysql", dsn)
	sqlxDB.SetMaxOpenConns(500)
	sqlxDB.SetMaxIdleConns(100)

	gormDB, err := gorm.Open(mysql.Open(dsn), &gorm.Config{NowFunc: func() time.Time { return time.Now().UTC().Round(time.Microsecond) }})
	db, _ := gormDB.DB()
	db.SetMaxOpenConns(500)
	db.SetMaxIdleConns(100)

	fmt.Println("=============================== CPU:8 MEM:16G MaxOpenConns:500 MaxOpenConns:100 ====================================")

	for _, lim := range limits {
		lim := lim
		

		// Benchmark sqlx
		b.Run(fmt.Sprintf("sqlx limit:%d", lim), func(b *testing.B) {
			for i := 0; i < b.N; i++ {
				q := fmt.Sprintf("SELECT * FROM sharding_db ORDER BY id LIMIT %d", lim)
				res := []ShardingDB{}
				err := sqlxDB.Select(&res, q)
				if err != nil {
					b.Fatal(err)
				}
			}
		})

		
		if err != nil {
			panic(err)
		}
		// Benchmark gormDB
		b.Run(fmt.Sprintf("gormDB limit:%d", lim), func(b *testing.B) {
			for i := 0; i < b.N; i++ {
				for i := 0; i < b.N; i++ {
					var res = []ShardingDB{}
					err := gormDB.Order("id").Limit(lim).Find(&res).Error

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

		fmt.Println("==================================================================================================================")
	}
}

测试结果:

=============================== CPU:8 MEM:16G MaxOpenConns:1 ====================================
goos: linux
goarch: amd64
pkg: go-operation-demo/sqlx/bench-test-demo
Benchmark/sqlx_limit:5-8         	   10000	    102170 ns/op	    7275 B/op	     134 allocs/op
Benchmark/gormDB_limit:5-8       	     100	  13250163 ns/op	 1199221 B/op	   29300 allocs/op
==================================================================================================================
Benchmark/sqlx_limit:50-8        	    4472	    270684 ns/op	   52652 B/op	    1082 allocs/op
Benchmark/gormDB_limit:50-8      	     100	  52683096 ns/op	 7362504 B/op	  250000 allocs/op
==================================================================================================================
Benchmark/sqlx_limit:500-8       	     814	   1480242 ns/op	  480048 B/op	   10536 allocs/op
Benchmark/gormDB_limit:500-8     	     100	 395335077 ns/op	68082795 B/op	 2455402 allocs/op
==================================================================================================================
Benchmark/sqlx_limit:10000-8     	      39	  28053826 ns/op	14964052 B/op	  210053 allocs/op
Benchmark/gormDB_limit:10000-8   	      14	1086946301 ns/op	254799549 B/op	 6860912 allocs/op
==================================================================================================================
PASS
ok  	go-operation-demo/sqlx/bench-test-demo	66.189s
=============================== CPU:8 MEM:16G MaxOpenConns:500 MaxIdleConns:100 ====================================
goos: linux
goarch: amd64
pkg: go-operation-demo/sqlx/bench-test-demo
Benchmark/sqlx_limit:5-8         	   10000	    100589 ns/op	    7274 B/op	     134 allocs/op
Benchmark/gormDB_limit:5-8       	     100	  13322948 ns/op	 1199224 B/op	   29300 allocs/op
==================================================================================================================
Benchmark/sqlx_limit:50-8        	    4651	    265969 ns/op	   52652 B/op	    1082 allocs/op
Benchmark/gormDB_limit:50-8      	     100	  52195341 ns/op	 7362505 B/op	  250000 allocs/op
==================================================================================================================
Benchmark/sqlx_limit:500-8       	     820	   1466335 ns/op	  480048 B/op	   10536 allocs/op
Benchmark/gormDB_limit:500-8     	     100	 395699610 ns/op	68082802 B/op	 2455402 allocs/op
==================================================================================================================
Benchmark/sqlx_limit:10000-8     	      40	  28599357 ns/op	14964058 B/op	  210053 allocs/op
Benchmark/gormDB_limit:10000-8   	      15	1158298778 ns/op	272999555 B/op	 7350977 allocs/op
==================================================================================================================
PASS
ok  	go-operation-demo/sqlx/bench-test-demo	68.395s

总结:

Sqlx 查询时间和内存占用远远小于 Gorm。

Sqlx 封装对比

对于系统db层面来说,每个model都需要CRUD操作,如果不对Sqlx再封装一层的话,使用起来需要写大量的Sqlx的冗余代码。但是如果封装一层的话,就需要考虑性能消耗和临时变量的消耗,这里仅举一个Select的示例。

测试代码

package database

import (
	"fmt"
	"log"
	"testing"
	"time"

	"github.com/jmoiron/sqlx"
	_ "gorm.io/driver/mysql"
)

type ShardingDB struct {
	ID        uint64    `db:"id" gorm:"column:id"`
	DBID      string    `db:"db_id" gorm:"column:db_id"`
	Host      string    `db:"host" gorm:"column:host"`
	Port      int32     `db:"port" gorm:"column:port"`
	User      string    `db:"user" gorm:"column:user"`
	Password  string    `db:"password" gorm:"column:password"`
	Memo      string    `db:"memo" gorm:"column:memo"`
	State     int32     `db:"state" gorm:"column:state"`
	CreatedAt time.Time `db:"create_time" gorm:"column:create_time"`
	UpdatedAt time.Time `db:"update_time" gorm:"column:update_time"`
}

// TableName returns table name of sharding_db.
func (s *ShardingDB) TableName() string {
	return "sharding_db"
}

var (
	dbhostsip  = "127.0.0.1"  // IP地址
	dbport     = 3306         // Port
	dbusername = "root"       // 用户名
	dbpassword = "admin"      // 密码
	dbname     = "test" // 表名
)

func Benchmark(b *testing.B) {
	dsn := fmt.Sprintf("%s:%s@(%s:%d)/%s?charset=utf8&parseTime=True&loc=UTC",
		dbusername, dbpassword, dbhostsip, dbport, dbname)

	limits := []int{
		5,
		50,
		500,
		10000,
	}

	fmt.Println("=============================== CPU:8 MEM:16G MaxOpenConns:500 MaxIdleConns:100 ====================================")

	for _, lim := range limits {
		lim := lim
		// Benchmark sqlx
		b.Run(fmt.Sprintf("sqlx query limit:%d", lim), func(b *testing.B) {
			sqlxDB, err := sqlx.Connect("mysql", dsn)
			if err != nil {
				log.Fatal(err)
			}
			sqlxDB.SetMaxOpenConns(1)

			for i := 0; i < b.N; i++ {
				sqlxQuery(sqlxDB, lim)
			}
		})

		// 指针使用方式不可行,没办法映射字段
		// b.Run(fmt.Sprintf("sqlx ptr query limit:%d", lim), func(b *testing.B) {
		// 	sqlxDB, err := sqlx.Connect("mysql", dsn)
		// 	if err != nil {
		// 		log.Fatal(err)
		// 	}
		// 	sqlxDB.SetMaxOpenConns(1)
		//
		// 	for i := 0; i < b.N; i++ {
		// 		sqlxPtrQuery(sqlxDB, lim)
		// 	}
		// })

		b.Run(fmt.Sprintf("sqlx select query limit:%d", lim), func(b *testing.B) {
			sqlxDB, err := sqlx.Connect("mysql", dsn)
			if err != nil {
				log.Fatal(err)
			}
			sqlxDB.SetMaxOpenConns(1)

			for i := 0; i < b.N; i++ {
				sqlxSelectQuery(sqlxDB, lim)
			}
		})

		// 指针使用方式不可行,没办法映射字段
		// b.Run(fmt.Sprintf("sqlx select ptr query limit:%d", lim), func(b *testing.B) {
		// 	sqlxDB, err := sqlx.Connect("mysql", dsn)
		// 	if err != nil {
		// 		log.Fatal(err)
		// 	}
		// 	sqlxDB.SetMaxOpenConns(1)
		//
		// 	for i := 0; i < b.N; i++ {
		// 		sqlxSelectPtrQuery(sqlxDB, lim)
		// 	}
		// })


		fmt.Println("==================================================================================================================")
	}
}

func sqlxQuery(sqlxDB *sqlx.DB, lim int) (shardingDB []ShardingDB) {
	q := fmt.Sprintf("SELECT * FROM sharding_db ORDER BY id LIMIT %d", lim)
	err := sqlxDB.Select(&shardingDB, q)
	if err != nil {
		log.Fatal(err)
	}
	return
}

func sqlxPtrQuery(sqlxDB *sqlx.DB, lim int) (shardingDB *[]ShardingDB) {
	q := fmt.Sprintf("SELECT * FROM sharding_db ORDER BY id LIMIT %d", lim)
	err := sqlxDB.Select(shardingDB, q)
	if err != nil {
		log.Fatal(err)
	}
	return
}

func sqlxSelectQuery(sqlxDB *sqlx.DB, lim int) (shardingDB []ShardingDB) {
	q := fmt.Sprintf("SELECT * FROM sharding_db ORDER BY id LIMIT %d", lim)
	err := SqlxSelect(sqlxDB, &shardingDB, q)
	if err != nil {
		log.Fatal(err)
	}
	return
}

func sqlxSelectPtrQuery(sqlxDB *sqlx.DB, lim int) (shardingDB *[]ShardingDB) {
	q := fmt.Sprintf("SELECT * FROM sharding_db ORDER BY id LIMIT %d", lim)
	err := SqlxSelect(sqlxDB, shardingDB, q)
	if err != nil {
		log.Fatal(err)
	}
	return
}

封装举例

package database

import "github.com/jmoiron/sqlx"

var (
   SqlxSelect = sqlxSelectFunc
)

// ================== raw execute func ==================
func sqlxSelectFunc(db *sqlx.DB, dest interface{}, query string, args ...interface{}) error {
   query, args, err := sqlx.In(query, args...)
   if err != nil {
      return err
   }
   err = db.Select(dest, query, args...)
   return err
}

测试结果

=============================== CPU:8 MEM:16G MaxOpenConns:500 MaxIdleConns:100 ====================================
goos: linux
goarch: amd64
pkg: go-operation-demo/sqlx/bench-test-demo/pkg-comp
Benchmark/sqlx_query_limit:5-8         	   12006	     99514 ns/op	    7276 B/op	     134 allocs/op
Benchmark/sqlx_select_query_limit:5-8  	   10000	    101496 ns/op	    7276 B/op	     134 allocs/op
==================================================================================================================
Benchmark/sqlx_query_limit:50-8        	    4536	    263568 ns/op	   52654 B/op	    1082 allocs/op
Benchmark/sqlx_select_query_limit:50-8 	    4440	    264001 ns/op	   52655 B/op	    1082 allocs/op
==================================================================================================================
Benchmark/sqlx_query_limit:500-8       	     835	   1447305 ns/op	  480066 B/op	   10536 allocs/op
Benchmark/sqlx_select_query_limit:500-8         	     826	   1446552 ns/op	  480066 B/op	   10536 allocs/op
==================================================================================================================
Benchmark/sqlx_query_limit:10000-8              	      40	  26950958 ns/op	14964317 B/op	  210055 allocs/op
Benchmark/sqlx_select_query_limit:10000-8       	      40	  26891873 ns/op	14964402 B/op	  210054 allocs/op
==================================================================================================================
PASS
ok  	go-operation-demo/sqlx/bench-test-demo/pkg-comp	10.580s
上一篇:【REACT NATIVE 系列教程之一】触摸事件的两种形式与四种TOUCHABLE组件详解


下一篇:【Hadoop报错】The directory item limit of /tmp/hadoop-yarn/staging/history/done_intermediate/hdfs is exc