Rust Web入门(七):WebAssembly

news2025/1/25 4:38:06

本教程笔记来自 杨旭老师的 rust web 全栈教程,链接如下:

https://www.bilibili.com/video/BV1RP4y1G7KF?p=1&vd_source=8595fbbf160cc11a0cc07cadacf22951

学习 Rust Web 需要学习 rust 的前置知识可以学习杨旭老师的另一门教程

https://www.bilibili.com/video/BV1hp4y1k7SV/?spm_id_from=333.999.0.0&vd_source=8595fbbf160cc11a0cc07cadacf22951

项目的源代码可以查看 git:(注意作者使用的是 mysql 数据库而不是原教程的数据库)

https://github.com/aiai0603/rust_web_mysql

在之前的项目中,我们使用 rust 的模板引擎 Tera 编写一个项目,这节课我们来介绍一种 Rust 提供的更加高级的功能来编写前端应用—— WebAssembly,这是官方文档

https://rustwasm.github.io/docs/book/

WebAssembly

WebAssembly/wasm WebAssembly 或者 wasm 是一个可移植、体积小、加载快并且兼容 Web 的全新格式。

  • 他是一种低级的类汇编语言
  • 具有紧凑的二进制格式
  • 可以以接近原生的性能运行
  • 它可以为 C/C++ Rust 提供一个编译目标,以便它们可以在 Web 上运行
  • 它 被设计为与 js 共存,一起工作

通过它,我们可以将我们编写的其他语言(C++,Rust)语言编译成 WebAssembly 模块,然后在 Web 应用中加载这些模块,在 JS 中调用它。

WebAssembly 的优势是:

  • 快速高效可以移植,WebAssembly 代码可以在不同的平台上以接近本地的速度运行
  • 可读可调试,它虽然是低级语言,但是有一种人类可读的文本格式
  • 安全,限制运行在安全的沙箱中,也遵循浏览器的同源策略和授权策略
  • 不破坏网络,它与其他网络技术共存并且保持向后兼容

配置环境

在开始我们的项目编写之前,我们需要先配置 rust-wasm 的环境,你可以阅读官方文档的这一章来完成环境的安装:

https://rustwasm.github.io/docs/book/game-of-life/setup.html

  • 安装 wasm-pack

根据不同系统选用不一样的方式:https://rustwasm.github.io/wasm-pack/installer/

  • 安装cargo-generate
cargo install cargo-generate
  • 安装 node

前往官网下载: https://docs.npmjs.com/getting-started

项目搭建

  • 下载 rust 模板

在安装完依赖以后,我们可以使用模板来新建一个 wasm 项目,然后为我们的项目取一个名字,这里作者取名是 stage_9 :

cargo generate --git https://github.com/rustwasm/wasm-pack-template

在命令行使用 wasm-pack 可以编译我们模板的项目,获得一个 pkg 文件夹

wasm-pack build
  • 下载前端模板

同样我们要使用我们刚刚编译成功的 wasm ,我们需要一个前端页面来调用它,我们也可以使用模板,我们进入 rust 模板的项目中,输入如下的命令:

npm init wasm-app www

之后我们找到 www/package.json 这个文件,将我们刚刚的 rust 模板作为依赖引入进来。注意,这里的 stage_9 就是刚刚我们给 rust 模板的名字,它的来源是刚刚编译完成后生成的 pkg 文件夹

"dependencies": {
   "stage_9": "file:../pkg"
 },

之后我们安装这个依赖,我们进入 www 目录,使用 npm i 指令安装依赖。现在我们已经将 wasm 安装进我们的前端项目了,我们最后将它引入,我们修改 www/index.js 文件,将我们的函数引入进来。

import * as wasm from "stage_9";
wasm.greet();

最后我们在 www 目录启动我们的项目:

npm run start

如果现在你打开 localhost:8080 端口运行我们的项目,看到弹出一个对话框,那就说明我们的项目运行成功了。

业务逻辑

刚刚我们运行了一个简单的 wasm 项目,现在我们结合项目逻辑来介绍 wasm 项目的各个模块,我们首先回到我们的 stage_9 这个项目,然后实现我们的逻辑:

我们首先创建 models 文件夹,创建 models/course.rs 和 models/mod.rs 两个文件,之后创建 errors.rs 、 lib.rs 和 utils.rs 这几个文件用于编写我们的逻辑,之后我们还是首先更新我们的依赖:

