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

Rust 实战练习 - 7. FFI, 库, ABI, libc

FFI

FFI(Foreign Function Interface)是这样一种机制:用一种编程语言写的程序能调用另一种编程语言写的函数(routines)。

调用约定,类型表示和名称修饰这三者的统称,即是众所周知的应用二进制接口(ABI)。

如果所有的语言在调用时都能认识同样一套 ABI 规范,那么就能完全畅通的调用了。可惜,世界不会像我们人为想象的那样干净。

在计算机技术发展的过程中,出现了各种 ABI 规范:

  • cdecl
  • syscall
  • optlink
  • pascal
  • register
  • stdcall
  • fastcall
  • thiscall
  • winapi
  • Intel ABI
  • System V

等。详情可参考:X86调用约定。

而 Rust 目前支持如下 ABI 约定:

  • stdcall
  • aapcs
  • cdecl
  • fastcall
  • vectorcall
  • Rust
  • rust-intrinsic
  • system
  • C
  • win64
  • sysv64

目前我们 IT 工业的基石,绝大部分是由 C 语言写成。于是自然而然,绝大多数库都遵循 cdecl(或 C)规范。所以我们可以专注于 C 规范来讨论问题。

目前所知,能(较方便地)对其它语言提供 FFI 库支持的语言有:

  • C
  • C++(通过定义 C 接口)
  • Rust(通过使用 C 约定)
  • Ada
  • Fortran

通用方案

在开发的过程中,要一个一个对大量的 C/C++ 库写绑定来进行 FFI,毕竟是一项费时费力的活儿。聪明的程序员们就开始构想一些“通用”的方案,实现批量快速绑定。

SWIG

以下定义来自 https://zh.wikipedia.org/wiki/SWIG:

简单包装界面产生器(SWIG)是一个开源软件工具,用来将C语言或C++写的计算机程序或函式库,连接脚本语言,例如Lua, Perl, PHP, Python, R, Ruby, Tcl, 和其它语言,例如C#, Java, JavaScript, Go, D, OCaml, Octave, Scilab以及Scheme. 也可以输出成XML格式。

也就是说,使用了 SWIG 这套工具和规范,就可以直接在上层语言(动态语言居多)中调用 C/C++ 库了,省却大量烦恼。但在实际使用中,还会有一些细节问题,往往需要人工调整。所以也不是那么完美。

Gnome 的 GI 规范

Gnome/Gtk 那一帮理想主义青年,发明了 GI(GObject Introspection)。用于对基于 glib/gobject 生态的众多软件(C 代码库)自动生成完整的接口描述文件(及 typelib),然后其它语言只要实现了对 Gir 这一个标准的支持,那么就可以无缝调用所有经过 Gir 化处理的 C 库。而不再需要单独为每一个 C 库做绑定了。这样就大大简化了 FFI 接口项目的编写工作。

目前这一杰出创意的重量级工作成果有 cairo, pango, gtk 等库。

更多信息请参考:https://gi.readthedocs.io/en/latest/。

另一种思路——基于字节码的平台级路线

语言间的相互调用,历史的发展提供了另一条路线:建立一个共同的字节码平台,这个平台之上的所有语言,皆可便捷地相互调用。

  • JVM 平台语言之间的 FFI
  • WASM 平台的 FFI

https://rustcc.cn/search?q=Rust%20FFI%20%E7%BC%96%E7%A8%8B

https://rustcc.cn/article?id=44a47b42-184c-4e2f-9318-be07ee5a516c

Rust 语言主要在关键字和标准库两个方面对 FFI 提供了支持,具体如下:

  • 关键字 extern
    • 属性 #[no_mangle]
    • 外部块 ExternBlock 及其属性 linklink_name
  • 标准库
    • std:os:raw 模块
    • std:ffi 模块

Rust编写lib给别人用

通过使用 ABI 字符串来指定具体的 ABI,其中有三个 ABI 字符串是跨平台的:

  • extern "Rust",默认的 ABI,在 Rust 代码中对任何普通函数fn foo()声明时都将使用该 ABI。
  • extern "C",指定使用 C-ABI,类似extern fn foo(),无论 C 编译器支持哪种默认设置。
  • extern "system",通常类似extern "C",但在 Win32 平台上,它是"stdcall",或用于链接到 Windows API。

