实战:Zig 编写高性能 Web 服务(1)

2024-06-03 19:04

本文主要是介绍实战:Zig 编写高性能 Web 服务(1),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.1  认识 std.http

std.http 是 Zig 标准库中用于处理 HTTP 相关操作的类库。以我学习新的编程语言的经历来看,编写web程序是最常见的技术场景,所以熟练掌握 HTTP server/client 服务相关的编程知识是比较重要的。

std.http 主要包含以下API:

  • Client: HTTP client implementation.
  • Server: HTTP server implementation.
  • protocol:headers parse methods.
  • Connection: Connection type (keep_aliveclose)
  • ContentEncoding: Content encoding options (compressdeflategzip and zstd)
  • Field: Common type for name and value
  • Headers: HTTP headers
  • Method: HTTP methods such as GET and POST
  • Status: HTTP status codes (not_found = 404teapot = 418, etc.)
  • TransferEncoding: Form of encoding used to transfer the body (chunked)
  • Version: Currently HTTP/1.0 and HTTP/1.1

1.2 编写一个HTTP client程序

先创建一个开工项目:

$ mkdir -p httpz
$ cd httpz
$ zig init$ ls -ls
total 20
4 -rw-r--r-- 1 xiaods xiaods 3879 Jun  3 11:53 build.zig
4 -rw-r--r-- 1 xiaods xiaods 3080 Jun  3 11:50 build.zig.zon
4 drwxr-xr-x 2 xiaods xiaods 4096 Jun  3 13:33 src
4 drwxr-xr-x 6 xiaods xiaods 4096 Jun  3 11:51 zig-cache
4 drwxr-xr-x 4 xiaods xiaods 4096 Jun  3 11:51 zig-out

编辑 src/main.zig,我们将使用 std.heap.GeneralPurposeAllocator,这是一个安全的分配器,可以防止双重释放(double-free)、使用后释放(use-after-free),并且能够检测内存泄漏。

const std = @import("std");
const print = std.debug.print;
const http = std.http;var gpa = std.heap.GeneralPurposeAllocator(.{}){};defer _ = gpa.deinit();const allocator = gpa.allocator();

下一步,为了发送一个请求,我们需要几样东西:

  • client.open 函数
  • 一个从URL解析而来的 std.Uri

下面是我们如何将这些参数组合在一起的方法:

    const uri = try std.Uri.parse("http://httpbin.org/headers");const buf = try allocator.alloc(u8, 1024 * 1024 * 4);defer allocator.free(buf);var req = try client.open(.GET, uri, .{.server_header_buffer = buf,});defer req.deinit();

为了真正的发送请求,需要通过send,finish,wait来完成:

 try req.send();try req.finish();try req.wait();

打印返回的服务器headers 信息:

var iter = req.response.iterateHeaders();while (iter.next()) |header| {std.debug.print("Name:{s}, Value:{s}\n", .{ header.name, header.value });}try std.testing.expectEqual(req.response.status, .ok);

打印返回的服务端内容:

    var rdr = req.reader();const body = try rdr.readAllAlloc(allocator, 1024 * 1024 * 4);defer allocator.free(body);print("Body:\n{s}\n", .{body});

把上面的代码所有内容放在一起,并打印出响应内容:

const std = @import("std");
const print = std.debug.print;
const http = std.http;pub fn main() !void {var gpa = std.heap.GeneralPurposeAllocator(.{}){};defer _ = gpa.deinit();const allocator = gpa.allocator();var client = http.Client{ .allocator = allocator };defer client.deinit();const uri = try std.Uri.parse("http://httpbin.org/headers");const buf = try allocator.alloc(u8, 1024 * 1024 * 4);defer allocator.free(buf);var req = try client.open(.GET, uri, .{.server_header_buffer = buf,});defer req.deinit();try req.send();try req.finish();try req.wait();var iter = req.response.iterateHeaders();while (iter.next()) |header| {std.debug.print("Name:{s}, Value:{s}\n", .{ header.name, header.value });}try std.testing.expectEqual(req.response.status, .ok);var rdr = req.reader();const body = try rdr.readAllAlloc(allocator, 1024 * 1024 * 4);defer allocator.free(body);print("Body:\n{s}\n", .{body});
}

跑一下:

 $ zig build run
Name:Date, Value:Mon, 03 Jun 2024 08:24:19 GMT
Name:Content-Type, Value:application/json
Name:Content-Length, Value:202
Name:Connection, Value:keep-alive
Name:Server, Value:gunicorn/19.9.0
Name:Access-Control-Allow-Origin, Value:*
Name:Access-Control-Allow-Credentials, Value:true
Body:
{"headers": {"Accept-Encoding": "gzip, deflate","Host": "httpbin.org","User-Agent": "zig/0.12.0 (std.http)","X-Amzn-Trace-Id": "Root=1-665d7db3-258c846d0fcca0912fadfa8b"}
}

成功了!我们成功地向服务器发送了一个GET请求并打印出了响应。

GET请求的例子我们看到了,那么如何发起POST请求呢?让我们继续拿例子说话。

准备好发送内容:

const uri = try std.Uri.parse("http://httpbin.org/anything");const payload =\\ {\\  "name": "zig-learning",\\  "author": "xiaods"\\ };

发送POST 请求:

var buf: [1024]u8 = undefined;var req = try client.open(.POST, uri, .{ .server_header_buffer = &buf });defer req.deinit();req.transfer_encoding = .{ .content_length = payload.len };try req.send();var wtr = req.writer();try wtr.writeAll(payload);try req.finish();try req.wait();try std.testing.expectEqual(req.response.status, .ok);