这里要解释一下 wasm-bindgen 和它相关的包是用于将 rust 代码和 js 绑定用的 , js-sys 和 web-sys 则是 js 在 rust 中使用 js 和 web 开发相关函数的库,web-sys 包含大部分的包括 页面操作,DOM 操作 ,BOM 操作,js 请求收到等的函数,可以按需引入:

[package]
authors = ["zhangshuai <1016868503@qq.com>"]
edition = "2018"
name = "stage-9"
version = "0.1.0"

[lib]
crate-type = ["cdylib", "rlib"]

[features]
default = ["console_error_panic_hook"]

[dependencies]
chrono = {version = "0.4.19", features = ["serde"]}
js-sys = "0.3.56"
serde = {version = "1.0.136", features = ["derive"]}
serde_derive = "1.0.136"
serde_json = "1.0.79"
wasm-bindgen = {version = "0.2.79", features = ["serde-serialize"]}
wasm-bindgen-futures = "0.4.29"
web-sys = {version = "0.3.56", features = [
  "Headers",
  "Request",
  "RequestInit",
  "RequestMode",
  "Response",
  "Window",
  "Document",
  "Element",
  "HtmlElement",
  "Node",
  "console",
  "HtmlButtonElement",
  "HtmlElement",
  "MouseEvent",
  "Location"
]}
console_error_panic_hook = {version = "0.1.6", optional = true}
wee_alloc = {version = "0.4.5", optional = true}

[dev-dependencies]
wasm-bindgen-test = "0.3.13"

[profile.release]
# Tell `rustc` to optimize for small code size.
opt-level = "s"

[package.metadata.wasm-pack.profile.release]
wasm-opt = false

之后我们首先编写 models / course.rs ,其中我们先定义一个数据结构来操作我们的数据

use chrono::{DateTime, Utc};
use js_sys::Promise;
use serde::{Deserialize, Serialize};
use serde_json::from_str;
// use crate::models::course::Course;
use crate::errors::MyError;
use wasm_bindgen::{JsCast, JsValue, prelude::wasm_bindgen};
use wasm_bindgen_futures::JsFuture;
use web_sys::{Request, RequestInit, RequestMode, Response};

#[derive(Serialize, Deserialize, Debug)]
pub struct Course {
    pub id: i32,
    pub teacher_id: i32,
    pub name: String,
    pub time: Option<DateTime<Utc>>,

    pub description: Option<String>,
    pub format: Option<String>,
    pub structure: Option<String>,
    pub duration: Option<String>,
    pub price: Option<i32>,
    pub language: Option<String>,
    pub level: Option<String>,
}

之后我们编写一些增删改查的操作函数:

这里以查找为例子简单来说明一下:

  • RequestInit::new() 可以初始化一个请求,我们可以用这个请求来调用接口,
  • 因为是查找,我们设置它为 GET 方法,之后配置它为跨域,因为我们的后台在 localhost:3077 而前端在 localhost:8080,所以请求需要跨域,之后使用 Request 相关的 api 来访问指定的接口
  • 之后我们为我们的请求添加请求头,之后使用 fetch_with_request 来调用 fetch 的 api 发送我们的请求,因为是 fetch 相关的 api 是在 js 的 window 对象中的,所以我们要先初始化一个 window 对象,他在 web_sys 这个库中
  • 我们使用 JsFuture 接收信息,之后把它转为 Response 的形式, Response 也是 wasm 提供的一个接收请求的结构
  • 最后我们使用 json 相关的 json 将我们的 json 数据转为 Vec 的数据结构,因为 Course 实现了 Serialize ,所以我们可以转化我们的 json 数据

删除的函数编写方式大同小异:

pub async fn get_course_by_teacher(teacher_id: i32) -> Result<Vec<Course>, MyError> {
    let mut opts = RequestInit::new();
    opts.method("GET");
    opts.mode(RequestMode::Cors);

    let url = format!("http://localhost:3077/courses/{}", teacher_id);

    let request = Request::new_with_str_and_init(&url, &opts)?;

    request.headers().set("Accept", "application/json")?;

    let fetch  = web_sys::window().ok_or("no window exisits".to_string())?;

    let resp_value = JsFuture::from(window.fetch_with_request(&request)).await?;

    assert!(resp_value.is_instance_of::<Response>());

    let resp: Response = resp_value.dyn_into().unwrap();

    let json = JsFuture::from(resp.json()?).await?;

    let courses: Vec<Course> = json.into_serde().unwrap();

    Ok(courses)
}

