日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
使用SpringBoot和Rust生成二維碼的性能比較(附代碼)

本文重點比較使用虛擬線程的SpringBoot和使用Actix框架的Rust,來實現(xiàn)QR碼生成器API。這兩種技術(shù)都是成熟的,無需進一步介紹。接下來,讓我們直接深入測試設置的細節(jié)。

成都創(chuàng)新互聯(lián)公司是專業(yè)的揚州網(wǎng)站建設公司,揚州接單;提供網(wǎng)站建設、網(wǎng)站設計,網(wǎng)頁設計,網(wǎng)站設計,建網(wǎng)站,PHP網(wǎng)站建設等專業(yè)做網(wǎng)站服務;采用PHP框架,可快速的進行揚州網(wǎng)站開發(fā)網(wǎng)頁制作和功能擴展;專業(yè)做搜索引擎喜愛的網(wǎng)站,專業(yè)的做網(wǎng)站團隊,希望更多企業(yè)前來合作!

一、測試設置

1. 環(huán)境

所有測試都在裝有16GB RAM的MacBook Pro M1上進行。使用的測試工具是Bombardier的定制版本,支持在請求體中包含隨機URL。這些測試使用的軟件版本如下:

  • SpringBoot 3.1.3,帶有Java v20(啟用預覽以獲取虛擬線程)
  • Rust 1.72.0

2. 代碼

這個QR碼生成器應用程序被設計成接收一個JSON請求體,其中包含一個名為"urlToEmbed"的必需參數(shù)。該應用程序的主要功能是為指定的URL生成一個QR碼,并在HTTP響應中以PNG格式傳送QR碼。為增加復雜性,該應用程序在HTTPS上運行。

(1) SpringBoot(虛擬線程)

server.port=3000
server.ssl.certificate=/Users/mayankc/Work/source/certs/cert.pem
server.ssl.certificate-private-key=/Users/mayankc/Work/source/certs/key.pem
package com.example.qr;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.embedded.tomcat.TomcatProtocolHandlerCustomizer;
import org.springframework.context.annotation.Bean;
import java.util.concurrent.Executors;

@SpringBootApplication
public class QrApplication {

  public static void main(String[] args) {
    SpringApplication.run(QrApplication.class, args);
  }

  @Bean
  public TomcatProtocolHandlerCustomizer protocolHandlerVirtualThreadExecutorCustomizer() {
    return protocolHandler -> {
      protocolHandler.setExecutor(Executors.newVirtualThreadPerTaskExecutor());
    };
  }
}
package com.example.qr;

import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.http.ResponseEntity;
import org.springframework.http.HttpStatus;
import org.springframework.http.HttpHeaders;
import org.springframework.web.bind.annotation.RestController;
import java.util.Optional;
import com.example.qr.QrRequest;
import com.example.qr.QrGenerator;

@RestController
public class QrController {

  @PostMapping("/qr")
  public ResponseEntity handleRequest(@RequestBody QrRequest qrRequest) {
    if(qrRequest.getUrlToEmbed() == null) {
      return new ResponseEntity<>(HttpStatus.BAD_REQUEST);
    }

    try {
      HttpHeaders httpHeaders = new HttpHeaders();
      httpHeaders.add(HttpHeaders.CONTENT_TYPE, "image/png");
      return new ResponseEntity(
         QrGenerator.generateQR(qrRequest.getUrlToEmbed(), 512, 512), 
         httpHeaders,
         HttpStatus.OK);
    } catch (Exception e) {
      return new ResponseEntity<>(HttpStatus.INTERNAL_SERVER_ERROR);
    }
  }
}
package com.example.qr;

import java.io.ByteArrayOutputStream;
import java.io.IOException;

import com.google.zxing.BarcodeFormat;
import com.google.zxing.WriterException;
import com.google.zxing.client.j2se.MatrixToImageConfig;
import com.google.zxing.client.j2se.MatrixToImageWriter;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.QRCodeWriter;

public class QrGenerator {

  public static byte[] generateQR(String text, int width, int height) throws WriterException, IOException {
    QRCodeWriter qrCodeWriter = new QRCodeWriter();
    BitMatrix bitMatrix = qrCodeWriter.encode(text, BarcodeFormat.QR_CODE, width, height);

    ByteArrayOutputStream pngOutputStream = new ByteArrayOutputStream();
    MatrixToImageConfig con = new MatrixToImageConfig() ;

    MatrixToImageWriter.writeToStream(bitMatrix, "PNG", pngOutputStream, con);
    byte[] pngData = pngOutputStream.toByteArray();
    return pngData;
  }
}
package com.example.qr;

public class QrRequest {
  private String urlToEmbed;

  public String getUrlToEmbed() {
    return this.urlToEmbed;
  }

  public void setUrlToEmbed(String urlToEmbed) {
    this.urlToEmbed = urlToEmbed;
  }
}

(2) Rust

[package]
name = "actix_qr_generator"
version = "0.1.0"
edition = "2021"

[dependencies]
actix-web = { version = "4", features = ["openssl"] } 
qrcode-generator = "4.1.8"
serde = { version = "1.0", features = ["derive"] }
serde_json = "1"
openssl = { version = "0.10" , features = ["vendored"] }
use actix_web::{web, post, App, HttpServer, HttpResponse, Responder};
use qrcode_generator::QrCodeEcc;
use serde::Deserialize;
use openssl::ssl::{SslAcceptor, SslFiletype, SslMethod};

#[derive(Deserialize)]
struct QrRequest {
  urlToEmbed: String,
}

#[post("/qr")]
async fn generate_qr(qr_request: web::Json) -> impl Responder {
  if qr_request.urlToEmbed.is_empty() {
    return HttpResponse::BadRequest().into();
  }


  let result: Vec = qrcode_generator::to_png_to_vec(qr_request.urlToEmbed.clone(), QrCodeEcc::Low, 512)
    .unwrap();
  return HttpResponse::Ok()
    .content_type("image/png")
    .body(result);
}

#[actix_web::main]
async fn main() -> std::io::Result<()> {
  let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls()).unwrap();
  builder
    .set_private_key_file("/Users/mayankc/Work/source/perfComparisons/certs/key.pem", SslFiletype::PEM)
    .unwrap();
  builder
    .set_certificate_chain_file("/Users/mayankc/Work/source/perfComparisons/certs/cert.pem")
    .unwrap();

  HttpServer::new(|| App::new().service(generate_qr))
    .bind_openssl("127.0.0.1:3000", builder)?
    .run()
    .await
}


// 注意 ================================================
// 該應用程序已在發(fā)布模式下構(gòu)建。
// =====================================================

二、結(jié)果

為了全面評估性能,這里進行了一系列細致的檢查。每個檢查包括10萬個請求,并在10、50和100個并發(fā)連接的范圍內(nèi)評估它們的效率??紤]到QR碼生成的資源密集型特性,故意保持了稍低的請求量,與其他場景相比。

結(jié)果如下:

根據(jù)以下公式,還生成了一個得分卡。對于每個測量,獲取獲勝的差距。如果獲勝的差距是:

  • < 5%,不給予任何分數(shù)
  • 在5%到20%之間,獲勝者得1分
  • 在20%到50%之間,獲勝者得2分
  • 50%,獲勝者得3分

得分卡如下:


本文題目:使用SpringBoot和Rust生成二維碼的性能比較(附代碼)
分享鏈接:http://www.dlmjj.cn/article/dpjeoco.html