ใช้ FastAPI ให้เร็วขึ้นถึงเกือบเท่า Go Gin

จากบทความก่อนหน้านี้

Python ก็เร็วเท่า GoLang ได้ จริงไหม?

เรามาทำให้ FastAPI ของเราให้เร็วขึ้นถึงเกือบเท่า Go Gin ในบทความนี้ จะทำการใช้ Docker เข้ามาช่วยในการทดสอบความเร็วในการประมวลผล ระหว่าง Python3.11 pypy3.10 และ Go1.20.6 นะครับ

โดย Docker จะถึงตั้งค่าไว้แบบเดี่ยวกัน และใช้ โค๊ต การทำงานแบบเดียวกัน

โค๊ตที่ใช้ในการทดสอบด้วย FastAPI

import timeit
import platform
from fastapi import FastAPI

def make_tree(d):
    if d > 0:
        d -= 1
        return (make_tree(d), make_tree(d))
    return (None, None)

def check_tree(node):
    (l, r) = node
    return 1 if l is None else 1 + check_tree(l) + check_tree(r)


def make_check(d, make=make_tree, check=check_tree):
    return check(make(d))

def main(n, min_depth=4):
    max_depth = max(min_depth + 2, n)
    stretch_depth = max_depth + 1
    print('stretch tree of depth {0}\t check: {1}'.format(
          stretch_depth, make_check(stretch_depth)))

    long_lived_tree = make_tree(max_depth)

    mmd = max_depth + min_depth
    for d in range(min_depth, stretch_depth, 2):
        i = 2 ** (mmd - d)
        cs = sum(make_check(d) for _ in range(i))
        print('{0}\t trees of depth {1}\t check: {2}'.format(i, d, cs))

    print('long lived tree of depth {0}\t check: {1}'.format(
          max_depth, check_tree(long_lived_tree)))


app = FastAPI()

@app.get("/")
async def Home():
    starttime = timeit.default_timer()
    n = 18
    d = 4
    print("Tree recursion ==> ",n)
    print("Python Version ==> ",platform.python_version())
    main(n,d)
    print("The time difference is :", timeit.default_timer() - starttime," seconds")
    return {}

โค๊ตที่ใช้ในการทดสอบด้วย Go gin

package main

import (
	"fmt"
	"math"
	"net/http"
	"runtime"
	"strings"
	"time"

	"github.com/gin-gonic/gin"
)

type node struct {
	left, right *node
}

func makeTree(d int) *node {
	if d > 0 {
		d--
		return &node{left: makeTree(d), right: makeTree(d)}
	}
	return nil
}

func checkTree(node *node) int {
	if node == nil {
		return 0
	}
	return 1 + checkTree(node.left) + checkTree(node.right)
}

func makeCheck(d int) int {
	return checkTree(makeTree(d))
}

func run(n, minDepth int) {
	maxDepth := int(math.Max(float64(minDepth+2), float64(n)))
	stretchDepth := maxDepth + 1

	fmt.Printf("stretch tree of depth %d\t check: %d\n", stretchDepth, makeCheck(stretchDepth))

	longLivedTree := makeTree(maxDepth)

	mmd := maxDepth + minDepth
	for d := minDepth; d <= maxDepth; d += 2 {
		i := int(math.Pow(2, float64(mmd-d)))
		cs := 0
		for j := 0; j < i; j++ {
			cs += makeCheck(d)
		}
		fmt.Printf("%d\t trees of depth %d\t check: %d\n", i, d, cs)
	}

	fmt.Printf("long lived tree of depth %d\t check: %d\n", maxDepth, checkTree(longLivedTree))
}

func homeHandler(c *gin.Context) {
	startTime := time.Now()
	n := 18
	d := 4
	fmt.Println("Tree recursion ==> ", n)
	fmt.Println("Go Version ==> ", runtime.Version())
	run(n, d)
	fmt.Println("The time difference is :", time.Since(startTime), " seconds")

	c.JSON(http.StatusOK, gin.H{})
}