pub async fn delete_course(teacher_id: i32, course_id: i32) -> () {
    let mut opts = RequestInit::new();
    opts.method("DELETE");
    opts.mode(RequestMode::Cors);

    let url = format!("http://localhost:3077/courses/{}/{}", teacher_id, course_id);

    let request = Request::new_with_str_and_init(&url, &opts).unwrap();

    request.headers().set("Accept", "application/json").unwrap();

    let window = web_sys::window()
        .ok_or("no window exisits".to_string())
        .unwrap();

    let resp_value = JsFuture::from(window.fetch_with_request(&request))
        .await
        .unwrap();

    assert!(resp_value.is_instance_of::<Response>());

    let resp: Response = resp_value.dyn_into().unwrap();

    let json = JsFuture::from(resp.json().unwrap()).await.unwrap();

    let courses: Course = json.into_serde().unwrap();
}


#[wasm_bindgen]
pub async fn add_course(name: String, desc: String) -> Result<Promise, JsValue> {
    let mut opts = RequestInit::new();
    opts.method("POST");
    opts.mode(RequestMode::Cors);

    let str_json = format!(
        r#"{{
            "teacher_id":1,
            "name":"{}",
            "description":"{}"
        }}"#,
        name, desc
    );
    opts.body(Some(&JsValue::from_str(str_json.as_str())));

    let url = "http://localhost:3077/courses/";

    let request = Request::new_with_str_and_init(&url, &opts)?;

    request.headers().set("Accept", "application/json")?;
    request.headers().set("Content-type", "application/json")?;

    let window = web_sys::window()
        .ok_or("no window exisits".to_string())
        .unwrap();

    let resp_value = JsFuture::from(window.fetch_with_request(&request))
        .await
        .unwrap();

    assert!(resp_value.is_instance_of::<Response>());

    let resp: Response = resp_value.dyn_into().unwrap();

    Ok(resp.json()?)
}

而新增的函数因为我们后续将要在 js 中直接调用,所以我们为它加上 #[wasm_bindgen] ,加上这个标识的函数后续将会绑定到 wasm 对象上,我们可以直接在 js 代码中调用他们,新增数据的函数不同的是我们将构造一个 json 字符串,然后将它序列化后放入 body 随着 POST 方法提交,要注意,我们的函数是异步的,它需要在提交成功后能被 js 继续响应处理,所以我们返回一个 Promise 类型,使用 resp.json() 就可以返回一个 Promise 对象:

#[wasm_bindgen]
pub async fn add_course(name: String, desc: String) -> Result<Promise, JsValue> {
    let mut opts = RequestInit::new();
    opts.method("POST");
    opts.mode(RequestMode::Cors);

    let str_json = format!(
        r#"{{
            "teacher_id":1,
            "name":"{}",
            "description":"{}"
        }}"#,
        name, desc
    );
    opts.body(Some(&JsValue::from_str(str_json.as_str())));

    let url = "http://localhost:3077/courses/";

    let request = Request::new_with_str_and_init(&url, &opts)?;

    request.headers().set("Accept", "application/json")?;
    request.headers().set("Content-type", "application/json")?;

    let window = web_sys::window()
        .ok_or("no window exisits".to_string())
        .unwrap();

    let resp_value = JsFuture::from(window.fetch_with_request(&request))
        .await
        .unwrap();

    assert!(resp_value.is_instance_of::<Response>());

    let resp: Response = resp_value.dyn_into().unwrap();

    Ok(resp.json()?)
}

我们在 mod.rs 中导出我们的 course ,之后我们编写 lib.rs 作为我们的页面生成函数:

这部分是我们之后的模板中包含的部分,其中 extern “C” 模块中声明了我们需要的 js 函数,比如 alert 就是 js 的 alert 函数,可以弹出一个对话框包含一些数据,我们添加一些我们需要的函数,比如 confirm 用于弹出包含取消和确定的模态框, log 函数是 console.log 函数,用于在控制台打印数据,因为 js 中函数是 console.log ,所以我们需要用 #[wasm_bindgen(js_namespace = console)] 绑定命名空间:

mod utils;

use wasm_bindgen::prelude::*;

use wasm_bindgen_futures::spawn_local;
use web_sys::*;