crate-type: [bin|lib|rlib|dylib|cdylib|staticlib|proc-macro]

  • bin 二进制可执行文件,main函数为入口。
  • lib Rust Library的编译工具默认值,一般为rlib。由编译器推荐其值。
  • rlib Rust Library的静态库,满足Rust API。 *.rlib 实现为一个 ar 归档文件。
  • dylib Rust Library的动态库,在编译的时候,生成动态库(Linux 上为 .so, MacOS 上为 .dylib, Windows 上为 .dll)。
  • cdylib 满足C调用规范的动态链接库。在编译的时候,生成动态库(Linux 上为 .so, MacOS 上为 .dylib, Windows 上为 .dll)。
  • staticlib 满足C调用规范的静态链接库。编译会生成 .a 文件(在 Linux 和 MacOS 上),或 .lib 文件(在 Windows 上)。
  • proc-macro 过程宏crate.
// Rust定义函数给外部使用
#[no_mangle]
pub extern "C" fn addtwo1(a: u32, b: u32) -> u32 {let c = a + b;println!("print in rust, sum is: {}", c);c
}#[no_mangle]
pub extern "C" fn sum_of_array(array: *const u32, len: usize) -> u32 {let array = unsafe {assert!(!array.is_null());slice::from_raw_parts(array, len)};array.iter().sum()
}// 在 Cargo.toml 中,加入如下两行
[lib]
crate-type = ["cdylib"]

C代码

#include <stdio.h>
#include <stdint.h>extern uint32_t addtwo1(uint32_t, uint32_t);
extern uint32_t sum_of_array(const uint32_t *numbers, size_t length);int main(void) {uint32_t sum = addtwo1(10, 20);printf("print in c, sum is: %d\n", sum);uint32_t numbers[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};unsigned int length = sizeof(numbers) / sizeof(*numbers);uint32_t sum = sum_of_array(numbers, length);printf("print in c, sum is: %d\n", sum);
}

build

编译 rust so:cargo build
编译 c binary:gcc -o ./ccode03 ./ccode03.c -L ./ -lrustffi3
运行:LD_LIBRARY_PATH=. ./ccode03

结构体导出

只有当#[repr(C)] / #[repr(C, packed)]属性应用于一个struct时,Rust 才能保证该结构的布局与平台的 C 语言表示兼容

#[repr(C)]
pub struct Foo {a: isize,b: isize
}// Struct的方法
impl Foo {// 成员函数,需要this指针pub fn method1(&self) -> isize{self.a+self.b}// 成员函数pub fn method2(&self, x: isize) -> isize {x + self.a + self.b}// 静态成员函数,不需要thispub fn method3(x: isize) -> isize {x*2}
}

对应的C代码大概是这样子

struct Foo {int a;int b;
}void foo_method1(Foo* foo);
int foo_method2(Foo* foo, int x);
int foo_method1(int x);

然而,这种映射是不能自动转换的(毕竟只是我们自己的约定),需要手动写出来。

#[no_mangle]
pub unsafe extern "C" fn foo_method1(foo: *mut Foo) {let foo = &*foo;foo.method1();
}#[no_mangle]
pub unsafe extern "C" fn foo_method2(foo: *const Foo, x: isize) -> isize {let foo = &*foo;foo.method2(x)
}#[no_mangle]
pub unsafe extern "C" fn foo_method3(x: isize) -> isize {Foo::method3(x)
}

编译cargo build --release, 得到

$ nm -D --defined-only libd7.so 
0000000000001100 T foo_method1
0000000000001110 T foo_method2
0000000000001120 T foo_method3

最后,C代码:

#include <stdio.h>struct Foo
{long a;long b;
};extern void foo_method1(struct Foo *f);
extern long foo_method2(struct Foo *f, long x);
extern long foo_method3(long x);int main(int argc, char* argv){struct Foo f;f.a = 10;f.b = 100;long val = -1;foo_method1(&f);val = foo_method2(&f, 2);printf("m2: %ld\r\n", val);val = foo_method3(3);printf("m3: %ld\r\n", val);return 0;
}// gcc -o ../target/release/c_ffi main.c -L ../target/release/ -ld7
// LD_LIBRARY_PATH=. ./c_ffi
m2: 112
m3: 6

