当前位置: 首页 > news >正文

一种基于宏和serde_json实现的rust web中统一返回类

本人rust萌新,写web碰到了这个,基于ChatGPT和文心一言学了宏,强行把这玩意实现出来了,做个学习记录,如果有更好的方法,勿喷。

先看效果,注意不支持嵌套,且kv映射要用=>(因为它这个只支持用箭头),即在这个宏语法内只支持单层kv,但是你可以传value为HashMap。
对此我曾尝试用#[proc_macro]TokenStream拿到变量名和值的方式实现正常使用花括号和冒号和深层字典,但是太菜了搞不会。
如果能基于我的想法实现出深层kv,欢迎评论区分享。

//第一个变量是msg,后面的全部存入data{key1:...,key2:...}
success!("success","key1"=> vec![1,2,3,4,5],"key2" =>"qwer","key3"=>String::from("value")
);

在这里插入图片描述

//第一个变量没了就是默认空msg,后面的全部存入data{key1:...,key2:...}
success!("key"=>"qwer","key2"=>1324,
);

在这里插入图片描述
fail!同理

rust中实现统一返回类有点麻烦,比如我想返回固定格式

{"success":true,"msg":"登录成功","data":{"id":15,"name":"qwer"}
}

我目前用的是actix-web,虽然支持直接传入结构体对象作为json返回值,但我有时候想对于data进行灵活的处理,不想建那么多结构体

这里我基于serde_jsonjson!进行进一步封装
这里一个宏里面写两个形式,因为要匹配传入msg和不传入msg的两种情形。最后一个$(,)?是允许最后一个多余的逗号,换行时好看一点

// 随便找个文件放
// 用了#[macro_export]的宏会直接放在crate下//下面三个库需要在使用下面宏的地方进行use,此处use没用
// use actix_web::web;
// use serde_json::json;
// use std::collections::HashMap;#[macro_export]
macro_rules! success {// 没有msg的情况($($key:expr => $value:expr),* $(,)?) => {{let mut resp = HashMap::new();resp.insert("success", json!(true));resp.insert("msg", json!("")); // 提供一个默认的消息let mut data:HashMap<String,Value> = HashMap::new();$(data.insert($key, json!($value));)*resp.insert("data",json!(data));web::Json(json!(resp))}};// 有msg的情况($msg:expr,  $($key:expr => $value:expr),* $(,)?) => {{let mut resp = HashMap::new();resp.insert("success", json!(true));resp.insert("msg", json!($msg)); // 提供一个默认的消息let mut data:HashMap<String,Value> = HashMap::new();$(data.insert($key, json!($value));)*resp.insert("data",json!(data));web::Json(json!(resp))}};
}#[macro_export]
macro_rules! fail {// 没有$msg的情况($($key:expr => $value:expr),* $(,)?) => {{let mut resp = HashMap::new();resp.insert("success", json!(false));resp.insert("msg", json!(""));let mut data:HashMap<String,Value> = HashMap::new();$(data.insert($key, json!($value));)*resp.insert("data",data);web::Json(json!(resp))}};// 有$msg的情况($msg:expr,  $($key:expr => $value:expr),* $(,)?) => {{let mut resp = HashMap::new();resp.insert("success", json!(false));resp.insert("msg", json!($msg));let mut data:HashMap<String,Value> = HashMap::new();$(data.insert($key, json!($value));)*resp.insert("data",json!(data));web::Json(json!(resp))}};
}

注意:过程宏需要定义在一个单独的crate中,主要是因为过程宏是一段在编译crate前,对其代码进行加工的代码,而这段是需要在编译后执行的。若是将定义过程宏和使用过程宏放到同一个crate中,就会陷入编译“死锁”:

另一个文件对其进行调用

// 注意是在crate下面的
use crate::success;
use actix_web::{get,web::{self, Json},
};
use serde_json::{json, Value};
use std::collections::HashMap;#[get("/login")]
async fn login() -> Json<Value> {let resp = success!("success","key1"=> vec![1,2,3,4,5],"key2" =>"qwer","key3"=>String::from("value"));resp
}

我理解的这个宏就是字符串替换,因为他是编译阶段处理的,所以宏所在rs文件的use是无效的,即使用了宏的文件需要对于宏使用的第三方库进行use,这个有点恶心,问题不大。

http://www.lryc.cn/news/317896.html

相关文章:

  • 每周一算法:A*(A Star)算法
  • 爬虫练习:获取某网站的房价信息
  • 第一个C语言hello world
  • 【Python】新手入门学习:详细介绍依赖倒置原则(DIP)及其作用、代码示例
  • 嵌入式驱动学习目录索引(更新中)
  • ruoyi-vue插件集成websocket
  • 华为ce12800交换机m-lag(V-STP模式)配置举例
  • STM32第九节(中级篇):RCC——时钟树讲解(第一节)
  • c/c++字符串处理标准库 string 介绍
  • HarmonyOS NEXT应用开发之深色模式适配
  • Go微服务: 基于Go Micro框架实现微服务调用
  • 大模型prompt提示词如何调优?
  • 【Python/crawl】如何使用Python爬虫将一系列网页上的同类图片下载到本地
  • Postgresql 连接数查看,死锁问题解决
  • ssm蛋糕甜品商城系统(程序+文档+数据库)
  • 算法空间复杂度计算
  • C++ lambda函数个人理解
  • SwiftUI的context Menu
  • 【数据结构】树与堆 (向上/下调整算法和复杂度的分析、堆排序以及topk问题)
  • 安装CDH平台的服务器磁盘满了,磁盘清理过程记录
  • 《互联网的世界》第七讲-能源
  • 前端代码整洁与规范之CSS篇
  • 在【IntelliJ IDEA】中配置【Tomcat】【2023版】【中文】【图文详解】
  • 【SSM】任务列表案例 基本CRUD SSM整合
  • 基于微信小程序的校园跑腿小程序,附源码
  • 网络学习:9个计算机的“网络层”知识点
  • web项目的搭建
  • C++for语句
  • 最新基于R语言lavaan结构方程模型(SEM)技术
  • 【网络安全】-数字证书