// When the `wee_alloc` feature is enabled, use `wee_alloc` as the global
// allocator.
#[cfg(feature = "wee_alloc")]
#[global_allocator]
static ALLOC: wee_alloc::WeeAlloc = wee_alloc::WeeAlloc::INIT;

#[wasm_bindgen]
extern "C" {
    fn alert(s: &str);
    fn confirm(s:&str) -> bool;

    #[wasm_bindgen(js_namespace = console)]
    fn log(s: &str);
}

#[wasm_bindgen]
pub fn greet(s: &str) {
    alert(format!("Hello, {}!", s).as_str());
}

之后我们添加我们的页面生成的逻辑,它让我们在查找出课程数据后生成我们的表格结构:

这里我们使用了 document 这个对象,它和 js 中的 document 用法一致,我们可以使用它查找 DOM 元素,生成添加 DOM 元素,使用它就可以生成一个包含我们查询到数据的表格结构。

之后需要为我们的表格中添加一个 删除按钮,我们创建一个闭包 click_closure 来处理这个事件,并且使用 add_event_listener_with_callback 将它绑定到按钮上,要注意,虽然我们要将我们的数据编译成 wasm 运行,但是它的逻辑还是遵循基本的 rust 规则,在函数结束之后,我们的 click_closure 就到作用域底部了,将被销毁,此时我们的事件也会失效,所以我们使用 click_closure.forget() 来解决这个问题。

最后我们在函数头部加上 #[wasm_bindgen(start)] 这个标识,它代表当我们运行我们的项目,这个函数就会被调用,也就是我们每次进入界面,这个函数就会被自动调用了

pub mod errors;
pub mod models;

use models::course::{delete_course, Course};

click_closure.forget();
pub async fn main() -> Result<(), JsValue> {
    let window = web_sys::window().expect("no");
    let document = window.document().expect("no");

    let left_body = document.get_element_by_id("left_body").expect("no");

    let courses: Vec<Course> = models::course::get_course_by_teacher(1).await.unwrap();

    for c in courses.iter() {
        let tr = document.create_element("tr")?;
        tr.set_attribute("id", format!("tr-{}", c.id).as_str())?;
        let td = document.create_element("td")?;
        td.set_text_content(Some(format!("tr-{}", c.id).as_str()));
        tr.append_child(&td)?;

        let td = document.create_element("td")?;
        td.set_text_content(Some(c.name.as_str()));
        tr.append_child(&td)?;

        let td = document.create_element("td")?;
        if let Some(time) = c.time.clone() {
            td.set_text_content(Some(time.to_string().as_str()));
        }
        tr.append_child(&td)?;

        let td = document.create_element("td")?;
        if let Some(desc) = c.description.clone() {
            td.set_text_content(Some(desc.as_str()));
        }
        tr.append_child(&td)?;

        let td = document.create_element("td")?;
        let btn: HtmlButtonElement = document
            .create_element("button")
            .unwrap()
            .dyn_into::<HtmlButtonElement>()
            .unwrap();

        let cid = c.id;
        let click_closure = Closure::wrap(Box::new(move |_event: web_sys::MouseEvent| {
            let r = confirm(format!("确认删除 ID 为 {} 的课程?", cid).as_str());
            match r {
                true => {
                    spawn_local(delete_course(1, cid));
                    alert("删除成功");
                    web_sys::window().unwrap().location().reload().unwrap();
                }
                _ => {}
            }
        }) as Box<dyn Fn(_)>);

        btn.add_event_listener_with_callback("click", click_closure.as_ref().unchecked_ref())?;
        click_closure.forget();
        btn.set_attribute("class", "btn btn-danger btn-sm")?;
        btn.set_text_content(Some("Delete"));
        td.append_child(&btn)?;
        tr.append_child(&td)?;

        left_body.append_child(&tr)?;
    }

    Ok(())
}

我们为它加上简单的 errors.rs 处理错误,这里我们主要才处理的是 JsValue 这个类型,当我们的 js 相关的 api 代码执行中发生错误,它就会返回 JsValue ,其中包含了 js 相关的报错,我们可以如下的方式取得它然后处理我们的异常:

use serde::Serialize;

#[derive(Debug, Serialize)]
pub enum MyError {
    SomeError(String),
}

impl From<String> for MyError {
    fn from(s: String) -> Self {
        MyError::SomeError(s)
    }
}
impl From<wasm_bindgen::JsValue> for MyError {
    fn from(js_value: wasm_bindgen::JsValue) -> Self {
        MyError::SomeError(js_value.as_string().unwrap())
    }
}