完美!

Rust调用别人写好的C库

// Rust使用外部的库
#[cfg(all(target_os = "win32", target_arch = "x86"))]
#[link(name = "c_library",kind="cdylib" )]
#[allow(non_snake_case)]
extern "C" {// 访问全局变量static rl_readline_version: libc::c_int; // 可以使用link_name映射外部函数名#[link_name = "c_function_name"]fn c_function(input: i32) -> i32;
}// 标准库<stdlib.h>内置的abs函数
extern "C" {#[link_name = "abs"]fn abs_in_rust(input: i32) -> i32;
}fn main() {unsafe {println!("abs(-1) is {}", abs_in_rust(-1));}
}

kind是一个可选值,通过它来指定原生库的类型,它有以下三种可选的值:

  • dylib,表示为动态库。如果未指定kind,则它为默认值。
  • static,表示为静态库。
  • framework,表示 macOS 的框架,这仅对 macOS 目标有效。

对于互传递参数,Rust 标准库std::os::rawstd::ffi 这两个模块提供了这方面的支持。

参考链接

[1] 外部块支持的 ABI 字符串,https://doc.rust-lang.org/reference/items/external-blocks.html

[2] 标准库 std::os::raw 模块,https://doc.rust-lang.org/stable/std/os/raw/index.html

[3] 标准库 std::ffi 模块,https://doc.rust-lang.org/std/ffi/index.html

[4] Rust 中 String 与 UTF-8 编码,https://mp.weixin.qq.com/s/ZX_0G6JcNMusLz6JJOkNSg

libc

libc 是对各平台的系统库的原始 FFI 绑定。其代码地址在:https://github.com/rust-lang/libc。可以看到,这是 Rust 官方维护的一个库。

libc 提供了与 Rust 支持的各平台上的最基础系统 C 库打交道的所有必要设施。它导出了底层平台的类型、函数和常量。

libc 会导出底层 C 库的这些东西:

  • C 类型,比如 typedefs, 原生类型,枚举,结构体等等
  • C 常量,比如使用 #define 指令定义的那些常量
  • C 静态变量
  • C 函数(按它们的头文件中定义的函数签名来导出)
  • C 宏,在 Rust 中会实现为 #[inline] 函数

另外,libc 中导出的所有 C struct 都已经实现了 CopyClone trait.

libc 的所有函数调用,都必须放进 unsafe 块中。因为它的所有调用都是 unsafe 的;

Rust 默认与libclibm链接。

与C的库区别:

熟悉 linux 系统开发的同学都知道,linux 系统本身有个 libc 库,是几乎一切应用的基础库。基本上 linux 下 C 语言写的代码都要链接这个库才能运行。

而 Rust 的 libc crate,不完全等价于 C 的 libc 库的封装。具体区别如下:

  • Linux (以及其它 unix-like 平台)下,导出的是 libc, libm, librt, libdl, libutil 和 libpthread 这几个库的符号。
  • OSX 下,导出的是 libsystem_c, libsystem_m, libsystem_pthread, libsystem_malloc 和 libdyld 这几个库的符号。
  • Windows 下,导出的是 VS CRT(VS C RunTime VS C 运行时库)中的符号。但是这些符号,比前两个平台的符号,数量上要少得多。因此,可以直接这样说,Rust libc crate 在 Windows 平台上的功能有限。在 Windows 平台上,建议使用 winapi 这个 crate 进行开发。

比如:Rust 标准库中没有提供创建子进程的设施,不过可以创建一个子线程。而libc 可以创建一个子进程。

libc 与 std::os:😗::raw 的关系?

细心的同学会发现,在标准库的 os 模块下面,有一些东西与 libc 的重复。

页面 https://doc.rust-lang.org/std/os/raw/index.html 包含了 c_char, c_double, c_float, c_int, c_long, c_longlong, c_schar, c_short, c_uchar, c_uint, c_ulong, c_ulonglong, c_ushort

