使用Spring Boot和Rust生成二维码的性能比较(附代码)

开发
本文重点比较使用虚拟线程的SpringBoot和使用Actix框架的Rust,来实现QR码生成器API。

本文重点比较使用虚拟线程的SpringBoot和使用Actix框架的Rust,来实现QR码生成器API。这两种技术都是成熟的,无需进一步介绍。接下来,让我们直接深入测试设置的细节。

一、测试设置

1. 环境

所有测试都在装有16GB RAM的MacBook Pro M1上进行。使用的测试工具是Bombardier的定制版本,支持在请求体中包含随机URL。这些测试使用的软件版本如下:

  • SpringBoot 3.1.3,带有Java v20(启用预览以获取虚拟线程)
  • Rust 1.72.0

2. 代码

这个QR码生成器应用程序被设计成接收一个JSON请求体,其中包含一个名为"urlToEmbed"的必需参数。该应用程序的主要功能是为指定的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<byte[]>(
         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<QrRequest>) -> impl Responder {
  if qr_request.urlToEmbed.is_empty() {
    return HttpResponse::BadRequest().into();
  }


  let result: Vec<u8> = 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
}


// 注意 ================================================
// 该应用程序已在发布模式下构建。
// =====================================================

二、结果

为了全面评估性能,这里进行了一系列细致的检查。每个检查包括10万个请求,并在10、50和100个并发连接的范围内评估它们的效率。考虑到QR码生成的资源密集型特性,故意保持了稍低的请求量,与其他场景相比。

结果如下:

根据以下公式,还生成了一个得分卡。对于每个测量,获取获胜的差距。如果获胜的差距是:

  • < 5%,不给予任何分数
  • 在5%到20%之间,获胜者得1分
  • 在20%到50%之间,获胜者得2分
  • 50%,获胜者得3分

得分卡如下:

责任编辑:赵宁宁 来源: Java学研大本营
相关推荐

2023-11-17 15:44:06

C++库生成二维码

2023-11-17 09:07:51

.NET生成二维码识别二维码

2013-10-29 09:46:05

二维码

2015-09-24 09:56:19

.NET二维码

2013-03-27 16:13:17

Android开发Android二维码生QR生成

2013-01-30 12:16:49

二维码NFC近场通讯

2023-10-31 08:16:16

Go语言二维码

2014-12-31 15:37:16

二维码生成扫描

2020-12-30 10:55:01

代码Java二维码

2017-11-21 09:00:15

PHP二维码LOGO图像

2020-08-07 14:18:03

二维码Go语言编程

2021-02-08 08:40:00

Zint二维码开源

2015-01-14 09:46:52

Google API

2011-12-30 17:09:41

二维码APP灵动快拍

2015-11-09 13:03:47

JavaServlet创建二维码

2017-02-21 09:17:46

二维码漏洞

2012-04-01 09:53:13

二维码

2011-12-06 16:40:45

二维码快拍二维码灵动快拍

2011-11-24 16:00:23

信息图QR码二维码

2022-03-24 09:43:29

二维码二维码修改器github
点赞
收藏

51CTO技术栈公众号