现在我们完成了我们代码的编写,我们可以使用 wasm-pack build 来重新 build 我们的代码。

页面逻辑编写

我们完成了事务逻辑代码的编写后,我们就可以在前端页面调用这些逻辑代码了,我们进入 www 这个模板,编写我们的 index.html 这个页面:

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8" />
    <title>Hello wasm-pack!</title>
    <script src="./bootstrap.js"></script>
    <link
      href="https://cdn.staticfile.org/twitter-bootstrap/5.1.3/css/bootstrap.min.css"
      rel="stylesheet"
    />
  </head>
  <body>
    <noscript
      >This page contains webassembly and javascript content, please enable
      javascript in your browser.</noscript
    >
    <nav class="navbar navbar-dark bg-primary">
      <div class="container-fluid">
        <a class="navbar-brand" href="#"> WSAM 项目</a>
      </div>
    </nav>
    <div class="m-3" style="height: 600px">
      <div class="col">
        <div class="card border-info mb-3">
          <div class="card-header">Course</div>
          <div class="card-body">
            <form class="row g-3 needs-validation" id="form">
              <label for="name" class="form-label">课程名称</label>
              <div class="mb-3">
                <input
                  type="name"
                  class="form-control"
                  id="name"
                  required
                  placeholder="课程名称"
                />
                <div class="invalid-feedback">请填写课程名称</div>
              </div>
              <div class="mb-3">
                <label for="description" class="form-label">课程简介</label>
                <textarea
                  id="description"
                  rows="3"
                  class="form-control"
                ></textarea>
              </div>
              <div class="col-12">
                <button type="submit" class="btn btn-primary">提交</button>
              </div>
            </form>
  
          </div>
        </div>
        <table class="table table-hover table-bordered table-sm">
          <thead>
            <tr>
              <th scope="col">ID</th>
              <th scope="col">Name</th>
              <th scope="col">Time</th>
              <th scope="col">Description</th>
              <th scope="col">操作</th>
            </tr>
          </thead>
          <tbody id="left_body"></tbody>
        </table>
        <div id="left"></div>
      </div>
    </div>
  </body>
</html>

之后我们编写我们的 index.js 这个函数,在这个函数里,我们引入我们的 stage_9 这个项目,导出为 wasm 对象,现在在 wasm 上挂载了我们之前编写的由 #[wasm_bindgen] 标识的函数,比如我们的提交数据函数:

import * as wasm from "stage_9";

//wasm.set_panic_hook();


const myForm = document.getElementById('form');

myForm.addEventListener('submit',(e) => {
    e.preventDefault();

    const name = document.getElementById('name').value;
    const desc = document.querySelector('#description').value;

    wasm.add_course(name,desc).then((json) => {
        console.log(json)
        alert('成功!');
        window.location.reload();
    })

})

配置跨域

现在重新 npm install 然后 npm run start 就可以启动我们的 wasm 项目了,但是位了测试我们的接口,我们还需要在 3077 端口启动我们的后台项目,此时会发现报错了,原因是我们的后台项目没有配置跨域,所以跨域失败了,现在我们回到我们的后台项目,配置我们的跨域,我们还是先引入依赖:

[dependencies]
actix-cors = "0.6.0-beta.10"

之后我们在 teacher-service.rs 编写我们的逻辑:我们定义一个跨域的配置,允许 http://localhost:8080/ 来源的请求跨域进入我们的接口服务中,指定它的方法和请求头,两次配置分别是 “Accept”, “application/json” 和 “Content-type”, “application/json” 这两个请求头,对于情况和 POST 请求,最后我们在我们的项目 注入我们的跨域配置器

let app = move || {

        let cors = Cors::default()
        .allowed_origin("http://localhost:8080/")
        .allowed_origin_fn(|origin, _req_head| {
            origin.as_bytes().starts_with(b"http://localhost")
        }).allowed_methods(vec!["GET","POST","DELETE"])
        .allowed_headers(vec![http::header::AUTHORIZATION,http::header::ACCEPT])
        .allowed_header(http::header::CONTENT_TYPE)
        .max_age(3600);
    
        App::new()
            .app_data(shared_data.clone())
            .app_data(web::JsonConfig::default().error_handler(|_err, _req| {
                MyError::InvalidInput(" please  provide valid json input".to_string()).into()
            }))
            .configure(general_routes)
            .configure(course_routes)
            .wrap(cors)
            .configure(teacher_routes)
};