而 libc 中,对这些内容,也重新定义了一份(比如:https://docs.rs/libc/0.2.69/libc/type.c_char.html)。为什么呢?

std::os::raw 中这些定义,可以用于与一些简单的 C 代码进行交互,比如说不存在系统调用的 C 代码。这个时候,就不需要再引入 libc 库了。

而一旦产生了系统调用或者 Unix 环境编程,那么就得引入 libc 库来操作。

std 下面还有一些 std::os:😗::raw 的模块,这些模块现在已经被 Deprecated 了(比如:https://doc.rust-lang.org/std/os/unix/raw/index.html)。

nix库

nix 库 旨在提供对各种类 Unix 平台(Linux,Darwin等)API 的友好绑定(bindings),其代码地址在:https://github.com/nix-rust/nix。

它通过使用强制合法或安全的类型对 libc 库进行了一次封装,相对于 libc 库暴露的 unsafe API,它具有两个特点:

  • 用户代码中尽量没有 unsafe
  • Rust 风格的错误处理

不过尽管 nix 库尝试支持 libc 库支持的所有平台,但由于技术或人力限制,仅支持其中的某些平台。可能这也是一些底层库(比如:tokio项目中的mio)在版本v0.6.3之后 移除 对 nix 库依赖的一个原因吧。

nix 库中的模块大致如下:

  • dir,相对标准库中的std::fs::ReadDir更底层的目录接口。
  • errno, nix 库中处理各种类 Unix 系统的错误类型,对于 FreeBSD,IOS,MacOS 系统直接封装的 libc 库中的。
  • fcntl, Unix 系统中文件 IO 的数据结构,以及对文件的各种操作接口。
  • features,用于操作系统级功能的测试。
  • ifaddrs,使用 Linux 或 BSD 中的函数getifaddrs获取网络接口及地址列表。
  • kmod,包含加载和卸载内核模块的功能。
  • mount,包含设备文件的挂载操作,mountumount
  • mqueue, 对应 POSIX 规范中消息队列 API 的功能。
  • net,涉及网络接口的功能。
  • poll,在特点文件描述符上触发 wait 事件。
  • pty,创建主从虚拟伪终端 PTYs。
  • sched,提供 Linux 系统的调度接口。
  • sys,这个模块包括各种系统相关的功能:POSIX 异步 I/O,文件系统事件的监控 API,Socket接口函数等。
  • ucontext,提供协程上下文相关的借接口。
  • unistd,在 libc 库unistd.h头文件中函数的 safe 封装。

Demo

use nix::unistd::*;fn main() {match fork() {Ok(ForkResult::Parent { child }) => {// 在父进程中println!("Hello, I am parent thread: {}", getpid());}Ok(ForkResult::Child) => {// 在子进程中println!("Hello, I am child thread: {}", getpid());println!("My parent thread: {}", getppid());            }Err(errno) => {// fork 创建子进程失败println!("Fork creation failed!");}}  
}

相关文章:

Rust 实战练习 - 7. FFI, 库, ABI, libc

FFI FFI&#xff08;Foreign Function Interface&#xff09;是这样一种机制&#xff1a;用一种编程语言写的程序能调用另一种编程语言写的函数&#xff08;routines&#xff09;。 调用约定&#xff0c;类型表示和名称修饰这三者的统称&#xff0c;即是众所周知的应用二进制…...

vue实现把Ox格式颜色值转换成rgb渐变颜色值(开箱即用)

图示&#xff1a; 核心代码&#xff1a; //将0x格式的颜色转换为Hex格式&#xff0c;并计算插值返回rgb颜色 Vue.prototype.$convertToHex function (colorCode1, colorCode2, amount) {// 确保输入是字符串&#xff0c;并检查是否以0x开头let newCode1 let newCode2 if (t…...

Unity 窗口化设置

在Unity中要实现窗口化&#xff0c;具体设置如下&#xff1a; 在编辑器中&#xff0c;选择File -> Build Settings。在Player Settings中&#xff0c;找到Resolution and Presentation部分。取消勾选"Fullscreen Mode"&#xff0c;并选择"Windowed"。设…...

Android14之深入理解sp模板类(二百零二)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…...

.NET core 5.0 及以上的Windows Service开发

首先&#xff0c;一定要和.NET Framework区分开&#xff0c; 详细请看微软的2023年的最新官方文档 Create Windows Service using BackgroundService - .NET | Microsoft Learn Create a Windows Service installer - .NET | Microsoft Learn 同样微软的官方微博给出了开发…...

Nginx配置文件解释

Nginx可以作为静态页面的web服务器&#xff0c;同时还支持CGI协议的动态语言&#xff0c;比如perl、php等。但是不支持java。Java程序只能通过与tomcat配合完成。Nginx专为性能优化而开发&#xff0c;性能是其最重要的考量,实现上非常注重效率 &#xff0c;能经受高负载的考验,…...

R语言赋值符号<-、=、->、<<-、->>的使用与区别

R语言的赋值符号有&#xff1c;-、、-&#xff1e;、&#xff1c;&#xff1c;-、-&#xff1e;&#xff1e;六种&#xff0c;它们的使用与区别如下: <-’&#xff1a;最常用的赋值符号。它将右侧表达式的值赋给左侧的变量&#xff0c;像一个向左的箭头。例如&#xff0c;x …...

ffmpeg重点之时间戳,PTS、DTS、time_base

PTS和DTS和时间基time_base 首先我们知道PTS是一帧音频或视频显示的时间&#xff0c;DTS是解码时间戳 既然是时间&#xff0c;PST和DTS的单位是什么呢?秒还是毫秒&#xff0c;抑或是纳秒&#xff1f; 先说结论—都不是 先引入FFmpeg中时间基的概念&#xff0c;也就是time_bas…...

OpenGL 实现“人像背景虚化“效果

手机上的人像模式,也被人们称作“背景虚化”或 ”双摄虚化“ 模式,也称为 Bokeh 模式,能够在保持画面中指定的人或物体清晰的同时,将其他的背景模糊掉。突出画面的主体部分,主观上美感更强烈。 人像模式的一般实现原理是,利用双摄系统获取景深信息,并通过深度传感器和图…...

基于springboot实现校园周边美食探索及分享平台系统项目【项目源码+论文说明】计算机毕业设计

基于springboot实现园周边美食探索及分享平台系统演示 摘要 美食一直是与人们日常生活息息相关的产业。传统的电话订餐或者到店消费已经不能适应市场发展的需求。随着网络的迅速崛起&#xff0c;互联网日益成为提供信息的最佳俱渠道和逐步走向传统的流通领域&#xff0c;传统的…...

AUTOSAR关于内存栈的分层及描述

首先关于关于内存栈的分层&#xff1a;如下图所示&#xff0c;Nvm靠近RTE的&#xff1b;MemIf居中&#xff0c;EA和FEE被包含其中。 其次关于这三层的缩写&#xff1a;可以看到EEPROM的模拟和EEPROM的抽象层。 我们可以看到 大概的数据流&#xff1a; 和大致的结构分层作用&am…...

windows powershell连接linux 上传下载文件

连接&#xff1a;输入下面命令&#xff0c;回车 输入密码进入linux系统 ssh root192.168.188.128退出linux logoutwindow上传文件到Linux服务器 把桌面的123.txt 上传到linux home文件夹下 scp C:\Users\pzx\Desktop\123.txt root192.168.188.128:/homelinux下载文件到windo…...

Vue生命周期,从听说到深入理解(全面分析)

每个 Vue 组件实例在创建时都需要经历一系列的初始化步骤&#xff0c;比如设置好数据侦听&#xff0c;编译模板&#xff0c;挂载实例到 DOM&#xff0c;以及在数据改变时更新 DOM。在此过程中&#xff0c;它也会运行被称为生命周期钩子的函数&#xff0c;让开发者有机会在特定阶…...

故障诊断 | 一文解决,CNN-BiLSTM卷积神经网络-双向长短期记忆神经网络组合模型的故障诊断(Matlab)

效果一览 文章概述 故障诊断 | 一文解决,CNN-BiLSTM卷积神经网络-双向长短期记忆神经网络组合模型的故障诊断(Matlab) 模型描述 CNN-BiLSTM卷积神经网络-双向长短期记忆神经网络组合模型是一种深度学习模型,结合了卷积神经网络(CNN)和双向长短期记忆网络(BiLSTM)的优点…...

iOS library not found for -lMBProgressHUD

0x00 前因 一开始是使用 CocoaPods 管理 MBProgressHUD&#xff0c;后来直接导入 MBProgressHUD 源码&#xff0c;就出现了这个错误&#xff1a;library not found for -lMBProgressHUD 0x01 后果 在 Xcode 工程目录中找到文件夹&#xff1a;Frameworks 看看里面是否有个红色…...

Paper Digest|基于在线聚类的自监督自蒸馏序列推荐模型

论文标题&#xff1a; Leave No One Behind: Online Self-Supervised Self-Distillation for Sequential Recommendation 作者姓名&#xff1a; 韦绍玮、吴郑伟、李欣、吴沁桐、张志强、周俊、顾立宏、顾进杰 组织单位&#xff1a; 蚂蚁集团 录用会议&#xff1a; WWW 2024 …...

【CTFshow 电子取证】套的签到题

&#x1f36c; 博主介绍&#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 hacker-routing &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【应急响应】 【Java、PHP】 【VulnHub靶场复现】【面试分析】 &#x1f389;点赞➕评论➕收…...

SpringBoot集成Solr全文检索

SrpingBoot 集成 Solr 实现全文检索 一、核心路线 使用 Docker 镜像部署 Solr 8.11.3 版本服务使用 ik 分词器用于处理中文分词使用 spring-boot-starter-data-solr 实现增删改查配置用户名密码认证使用 poi 和 pdfbox 组件进行文本内容读取文章最上方有源码和 ik 分词器资源…...

厨余垃圾处理设备工业监控PLC连接APP小程序智能软硬件开发之功能原理篇

接着上一篇《厨余垃圾处理设备工业监控PLC连接APP小程序智能软硬件开发之功能结构篇》继续总结一下厨余垃圾处理设备智能软硬件统的原理。所有的软硬件系统全是自己一人独自开发&#xff0c;看法和角度难免有局限性。希望抛砖引玉&#xff0c;将该智能软硬件系统分享给更多有类…...

google浏览器网站不安全与网站的连接不安全怎么办?

使用google谷歌浏览器访问某些网站打开时google谷歌浏览器提示网站不安全,与网站的连接不安全,您之所以会看到此警告,是因为该网站不支持https造成的怎么办? 目录 1、打开谷歌google浏览器点击右上角【┇】找到设置...

基于Axios封装请求---防止接口重复请求解决方案

一、引言 前端接口防止重复请求的实现方案主要基于以下几个原因&#xff1a; 用户体验&#xff1a;重复发送请求可能导致页面长时间无响应或加载缓慢&#xff0c;从而影响用户的体验。特别是在网络不稳定或请求处理时间较长的情况下&#xff0c;这个问题尤为突出。 服务器压力…...

深入理解指针(7)函数指针变量及函数数组(文章最后放置本文所有原码)

一、函数指针变量 什么是函数指针变量呢&#xff1f; 既然是指针变量&#xff0c;那么它指向的一定是地址&#xff0c;而且我们可以通过地址来调用函数的。 函数是否有地址呢&#xff1f;地址是什么&#xff1f; 经过上面的测试可以看到函数也是有地址的&#xff0c;而且其地…...

office办公技能|word中的常见使用问题解决方案2.0

一、设置多级列表将表注从0开始&#xff0c;设置为从1开始 问题描述&#xff1a;word中插入题注&#xff0c;出来的是表0-1&#xff0c;不是1-1&#xff0c;怎么办&#xff1f; 写论文时&#xff0c;虽然我设置了“第一章”为一级标题&#xff0c;但是这三个字并不是自动插入的…...

华为2023年年度报告启示:大学生如何把握未来科技趋势,规划个人发展路径

华为2023年年度报告展现了公司在技术创新、生态构建、社会责任等方面的卓越成就与前瞻布局。对于身处数字化时代的大学生而言&#xff0c;这份报告不仅是洞察科技行业发展趋势的窗口&#xff0c;更是规划个人学业与职业道路的重要参考。本文将从报告中提炼关键信息&#xff0c;…...

刚刚,璞华科技、璞华易研PLM产品荣获智能制造领域两大奖项!

刚刚&#xff0c;在e-works数字化企业网于北京举办的“第十三届中国智能制造高峰论坛暨第二十一届中国智能制造岁末盘点颁奖典礼”上&#xff0c;璞华科技凭借在智能制造领域的雄厚实力和产品口碑&#xff0c;荣获两大奖项。 璞华科技被评为e-works【2023年度智能制造优秀供应…...

乐维更改IP地址

1.1 系统IP调整 vim /etc/sysconfig/network-scripts/ifcfg-ens1921.2 Web相关服务IP变更 1.2.1 编辑/itops/nginx/html/lwjkapp/.env文件,更改ZABBIXSERVER、ZABBIXRPCURL、DB_HOST中的IP 1.2.2 进入/itops/nginx/html/lwjk_app/目录下,执行php bin/manager process-conso…...

大话设计模式之简单工厂模式

简单工厂模式&#xff08;Simple Factory Pattern&#xff09;是一种创建型设计模式&#xff0c;属于工厂模式的一种。在简单工厂模式中&#xff0c;有一个工厂类负责根据输入参数的不同来创建不同类的实例。 简单工厂模式包含以下几个要素&#xff1a; 1. **工厂类&#xff0…...

设计模式之单例模式精讲

UML图&#xff1a; 静态私有变量&#xff08;即常量&#xff09;保存单例对象&#xff0c;防止使用过程中重新赋值&#xff0c;破坏单例。私有化构造方法&#xff0c;防止外部创建新的对象&#xff0c;破坏单例。静态公共getInstance方法&#xff0c;作为唯一获取单例对象的入口…...

论文复现3:Stable Diffusion v1

abstract: 通过将图像形成过程分解为去噪自动编码器的顺序应用,扩散模型 (DM) 在图像数据及其他方面实现了最先进的合成结果。此外,他们的公式允许一种指导机制来控制图像生成过程,而无需重新训练。然而,由于这些模型通常直接在像素空间中运行,因此强大的 DM 的优化通常会…...

Halcon与VisionMaster对比

作为一个经验丰富的机器视觉算法工程师&#xff0c;我对于机器视觉软件的评价会基于多年的实践经验和对不同软件功能的深入了解。在评价VisionMaster和Halcon软件时&#xff0c;我会从使用场景、工作效率、使用便捷性等方面进行全面分析&#xff0c;并结合软件的优缺点进行讨论…...

西安公司做网站/windows优化大师最新版本

cmd 进入E文件夹 E: 查看文件夹目录 dir 进入某个文件夹 cd 目录...

无锡网站制作公司排名/百度一下生活更好

型号&#xff1a;TD1000&#xff0c;TD2000华为TD1000&#xff0c;TD2000系列变频器1&#xff0e;TD1000G系列(单相&#xff0c;220V)型号 功率(KW)TD1000-2S0007G 0&#xff0e;75TD1000-2S0015G 1.5TD1000-2S0022G 2.22&#xff0e;TD1000G系列(三相&#xff0c;380V)型号 功…...

怎么做投票管理系统后台网站/专业软文平台

(给CPP开发者加星标&#xff0c;提升C/C技能)来源&#xff1a;鲸90830https://www.cnblogs.com/whale90830/p/10542361.html1) const 对象的一般形式类型名 const 对象名[(构造实参表列)];const 类型名 对象名[(构造实参表列)];注意&#xff1a;常对象必须要有初值(因为以后不能…...

龙岗网站推广/永久免费制作网页

我这个菜鸟真的弱爆了~~弱爆了~~以前竟然在写着超级垃圾的快速幂&#xff0c;彻底服了自己~~无语~~实在无语 program1(弱智快速幂取模) 图1&#xff1a;猴子爬下去&#xff1a; 图2&#xff1a;猴子很老实滴爬回来 #include<iostream> using namespace std; //求2^N%20…...

合作网站seo/友情链接多久有效果

题目 题目大意 平面上有一堆带权值的点。两种操作&#xff1a;交换两个点的权值&#xff0c;查找一个矩形的第kkk小 N<60000N<60000N<60000 M<10000M<10000M<10000 10000ms10000ms10000ms 思考历程&各种可能过的方法 先是想了一会儿&#xff0c;然后突…...

wordpress程序漏洞集合/百度账号个人中心

2019独角兽企业重金招聘Python工程师标准>>> Struts2上传文件默认大小是2M,超过则报错。解决办法是在struts.xml中配置&#xff1a; <struts> <constant name "struts.multipart.maxSize" value "10485760"></constant > <…...