【Linux】掌握库的艺术:我的动静态库封装之旅
🌈个人主页:Yui_
🌈Linux专栏:Linux
🌈C语言笔记专栏:C语言笔记
🌈数据结构专栏:数据结构
🌈C++专栏:C++
文章目录
- 1.什么是库
- 1.2 认识动静态库
- 1.2.1 动态库
- 1.2.2 静态库
- 2.封装动静态库
- 2.1 封装动态库
- 2.1.1 组织头文件和库文件
- 2.1.2 动态库的使用
- 2.2 封装静态库
- 2.2.1 使用静态库
- 3. 动静态库小知识
- 3.1 gcc对动静态的优先级
- 3.2 动静态库的区别
- 4.总结
1.什么是库
在计算机编程中,库(Library)是一个预先编写的代码集合,包含了可以被其他程序调用的函数、类、变量和资源。库的主要目的是为了简化编程过程,提供常用功能的实现,促进代码重用,从而减少开发时间和提高软件的可靠性。
在实践中,我们一定会使用到别人的库,如你在C语言时期一定会调用到c标准库。除了标准库,我们还可以使用第三方库,无论使用那种库都是为了节省我们的时间,不在需要我们自己来造“轮子”。
所有的库都可以从两个方面来认识:
- 创建者
- 使用者
提问:使用者在使用库时,是否能知道该库的源代码呢?
回答:在不逆向的情况下,使用者是无法得知库的源代码的,这也就牵扯到了,库的第二个属性隐藏源代码。
我们都知道,形成可执行文件的步骤有4步:
- 预处理:头文件展开、去注释、宏替换、条件编译等,生成
.i
文件。- 编译:语法分析、语义分析、符号汇总等,检查无误后将代码翻译成汇编指令,生成
.s
文件,- 汇编:将汇编指令转化成二进制指令,生成
.o
文件。- 链接:将生成的各个
.o
文件进行链接,生成可执行程序。
而库就是所有.o
文件用特定的方式,进行打包,形成的一个文件。
注意库文件需要配合对应的头文件进行使用,头文件不必隐藏
当我们在main.c
中使用对应的功能函数:
当有许多不同的源文件去调用这些功能函数时,那要的话,功能函数会被重复的进行预处理、编译、汇编的操作,各自生成.o
文件,然后我们的源文件再和这些功能函数一起生成可执行文件。
如此操作会有许多重复的操作,我们完全可以提前让功能函数变成一个个.o
文件,再去和源文件进行链接,但是我们还要考虑到,可能会存在非常多的.o
文件,为了减轻我们的操作,我们可以对这些.o
文件进行打包,这样的文件我们称为库。
如此一来,库的本质就是若干个目标文件的集合,每一个目标文件都包含了由源码编译生成的二进制代码,在保证使用的同时,还有很好的隐藏性
1.2 认识动静态库
动静态库是编程中常用的两种库类型,用于封装和重用代码。它们在链接、加载和使用方面存在显著的差异。
1.2.1 动态库
动态库(或共享库)是在运行时加载的库,通常以 .so
(Linux)或 .dll
(Windows)文件格式存在。动态库的代码不被复制到可执行文件中,而是在运行时由操作系统加载。
我们先来写一段简单的代码:
#include <stdio.h>int main()
{printf("hello world!!!!\n");return 0;
}
//生成可执行文件mybin
编译成功后,我们使用ldd
指令来查看其所链接的动态库
ldd语法:
ldd filename
功能:
用于打印程序或库文件所依赖的动态库(共享库)列表。ldd不是一个可执行程序,只是一个shell脚本。
ubuntu@VM-20-9-ubuntu:~/libraryTest$ ldd mybin linux-vdso.so.1 (0x00007ffc9c4db000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fd8df175000)/lib64/ld-linux-x86-64.so.2 (0x00007fd8df3ab000)
libc.so 就是该程序所依赖的动态库,那么我们一个如何识别一个动态库的名字呢?
去掉前缀
lib
,再去掉后缀.so
就是库名了。
也就是说,这个动态库的名字就是c
,没错这就是c运行库。
1.2.2 静态库
静态库是将一组对象文件(.o
文件)打包成一个库文件(通常为 .a
后缀),在编译时将其链接到最终生成的可执行文件中。链接过程是在编译阶段完成的,库的代码被复制到可执行文件中。
正常情况下,gcc默认都是连接的动态库,如果需要进行静态连接要特别指定。
gcc -o staticBin libTest.c -staticubuntu@VM-20-9-ubuntu:~/libraryTest$ ls -l
total 904
-rw-rw-r-- 1 ubuntu ubuntu 82 Oct 29 11:40 libTest.c
-rw-rw-r-- 1 ubuntu ubuntu 74 Oct 29 11:41 makefile
-rwxrwxr-x 1 ubuntu ubuntu 15952 Oct 29 11:41 mybin
-rwxrwxr-x 1 ubuntu ubuntu 900344 Oct 29 14:49 staticBin
可以发现静态连接的可执行程序所占的空间大小远大于动态连接的。
使用ldd
观察是否有依赖的库
ubuntu@VM-20-9-ubuntu:~/libraryTest$ ldd staticBin not a dynamic executable
发现静态链接生成的可执行程序不依赖其他库文件。
使用file查看:
ubuntu@VM-20-9-ubuntu:~/libraryTest$ file staticBin
staticBin: ELF 64-bit LSB executable, x86-64, version 1 (GNU/Linux), statically linked, BuildID[sha1]=d4a39b68ac04f9fd8b37bd8c17ce42a8b27ad8c2, for GNU/Linux 3.2.0, not stripped
发现是:statically linked
如果你是centos用户,可能会遇到静态库未安装的情况,只需要搜索一下安装方法即可。
2.封装动静态库
2.1 封装动态库
先写好需要封装的代码:
add.h:
#pragma once
int add(int a,int b);
sub.h:
#pragma once
int sub(int a,int b);
add.c:
#include "add.h"
int add(int a,int b)
{return a+b;
}
sub.c:
#include "sub.h"
int sub(int a,int b)
{return a-b;
}
dynLibTest.c:
#include <stdio.h>
#include "add.h"
#include "sub.h"int main()
{int a = 100;int b = 200;printf("%d + %d = %d\n",a,b,add(a,b));printf("%d - %d = %d\n",a,b,sub(a,b));return 0;
}
然后我们还需要知道位置无关码的概念(Position-Independent Code,PIC)
位置无关码(Position-Independent Code,PIC)是一种编译代码的方式,使得生成的代码可以在内存的任意位置执行,而不需要修改代码中的地址。这种特性在动态链接库和共享库中非常重要,因为它们可以被多个进程共享,并在加载时被放置到不同的内存地址。
为了实现位置无关,编译器在生成代码时使用相对地址而不是绝对地址。例如,在访问全局变量时,编译器不会生成直接访问变量的绝对地址的代码,而是使用相对于当前指令位置的偏移量。这样,无论代码被加载到哪个地址,访问都可以通过相对计算来实现。
位置无关码是一种重要的编程技术,尤其在动态链接和共享库中具有广泛的应用。它提供了灵活性和内存使用效率,同时也增强了程序的安全性。
如果我要实现位置无关码可以在gcc后面加上-fPIC
选项
gcc -fPIC -c add.c
gcc -fPIC -c sub.c
我们也知道不加-fPIC
一样可以生成.o
文件。但是它们之间还是很有区别的。
位置无关码对于gcc:
fPIC
作用于编译阶段,告诉编译器于位置无关的代码,此时产生的代码中没有绝对地址,全部都使用相对地址,从而代码可以被加载到内存的任意位置可以正确的执行。这正是共享库被加载时,在内存的位置不是不是固定的。- 如果不加
-fPIC
选项,则加载.so
文件的代码时,代码段引用的数据对象需要重定位,这个重定位会修改代码段地内容,这就会造成每一个使用这个.so
文件代码段的内核里都会生成这个.so
文件代码的拷贝,并且每一个拷贝都不一样,这就是使得内存的消耗变大。 - 为此我们总是会用
-fPIC
来生成.so
文件,但是不会用点-fPIC
来生成.a
静态文件。我们当然可以不用-fPIC
来生成.so
文件,只是这样的话.so
文件必须要在加载到用户的地址空间时重定向所有表目。
使用-shard
将文件打包为动态库
gcc -shared -o libmyc.so add.o sub.o
通过readelf -S
来查看库的信息,如偏移量offset
。
2.1.1 组织头文件和库文件
为了方便我们后续的操作,我会写一个makefile来组织头文件和库文件。
.PHONY:all
all:libmyc.solibmyc.so:add.o sub.ogcc -shared add.o sub.o -o libmyc.so
add.o:add.cgcc -fPIC -c add.c -o add.o
sub.o:sub.cgcc -fPIC -c sub.c -o sub.o
.PHONY:output
output:mkdir -p mylib/libmkdir -p mylib/includecp -rf *.h mylib/includecp -rf *.so mylib/lib
.PHONY:clean
clean:rm -rf *.o mylib *.a *.so output
2.1.2 动态库的使用
当我们把当前目录的头文件和目标文件删除后
ubuntu@VM-20-9-ubuntu:~/libraryTest$ rm -rf *.o *.h
此时如果我们直接使用gcc来编译会发生什么呢?
没错,肯定会报错说,找不到头文件啦。
ubuntu@VM-20-9-ubuntu:~/libraryTest$ gcc dynLibTest.c
dynLibTest.c:2:10: fatal error: add.h: No such file or directory2 | #include "add.h"| ^~~~~~~
compilation terminated.
为了让编译器能够找到我们的头文件,需要我们加上-I 头文件所在路径
gcc -I./mylib/include dynLibTest.c
现在的错误就是,找不到目标函数了,因为我们没有给编译器指明在哪里,它肯定就找不到了。
ubuntu@VM-20-9-ubuntu:~/libraryTest$ gcc -I./mylib/include dynLibTest.c
/usr/bin/ld: /tmp/ccOkC8RY.o: in function `main':
dynLibTest.c:(.text+0x25): undefined reference to `add'
/usr/bin/ld: dynLibTest.c:(.text+0x52): undefined reference to `sub'
collect2: error: ld returned 1 exit status
所有我们还需要加上`-L库文件所在路径 l库名
gcc -I./mylib/include -L./mylib/lib dynLibTest.c -lmyc
- ‘-I’:指定头文件搜索路径。
—L
:指定库文件搜索路径。-l
:指明需要链接库文件路径下的哪一个库。
然后我们就可以得到一个可执行文件了,但是如果你去执行它就会发现,它居然不能执行!!!
ubuntu@VM-20-9-ubuntu:~/libraryTest$ ./a.out
./a.out: error while loading shared libraries: libmyc.so: cannot open shared object file: No such file or directory
为什么会这样?
算了,先用ldd看看它依赖的动态库的吧。
居然没有连到,岂有此理,辛苦打了一连串指令竟然完全没作用。
其实不是这样啦,因为是动态库,在执行时我们仍然需要知道动态库的位置在哪。
对于动态库,编译时会搜索库的路径,运行时也会搜索库的路径。
为了解决这个问题,我们有4种解决方案:
- 直接将库进行安装(拷贝)到系统库当中。(最傻瓜操作)
- 将不在系统默认库搜索路径下的库路径,添加到
LD_LIBRARY_PATH
3.使用idconfig指令。
拷贝到系统目录
ubuntu@VM-20-9-ubuntu:~/libraryTest$ sudo cp mylib/lib/libmyc.so /lib/x86_64-linux-gnu/
ubuntu@VM-20-9-ubuntu:~/libraryTest$ ldd a.out linux-vdso.so.1 (0x00007fffc4f54000)libmyc.so => /lib/x86_64-linux-gnu/libmyc.so (0x00007f35c8f9c000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f35c8d73000)/lib64/ld-linux-x86-64.so.2 (0x00007f35c8fae000)
这样虽然简单,但是由于我们自己书写的库,大概率是不成熟的,这样会污染系统库目录。
更改环境变量
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:home/ubuntu/libraryTest/mylib/lib
ldd查看:
ubuntu@VM-20-9-ubuntu:~/libraryTest$ export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:home/ubuntu/libraryTest/mylib/lib
ubuntu@VM-20-9-ubuntu:~/libraryTest$ ldd a.outlinux-vdso.so.1 (0x00007fff8fdee000)libmyc.so (0x00007f85aaf99000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f85aad6a000)/lib64/ld-linux-x86-64.so.2 (0x00007f85aafa5000)
该方法为临时方法,在系统重新登入时就失效了。
使用Idconfig指令
/etc/ld.so.conf.d/目录下的文件用来指定动态库搜索路径。这些文件被包含在/etc/ld.so.conf文件中,ldconfig命令会在默认搜寻目录(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所列的目录下,搜索可共享的动态链接库,并创建出动态装入程序(ld.so)所需的连接和缓存文件。
.conf
文件用来存储各种文件的路径,我们只要把我们自己写的第三方库的路径存放进去,程序运行时就会去里面找了。
echo /home/ubuntu/libraryTest/mylib/lib > libmyc.conf
然后我们需要将,.conf
文件拷贝到/etc/ld.so.conf.d/
下。
sudo cp libmyc.conf /etc/ld.so.conf.d/
此时如果我们直接ldd是无法找到的,我们来需要更新一下。
sudo ldconfig
然后就可以了
那么动态库就先到这里,下面我们开始静态库的讲解。
2.2 封装静态库
静态库的操作会比动态库的更为简单。
代码的话,依然是上面的那些代码。
add.c sub.c add.h sub.h staLibTest.c
staLibTest.c
里面的代码和dynLibTest.c
首先我们需要把所有的.c
源文件都编译为目标文件。(此时不在需要位置无关码)
gcc -c add.c
gcc -c sub.c
我们可以直接将主程序和这些.o
文件进行编译。
ubuntu@VM-20-9-ubuntu:~/libraryTest$ gcc -o staticBin staLibTest.c add.o sub.o
ubuntu@VM-20-9-ubuntu:~/libraryTest$ ./staticBin
100 + 200 = 300
100 - 200 = -100
正常编译,不过目标文件少的情况下还好,如果有很多的目标文件,我们就有点吃力了。为此我们可以打一个包。
使用ar
指令将目标文件打包为静态库:
ar
指令是一个用于创建、修改和管理静态库文件的命令行工具,广泛应用于 Unix 和类 Unix 系统(如 Linux)。静态库通常以.a
后缀表示,ar
允许开发者将多个目标文件(.o
文件)打包成一个单一的库文件,便于代码重用和管理。
语法:
ar [选项] [库名] [依赖文件]
主要功能:
r
:插入文件。如果目标文件已经存在,则替换它。c
:创建一个新的库,如果库文件不存在。s
:创建索引,以加快链接过程。t
:列出库中包含的文件。x
:从库中提取指定的文件。d
:从库中删除指定的文件。v
:查看库的信息
举例:将add.o
和sub.o
打包为静态库
ar -rc libmyc.a add.o sub.o
利用-t -v
选项来查看静态库的文件以及信息。
ubuntu@VM-20-9-ubuntu:~/libraryTest$ ar -tv libmyc.a
rw-r--r-- 0/0 1224 Jan 1 08:00 1970 add.o
rw-r--r-- 0/0 1224 Jan 1 08:00 1970 sub.o
和动态库时一样,我们现在利用makefile来完成这些操作
libmyc.a:add.o sub.oar -rc libmyc.a add.o sub.o
add.o:add.cgcc -o add.o -c add.c
sub.o:sub.cgcc -o sub.o -c sub.c.PHONY:output
output:mkdir -p mylibs/libmkdir -p mylibs/includecp -rf *.h mylibs/includecp -rf *.a mylibs/lib
.PHONY:clean
clean:rm -rf *.o mylibs libmyc.a
2.2.1 使用静态库
- ‘-I’:指定头文件搜索路径。
—L
:指定库文件搜索路径。-l
:指明需要链接库文件路径下的哪一个库。
在动态库的篇章中,我们就已经了解了指定库的路径了。
在静态库也同理。
gcc staLibTest.c -I./mybins/include -L./mylibs/lib -lmyc
编译完后就可以直接运行了。
ubuntu@VM-20-9-ubuntu:~/libraryTest$ gcc staLibTest.c -I./mybins/include -L./mylibs/lib -lmyc -o testbin -static
ubuntu@VM-20-9-ubuntu:~/libraryTest$ ./testbin
100 + 200 = 300
100 - 200 = -100
同样的,如果你觉得没错都显示指定库连接很麻烦,可以把目标文件拷贝到系统库当中。
3. 动静态库小知识
3.1 gcc对动静态的优先级
如果我们同时提供动态库和静态库,gcc会默认使用动态库。如果我们非要静态链接,必须使用static指定。如果我们只提供静态库,那我们的程序只能对该库进行静态链接,但是程序不一定整体是静态链接的。如果我们只提供动态库,默认只能动态连接,非要静态链接的话会报错。
3.2 动静态库的区别
特性 | 静态库 | 动态库 |
---|---|---|
文件后缀 | .a | .so (Linux), .dll (Windows) |
链接方式 | 编译时链接 | 运行时链接 |
可执行文件大小 | 较大(包含所有库代码) | 较小(只包含引用) |
外部依赖 | 无 | 需要在运行时提供 |
更新方式 | 需要重新编译所有依赖的程序 | 只需替换库文件 |
性能 | 加载速度快 | 加载速度相对较慢 |
共享性 | 不支持多个进程共享 | 支持多个进程共享 |
4.总结
动静态库各有优缺点,选择使用哪种库通常取决于具体的项目需求、资源限制和开发环境。静态库适用于对依赖性和更新不敏感的应用,而动态库则更灵活,适合需要频繁更新和共享代码的场景。在实际开发中,合理选择和使用这两种库能够提高代码的复用性和维护性。
相关文章:

【Linux】掌握库的艺术:我的动静态库封装之旅
🌈个人主页:Yui_ 🌈Linux专栏:Linux 🌈C语言笔记专栏:C语言笔记 🌈数据结构专栏:数据结构 🌈C专栏:C 文章目录 1.什么是库1.2 认识动静态库1.2.1 动态库1.2.2…...

UE5动画控制 基础
素材 mixamo先去选择一个character 点击下载 就这个下载下来 然后选几个animation, 记得勾选 把动作下载了 without skin就是只要动作 然后把他们放在一个文件夹里先 UE里导入 找一个文件夹,直接拖拽进来那个character的fbx,默认配置就…...

流畅!HTMLCSS打造网格方块加载动画
效果演示 这个动画的效果是五个方块在网格中上下移动,模拟了一个连续的加载过程。每个方块的动画都是独立的,但是它们的时间间隔和路径被设计为相互协调,以创建出流畅的动画效果。 HTML <div class"loadingspinner"><…...

linux命令之top(Linux Command Top)
💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 本人主要分享计算机核心技…...
数据结构-希尔排序(ShellSort)笔记
看动画理解 【数据结构】八大排序(超详解附动图源码)_数据结构排序-CSDN博客 一 基本思想 先选定一个整数gap,把待排序文件中所有记录分成gap个组,所有距离为gap的记录分在同一组内,并对每一组内的元素进行排序。 然后将gap逐渐减小重复上…...

Junit + Mockito保姆级集成测试实践
一、做好单测,慢即是快 对于单元测试的看法,业界同仁理解多有不同,尤其是在业务变化快速的互联网行业,通常的问题主要有,必须要做吗?做到多少合适?现在没做不也挺好的吗?甚至一些大…...
软件项目管理要点
一.项目管理 1.盈亏平衡分析 销售额固定成本可变成本税费利润 当利润为0的时候就是盈亏平衡点。 2.范围管理 范围定义的输入包括:项目章程、项目范围管理计划、组织过程资产、批准的变更申请。 3.时间管理 项目时间管理中的过程包括活动定义、活动排序、活动的资…...

ESP8266 连接 MQTT 服务器EMQX 连接MQTTX
目录 1.先用有一台自己的云服务器 2. 使用FinalShell连接阿里云云服务器ECS 3.安装宝塔 4.在云服务器打开8888端口 5.使用外网面板地址打开宝塔面板 6.安装Docker 7.下载emqx 8.打开emqxWeb 界面 9.下载MQTTX 10.EMQX加一个客户端 11.开始通信 12.加入单片机ESP8266 …...
Python中如何处理异常情况?
1、Python中如何处理异常情况? 在Python中,处理异常情况通常使用try/except语句。try语句块包含可能会引发异常的代码,而except语句块包含处理异常的代码。如果try块中的代码引发了异常,控制流将立即转到相应的except块。 以下是…...

openpnp - 在openpnp中单独测试相机
文章目录 openpnp - 在openpnp中单独测试相机概述笔记END openpnp - 在openpnp中单独测试相机 概述 底部相机的位置不合适, 重新做了零件,准备先确定一下相机和吸嘴的距离是多少才合适。 如果在设备上直接实验,那么拆装调整相机挺麻烦的。 准备直接在电…...
Spark窗口函数
1、 Spark中的窗口函数 窗口就是单纯在行后面加一个列 可以套多个窗口函数,但彼此之间不能相互引用,是独立的 窗口函数会产生shuffle over就是用来划分窗口的 (1) 分组聚合里面的函数,基…...

Idea、VS Code 如何安装Fitten Code插件使用
博主主页:【南鸢1.0】 本文专栏:JAVA 目录 编辑 简介 所用工具 1、Idea如何安装插件 1.idea下载插件 2.需要从外部下载然后在安装, 2、VS Code如何安装插件 总结 简介 Fitten Code是由非十大模型驱动的AI编程助手,它可以自动生成代…...
elasticsearch7.x在k8s中的部署
一、说明 二、思路 三、部署 1、建nfs服务器 2、建持久卷 3、部署elasticsearch 四、附件 ?pv.yaml内容 elasticsearch.yaml内容 一、说明 本文章内容主要的参考来源是https://www.cnblogs.com/javashop-docs/p/12410845.html,但参考文献中的elasticsearc…...

校园社团信息管理平台:Spring Boot技术实战指南
3系统分析 3.1可行性分析 通过对本校园社团信息管理系统实行的目的初步调查和分析,提出可行性方案并对其一一进行论证。我们在这里主要从技术可行性、经济可行性、操作可行性等方面进行分析。 3.1.1技术可行性 本校园社团信息管理系统采用SSM框架,JAVA作…...

【Linux】从内核角度理解 TCP 的 全连接队列(以及什么是 TCP 抓包)
文章目录 概念引入理解全连接队列内核方面理解Tcp抓包方法注意事项 概念引入 我们知道,TCP的三次握手是由TCP协议 自动处理的,建立连接的过程与用户是否进行accept无关,accept()的作用主要是为当前连接创建一个套接字,用于进行后…...

太速科技-712-6U VPX飞腾处理器刀片计算机
6U VPX飞腾处理器刀片计算机 一、产品概述 该产品是一款基于国产飞腾FT-2000四核处理器或D2000八核处理器的高性能6U VPX刀片式计算机。产品提供了可支持全网状交换的高速数据通道,其中P1、P2均支持1个PCIe x16 Gen3或2个PCIe x8 Gen3或4个PCIe x4 Gen3总…...

深度学习(八) TensorFlow、PyTorch、Keras框架大比拼(8/10)
一、深度学习框架概述 深度学习框架在当今人工智能和机器学习领域中占据着至关重要的地位。其中,TensorFlow 由 Google 开发,自 2015 年发布以来,凭借其灵活的计算图、自动微分功能以及跨平台支持等特点,迅速成为主流深度学习框架…...
thinkphp中命令行工具think使用,可用于快速生成控制器,模型,中间件等
在使用tp框架的时候tp内置了一个think的命令行工具, 这个工具可以帮助我们快速的生成控制器,模型,中间件的代码。 这个工具就和laravel中的 artisan 的作用都差不多,只是名称不同而已。 tp中常用的命令行工具: 1. 快…...

Discourse 是否支持手机注册
Discourse 能不能使用手机进行注册? 功能总结 简答来说,Discourse 不能使用手机注册,Discourse 也没有提供这个功能。 这个功能多是需要 SSO 来实现。 比如说华为的那个社区,他们自己做了一个自己的用户管理 SSO,然…...

软件测试学习笔记丨Flask框架-请求与响应
本文转自测试人社区,原文链接:https://ceshiren.com/t/topic/23408 请求方法 from flask import Flaskapp Flask(__name__)app.route("/cases", methods["get"]) def get_case():return {"code": 0, "msg": &…...
KubeSphere 容器平台高可用:环境搭建与可视化操作指南
Linux_k8s篇 欢迎来到Linux的世界,看笔记好好学多敲多打,每个人都是大神! 题目:KubeSphere 容器平台高可用:环境搭建与可视化操作指南 版本号: 1.0,0 作者: 老王要学习 日期: 2025.06.05 适用环境: Ubuntu22 文档说…...
Linux链表操作全解析
Linux C语言链表深度解析与实战技巧 一、链表基础概念与内核链表优势1.1 为什么使用链表?1.2 Linux 内核链表与用户态链表的区别 二、内核链表结构与宏解析常用宏/函数 三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势5.1 插入效率5.2 安全…...

通过Wrangler CLI在worker中创建数据库和表
官方使用文档:Getting started Cloudflare D1 docs 创建数据库 在命令行中执行完成之后,会在本地和远程创建数据库: npx wranglerlatest d1 create prod-d1-tutorial 在cf中就可以看到数据库: 现在,您的Cloudfla…...

PL0语法,分析器实现!
简介 PL/0 是一种简单的编程语言,通常用于教学编译原理。它的语法结构清晰,功能包括常量定义、变量声明、过程(子程序)定义以及基本的控制结构(如条件语句和循环语句)。 PL/0 语法规范 PL/0 是一种教学用的小型编程语言,由 Niklaus Wirth 设计,用于展示编译原理的核…...

Ascend NPU上适配Step-Audio模型
1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统,支持多语言对话(如 中文,英文,日语),语音情感(如 开心,悲伤)&#x…...
关于 WASM:1. WASM 基础原理
一、WASM 简介 1.1 WebAssembly 是什么? WebAssembly(WASM) 是一种能在现代浏览器中高效运行的二进制指令格式,它不是传统的编程语言,而是一种 低级字节码格式,可由高级语言(如 C、C、Rust&am…...

c#开发AI模型对话
AI模型 前面已经介绍了一般AI模型本地部署,直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型,但是目前国内可能使用不多,至少实践例子很少看见。开发训练模型就不介绍了&am…...

UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)
UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中,UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化…...
Caliper 配置文件解析:config.yaml
Caliper 是一个区块链性能基准测试工具,用于评估不同区块链平台的性能。下面我将详细解释你提供的 fisco-bcos.json 文件结构,并说明它与 config.yaml 文件的关系。 fisco-bcos.json 文件解析 这个文件是针对 FISCO-BCOS 区块链网络的 Caliper 配置文件,主要包含以下几个部…...

RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文全面剖析RNN核心原理,深入讲解梯度消失/爆炸问题,并通过LSTM/GRU结构实现解决方案,提供时间序列预测和文本生成…...