效果预览

现在我们已经可以实现跨域了,我们将我们的项目启动起来,它的效果是这样的:

请添加图片描述

我们输入一门新的课程可以把它添加到项目中:

请添加图片描述

点击删除按钮可以删除指定的课程,但是你需要先进行确认:

请添加图片描述

如果你可以如上允许这个项目,说明你的项目成功了,如果你的项目运行有问题,可以查看作者的 git :https://github.com/aiai0603/rust_web_mysql

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/396780.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

5款电商团队必须会用的任务管理工具

随着企业数字化转型&#xff0c;高效率的团队协作和远程协同办公成为越来越多团队的强烈需求&#xff0c;今天说说电商团队&#xff0c;电商运营很大一个特点是多场景、跨部门、高频协作。并且要通过多平台和形式跟新兴媒体进行品宣争夺市场份额。 这种情形对跨部门的团队协作…

java基础学习 day50(内部类)

什么是内部类&#xff1f; 写在一个类里面的类就叫做内部类 什么时候用到内部类&#xff1f; B类表示的事物是A类的一部分&#xff0c;且B单独存在没有意义。 比如&#xff1a;汽车和发动机&#xff0c;ArrayList和迭代器&#xff0c;人和心脏 内部类的访问特点 内部类可以…

STM32U5开发(1)----通过 USART1 发送数据

概述 通过 USART1 发送一些数据。 最近在弄ST和GD的课程&#xff0c;需要样片的可以加群申请&#xff1a;6_15061293。 生成例程 使用STM32CUBEMX生成例程&#xff0c;这里使用NUCLEO-U575ZI开发板。 选择工程的时候&#xff0c;先不必选择加载了TrustZone。 样品申请 h…

STM32程序下载和启动方式

目录1 BOOT引脚配置和下载说明2 关于串口下载方式3 关于一按复位就跑代码4 关于下载调试速度5 关于三种启动方式5.1 FLASH启动5.2 系统存储器器启动5.3 SRAM启动6 关于程序的三种下载方式1 BOOT引脚配置和下载说明 BOOT0BOOT1程序运行ST-Link下载串口下载启动说明xx无0x√√用…

AList搭建网盘挂载硬盘并挂载网络资源(傻瓜式自配置教程)

AList搭建网盘挂载硬盘并挂载网络资源1.安装AList1.1 下载1.2 解压1.3 启动1.4 登录1.5 改密1.6 开机自启2.添加云盘存储2.1 添加阿里云盘2.2 阿里云刷新令牌2.3 查看云盘文件3.映射本地盘3.1 下载RaiDrive3.2 安装3.2 设置4.延伸参考资料&#xff1a; AList: https://alist.n…

Warshall算法

&#x1f680;write in front&#x1f680; &#x1f4dc;所属专栏&#xff1a;> 算法 &#x1f6f0;️博客主页&#xff1a;睿睿的博客主页 &#x1f6f0;️代码仓库&#xff1a;&#x1f389;VS2022_C语言仓库 &#x1f3a1;您的点赞、关注、收藏、评论&#xff0c;是对我…

数据中台架构体系理解

目前&#xff0c;大部分企业更倾向于数据集中采集、存储&#xff0c;并应用分层建设。这种方式一方面有利于应用系统的快速部署&#xff0c;另一方面也保证了数据的集中管理与运营&#xff0c;体现数据的资产、资源属性。 数据中台的出现弥补了数据开发和应用开发之间由于开发…

工厂设计模式

介绍 Java工厂设计模式主要分为三种: 简单工厂模式(Simple Factory Pattern):使用一个工厂类来封装对象创建的过程,客户端只需要通过传递不同的参数来获取不同的产品对象,从而避免了客户端直接创建产品对象的操作工厂方法模式(Factory Method Pattern):将工厂类抽象出来,每个…

fastjson 返回 $ref 数据

文章目录问题描述&#xff1a;1、重复引用&#xff1a;2、循环引用&#xff1a;原因分析&#xff1a;1、重复引用&#xff1a;2、循环引用&#xff1a;反序列化&#xff1a;1、开启引用检测&#xff1a;2、关闭引用检测&#xff1a;小结&#xff1a;问题描述&#xff1a; 问题…

小樽C++ 单章⑨ 文件