func getRealIP(c *gin.Context) string {
	// Check X-Real-IP first, then X-Forwarded-For.
	// X-Forwarded-For may return multiple IP addresses, and the original client IP will be the first one.
	xRealIP := c.Request.Header.Get("X-Real-IP")
	xForwardedFor := c.Request.Header.Get("X-Forwarded-For")

	if xRealIP != "" {
		return xRealIP
	} else if xForwardedFor != "" {
		return strings.Split(xForwardedFor, ",")[0]
	}

	return c.Request.RemoteAddr
}

func main() {
	r := gin.Default()

	r.GET("/", homeHandler)

	if err := r.Run(":8080"); err != nil {
		fmt.Println("Error starting server:", err)
	}
}

docker-compose.yaml

version: "3.9"
services:
  pypy-test:
    ports:
      - 8000:8000
    build:
      context: .
      dockerfile: Dockerfile-pypy
  python-test:
    ports:
      - 8001:8000
    build:
      context: .
      dockerfile: Dockerfile-python
  golang-test:
    ports:
      - 8002:8080
    build:
      context: .
      dockerfile: Dockerfile-golang

Dockerfile-python == Python3.11

FROM python:3.11-slim

WORKDIR /app

COPY . .

RUN python -m pip install --no-cache-dir -r requirements.txt

EXPOSE 8000

CMD ["python", "-m","uvicorn", "app2:app","--host", "0.0.0.0","--port","8000"]

Dockerfile-pypy == pypy3.10

FROM pypy:3.10-slim

WORKDIR /app

COPY . .

RUN pypy -m pip install --no-cache-dir -r requirements.txt

EXPOSE 8000

CMD ["pypy","-m","uvicorn", "app2:app","--host", "0.0.0.0","--port","8000"]

Dockerfile-golang == go1.20.6

FROM golang:1.20-alpine

WORKDIR /app

COPY go.mod go.sum ./

RUN go mod download

COPY app.go .

RUN go build app.go

EXPOSE 8080

CMD ["./app"]


Docker Golang (0.9 วินาที)
Docker pypy3.10 (1.3 วินาที)
Docker python3.11 (6.4 วินาที)

จะเห็นได้ว่าเมื่อใช้งานผ่าน Docker แล้ว Golang 1.20.6 เร็วกว่า pypy3.10 นิดหน่อย เพียง 300-400ms เท่านั้น ในทาง API แล้วถือว่าเป็นทางเลือกที่ดีเลยทีเดี่ยวสำหรับคนที่เขียน Python ด้วย FastAPI

Read more

iOS 27: เตรียมพบกับ 3 เครื่องมือ AI แต่งภาพสุดล้ำในแอป Photos ทั้ง Extend, Enhance และ Reframe

iOS 27: เตรียมพบกับ 3 เครื่องมือ AI แต่งภาพสุดล้ำในแอป Photos ทั้ง Extend, Enhance และ Reframe

เตรียมพบกับยุคใหม่แห่งการแต่งภาพบน iPhone ด้วย iOS 27! Apple จ่อเพิ่ม AI Tools ใหม่ในแอป Photos ทั้ง Extend, Enhance, Reframe ให้คุณสร้างสรรค์ภาพสวยง่ายดาย

By ทีมงาน devdog
ฉลอง 20 ปี Google Translate เปิดตัวฟีเจอร์ AI ฝึกออกเสียงเรียลไทม์ตามคำเรียกร้อง!

ฉลอง 20 ปี Google Translate เปิดตัวฟีเจอร์ AI ฝึกออกเสียงเรียลไทม์ตามคำเรียกร้อง!

Google Translate ฉลอง 20 ปี! เปิดตัวฟีเจอร์ AI ช่วยฝึกออกเสียงแบบเรียลไทม์ ตอบโจทย์คนอยากเก่งภาษา พร้อมวิเคราะห์และให้คำแนะนำทันที

By ทีมงาน devdog
PPV คืออะไร? เจาะลึกปรากฏการณ์ Pay-Per-View กับอีเวนต์สุดพิเศษแห่งยุค

PPV คืออะไร? เจาะลึกปรากฏการณ์ Pay-Per-View กับอีเวนต์สุดพิเศษแห่งยุค

ทำความเข้าใจ Pay-Per-View (PPV) กับเทรนด์การรับชมอีเวนต์สุดพิเศษ ทั้งศึก ONE Championship, คอนเสิร์ต Project Sekai และความบันเทิงหลากหลายผ่าน ABEMA PPV.

By ทีมงาน devdog