打印返回内容:

var rdr = req.reader();const body = try rdr.readAllAlloc(allocator, 1024 * 1024 * 4);defer allocator.free(body);print("Body:\n{s}\n", .{body});

完整的Post代码如下:

const std = @import("std");
const print = std.debug.print;
const http = std.http;pub fn main() !void {var gpa = std.heap.GeneralPurposeAllocator(.{}){};defer _ = gpa.deinit();const allocator = gpa.allocator();var client = http.Client{ .allocator = allocator };defer client.deinit();const uri = try std.Uri.parse("http://httpbin.org/anything");const payload =\\ {\\  "name": "zig-learning",\\  "author": "xiaods"\\ };var buf: [1024]u8 = undefined;var req = try client.open(.POST, uri, .{ .server_header_buffer = &buf });defer req.deinit();req.transfer_encoding = .{ .content_length = payload.len };try req.send();var wtr = req.writer();try wtr.writeAll(payload);try req.finish();try req.wait();try std.testing.expectEqual(req.response.status, .ok);var rdr = req.reader();const body = try rdr.readAllAlloc(allocator, 1024 * 1024 * 4);defer allocator.free(body);print("Body:\n{s}\n", .{body});
}

 运行结果:

$ zig run src/http-post.zig 
Body:
{"args": {}, "data": " {\n  \"name\": \"zig-learning\",\n  \"author\": \"xiaods\"\n }", "files": {}, "form": {}, "headers": {"Accept-Encoding": "gzip, deflate", "Content-Length": "52", "Host": "httpbin.org", "User-Agent": "zig/0.12.0 (std.http)", "X-Amzn-Trace-Id": "Root=1-665d8114-01b0167844d8d101012e6d6a"}, "json": {"author": "xiaods", "name": "zig-learning"}, "method": "POST", "origin": "219.133.170.77", "url": "http://httpbin.org/anything"
}

请消化消化以上代码,别着急,我们后面继续前行,编写web server

这篇关于实战:Zig 编写高性能 Web 服务(1)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1027847

相关文章

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

Java MQTT实战应用

《JavaMQTT实战应用》本文详解MQTT协议,涵盖其发布/订阅机制、低功耗高效特性、三种服务质量等级(QoS0/1/2),以及客户端、代理、主题的核心概念,最后提供Linux部署教程、Sprin... 目录一、MQTT协议二、MQTT优点三、三种服务质量等级四、客户端、代理、主题1. 客户端(Clien

Linux中SSH服务配置的全面指南

《Linux中SSH服务配置的全面指南》作为网络安全工程师,SSH(SecureShell)服务的安全配置是我们日常工作中不可忽视的重要环节,本文将从基础配置到高级安全加固,全面解析SSH服务的各项参... 目录概述基础配置详解端口与监听设置主机密钥配置认证机制强化禁用密码认证禁止root直接登录实现双因素

在Spring Boot中集成RabbitMQ的实战记录

《在SpringBoot中集成RabbitMQ的实战记录》本文介绍SpringBoot集成RabbitMQ的步骤,涵盖配置连接、消息发送与接收,并对比两种定义Exchange与队列的方式:手动声明(... 目录前言准备工作1. 安装 RabbitMQ2. 消息发送者(Producer)配置1. 创建 Spr

java向微信服务号发送消息的完整步骤实例

《java向微信服务号发送消息的完整步骤实例》:本文主要介绍java向微信服务号发送消息的相关资料,包括申请测试号获取appID/appsecret、关注公众号获取openID、配置消息模板及代码... 目录步骤1. 申请测试系统2. 公众号账号信息3. 关注测试号二维码4. 消息模板接口5. Java测试

深度解析Spring Boot拦截器Interceptor与过滤器Filter的区别与实战指南

《深度解析SpringBoot拦截器Interceptor与过滤器Filter的区别与实战指南》本文深度解析SpringBoot中拦截器与过滤器的区别,涵盖执行顺序、依赖关系、异常处理等核心差异,并... 目录Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现

深度解析Spring AOP @Aspect 原理、实战与最佳实践教程

《深度解析SpringAOP@Aspect原理、实战与最佳实践教程》文章系统讲解了SpringAOP核心概念、实现方式及原理,涵盖横切关注点分离、代理机制(JDK/CGLIB)、切入点类型、性能... 目录1. @ASPect 核心概念1.1 AOP 编程范式1.2 @Aspect 关键特性2. 完整代码实

MySQL中的索引结构和分类实战案例详解

《MySQL中的索引结构和分类实战案例详解》本文详解MySQL索引结构与分类,涵盖B树、B+树、哈希及全文索引,分析其原理与优劣势,并结合实战案例探讨创建、管理及优化技巧,助力提升查询性能,感兴趣的朋... 目录一、索引概述1.1 索引的定义与作用1.2 索引的基本原理二、索引结构详解2.1 B树索引2.2

从入门到精通MySQL 数据库索引(实战案例)

《从入门到精通MySQL数据库索引(实战案例)》索引是数据库的目录,提升查询速度,主要类型包括BTree、Hash、全文、空间索引,需根据场景选择,建议用于高频查询、关联字段、排序等,避免重复率高或... 目录一、索引是什么?能干嘛?核心作用:二、索引的 4 种主要类型(附通俗例子)1. BTree 索引(

如何使用Maven创建web目录结构

《如何使用Maven创建web目录结构》:本文主要介绍如何使用Maven创建web目录结构的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录创建web工程第一步第二步第三步第四步第五步第六步第七步总结创建web工程第一步js通过Maven骨架创pytho