目录 1.文件类型变量的定义与引用 1.1 文件的读写 1.2 fopen()版 (C专用) 1.3 文件输入输出流 (C专用) 文件有两种保存方式&#xff1a;二进制文件、文本文件。例如存121这个数字。 二进制存储效率高&#xff0c;但是对我们不友好&#xff0c;对每个值都要变成二进制太难啦…

uniapp实现自定义相机

自定义相机起因由于最近用uniapp调用原生相机容易出现闪退问题&#xff0c;找了很多教程又是压缩图片又是优化代码&#xff0c;我表示并没有太大作用!!实现自定义相机使用效果图拓展实现多种自定义相机水印相机身份证相机人像相机起因 由于最近用uniapp调用原生相机容易出现闪退…

MindAR的网页端WebAR图片识别功能的图片目标编译器中文离线版本功能(含源码)

前言 之前制作了基于MindAR实现的网页端WebAR图片识别叠加动作模型追踪功能的demo&#xff0c;使用了在线的图像目标编译器对识别图进行了编译&#xff0c;并实现了自制的WebAR效果&#xff0c;大致效果如下&#xff1a; 但是在线的编译器在操作中也不是很方便&#xff0c;我…

207. 课程表

207. 课程表https://leetcode.cn/problems/course-schedule/ 难度中等1526 你这个学期必须选修 numCourses 门课程&#xff0c;记为 0 到 numCourses - 1 。 在选修某些课程之前需要一些先修课程。 先修课程按数组 prerequisites 给出&#xff0c;其中 prerequisites[i] [a…

One-YOLOv5 v1.2.0 Released(支持分类,检测,实例分割)

0x0. 引言0x1. 快速开始0x2. 在COCO上的精度表现 yolov5s-defaultyolov5s-seg 0x3. 在COCO上的单GPU性能表现特性 & bug 修复 特性用户反馈的bug 下个版本的展望附件常用预训练模型下载列表 0x0. 引言 &#x1f31f; v1.2.0同步了ultralytics yolov5的上游分支v7.0 &…

Python+ChatGPT制作一个AI实用百宝箱

目录一、注册OpenAI二、搭建网站及其框架三、AI聊天机器人四、AI绘画机器人ChatGPT 最近在互联网掀起了一阵热潮&#xff0c;其高度智能化的功能能够给我们现实生活带来诸多的便利&#xff0c;可以帮助你写文章、写报告、写周报、做表格、做策划甚至还会写代码。只要与文字相关…

The Sandbox 中的独特体验——《奥米加》

在过去几年间&#xff0c;The Sandbox 游戏变得越来越受欢迎。因为我们为玩家提供了在虚拟世界中探索、创造和游戏的自由&#xff0c;没有线性游戏的限制。DeQuest 工作室创作的《奥米加》也正是如此&#xff0c;绝对是一个前所未有的体验&#xff01; 先了解一下《奥米加》的故…

代码审计之旅之百家CMS

前言 之前审计的CMS大多是利用工具&#xff0c;即Seay昆仑镜联动扫描出漏洞点&#xff0c;而后进行审计。感觉自己的能力仍与零无异&#xff0c;因此本次审计CMS绝大多数使用手动探测&#xff0c;即通过搜索危险函数的方式进行漏洞寻找&#xff0c;以此来提升审计能力&#xf…

网易云信 Crash 异常治理实践 | 智企技术委员会技术专题系列

前言Crash&#xff08;造成用户无法使用客户端所承载的服务&#xff09;作为客户端稳定治理的最主要问题之一。云信作为国内业界领先的 RTC/IM PaaS 服务商&#xff0c;对于客户端 SDK&#xff08;PaaS 服务商对外服务的主要载体&#xff09;的 Crash 治理再重视也不为过。关于…

【编程基础之Python】12、Python中的语句

【编程基础之Python】12、Python中的语句Python中的语句赋值语句条件语句循环语句for循环while循环continue语句break语句continue与break的区别函数语句pass语句异常处理语句结论Python中的语句 Python是一种高级编程语言&#xff0c;具有简单易学的语法&#xff0c;适用于各…

JAVA架构与开发(JAVA架构是需要考虑的几个问题)

在企业中JAVA架构师主要负责企业项目技术架构&#xff0c;企业技术战略制定&#xff0c;技术框架搭建&#xff0c;技术培训和技术攻坚的工作。 在JAVA领域&#xff0c;比较多的都是web项目。用于解决企业的数字化转型。对于JAVA架构师而言&#xff0c;平时对项目的架构主要考虑…