Rk3588 Android12 AIDL 开发
AIDL (Android Interface Definition Language) 和 HIDL (HAL Interface Definition Language) 都是 Android 系统中用于定义接口的工具,但它们有不同的用途和特性。
AIDL (Android Interface Definition Language)
-
用途:
- 主要用于应用程序之间的进程间通信 (IPC)。
- 适用于定义应用程序组件(如 Activity 和 Service)之间的接口。
HIDL (HAL Interface Definition Language)
-
用途:
- 主要用于定义硬件抽象层 (HAL) 接口。
- 适用于访问底层硬件功能,如摄像头、传感器等。
未来趋势
从 Android 13 开始,HIDL 已被弃用,建议使用 AIDL 来实现 HAL 接口⁴。这意味着未来的 Android 版本将更多地依赖 AIDL 来实现硬件抽象层的接口。
本文主要记录自己实现一个AIDL的过程
主要的架构逻辑
1.使能hidl-gen工具
source build/envsetup.sh
lunch rk3588_s-userdebug
make hidl-gen
2.建立自己的AIDL ,test目录 和ITest.hal,ITestWarpper.hal
1.mkdir vendor/rockchip/hardware/interfaces/test
2.mkdir vendor/rockchip/hardware/interfaces/test/1.03.新建ITest.hal文件
//vendor/rockchip/hardware/interfaces/test/1.0/ITest.hal
package rockchip.hardware.test@1.0;
import ITestWarpper;interface ITest
{setTestWarpper(ITestWarpper testWarpper);GetString()generates (string printString);
};4.新建ITestWarpper.hal文件 用于数据传递
//vendor/rockchip/hardware/interfaces/test/1.0/ITestWarpper.hal
package rockchip.hardware.test@1.0;
interface ITestWarpper
{GetString()generates (string printString);
};
生成编译文件vendor/rockchip/hardware/interfaces/test/1.0/Android.bp
do_makefiles_update rockchip.hardware:vendor/rockchip/hardware android.hardware:hardware/interfaces android.hidl:system/libhidl/transport
AIDL 的没有返回值都是VOID ,出参都是 generates (string printString);实现,类似C++的别名功能
生成对应的CPP和头文件
1.hidl-gen -o vendor/rockchip/hardware/test/1.0/defualt -Lc++-impl -rrockchip.hardware:vendor/rockchip/hardware -randroid.hidl:system/libhidl/transport rockchip.hardware //生成C++ 和 H2.hidl-gen -o vendor/rockchip/hardware/test/1.0/defualt -Landroidbp-impl -rrockchip.hardware:vendor/rockchip/hardware/ -randroid.hidl:system/libhidl/transport rockchip.hardware //生成Android.dp在defult 目录下生成对应的cpp文件和头文件,Android.bp
也可以使用我写的脚本生成对应的文件
./generate-source2.sh vendor/rockchip/hardware/interfaces/ test rockchip.hardware
#! /bin/bash
# v1.0if [ -z $1 ]; thenecho "Please input path "exit 1
elseLOC=$1
fiif [ -z $2 ]; thenecho "Please input module name"exit 1
elseMODULE=$2
fiif [ -z $3 ]; thenecho "Please input package head"exit 1
elsePACKAGE_HEAD=$3
fiPACKAGE=$PACKAGE_HEAD.$MODULE@1.0
echo "You package is: $PACKAGE"hidl-gen -o $LOC/$MODULE/1.0/defualt -Lc++-impl -r$PACKAGE_HEAD:$LOC \-randroid.hidl:system/libhidl/transport $PACKAGE
hidl-gen -o $LOC/$MODULE/1.0/defualt -Landroidbp-impl -r$PACKAGE_HEAD:$LOC \-randroid.hidl:system/libhidl/transport $PACKAGE
修改Test.cpp Test.h
// FIXME: your file license if you have one#include "Test.h"
#include "log/log.h"
#define LOG_TAG "test1.0"
namespace rockchip::hardware::test::implementation {
sp<::rockchip::hardware::test::V1_0::ITestWarpper> mTestWarpper = nullptr;
hidl_string mDeviceId;
// Methods from ::rockchip::hardware::test::V1_0::ITest follow.
Return<void> Test::setTestWarpper(const sp<::rockchip::hardware::test::V1_0::ITestWarpper>& testWarpper) {// TODO implementmTestWarpper = testWarpper;return Void();
}Test::Test(){ALOGD("@%s.",__FUNCTION__);
}
Test::~Test(){ALOGD("@%s",__FUNCTION__);
}
V1_0::ITest* HIDL_FETCH_ITest(const char* /* name */) {ALOGD("@%s",__FUNCTION__);return new Test();
}
// [&](const ::android::hardware::hidl_string &id
Return<void> Test::GetString(GetString_cb _hidl_cb) {ALOGD("@%s,mDeviceId:%s",__FUNCTION__,mDeviceId.c_str());// TODO implementif(mTestWarpper!=nullptr){mTestWarpper->GetString([&](const ::android::hardware::hidl_string &id){ALOGD("@%s,GetString mDeviceId:%s",__FUNCTION__,id.c_str());_hidl_cb(id);});}return Void();
}// Methods from ::android::hidl::base::V1_0::IBase follow.//ITest* HIDL_FETCH_ITest(const char* /* name */) {//return new Test();
//}
//
} // namespace rockchip::hardware::test::implementation
// FIXME: your file license if you have one#pragma once#include <rockchip/hardware/test/1.0/ITest.h>
#include <hidl/MQDescriptor.h>
#include <hidl/Status.h>namespace rockchip::hardware::test::implementation {using ::android::hardware::hidl_array;
using ::android::hardware::hidl_memory;
using ::android::hardware::hidl_string;
using ::android::hardware::hidl_vec;
using ::android::hardware::Return;
using ::android::hardware::Void;
using ::android::sp;struct Test : public V1_0::ITest {// Methods from ::rockchip::hardware::test::V1_0::ITest follow.Return<void> setTestWarpper(const sp<::rockchip::hardware::test::V1_0::ITestWarpper>& testWarpper) override;Return<void> GetString(GetString_cb _hidl_cb) override;Test();~Test();// Methods from ::android::hidl::base::V1_0::IBase follow.};// FIXME: most likely delete, this is only for passthrough implementations
extern "C" V1_0::ITest* HIDL_FETCH_ITest(const char* name);} // namespace rockchip::hardware::test::implementation
添加一个开机服务 test/1.0/defualt下添加rc 和 xml文件和service.cpp
vendor/rockchip/hardware/interfaces/test/1.0/defualt/rockchip.hardware.test@1.0-service.rc
service test_server /vendor/bin/hw/rockchip.hardware.test@1.0-serviceclass haluser systemgroup systemvendor/rockchip/hardware/interfaces/test/1.0/defualt/rockchip.hardware.test@1.0-service.xml
<manifest version="1.0" type="device"><hal format="hidl"><name>rockchip.hardware.test</name><transport>hwbinder</transport><version>1.0</version><interface><name>ITest</name><instance>default</instance></interface></hal>
</manifest>vendor/rockchip/hardware/interfaces/test/1.0/defualt/service.cpp
#define LOG_TAG "rockchip.hardware.test@1.0-service"#include <rockchip/hardware/test/1.0/ITest.h>
#include <hidl/LegacySupport.h>
#include <binder/ProcessState.h>using rockchip::hardware::test::V1_0::ITest;
using android::hardware::defaultPassthroughServiceImplementation;
using android::hardware::configureRpcThreadpool;
using android::hardware::joinRpcThreadpool;
using android::sp;int main() {android::ProcessState::initWithDriver("/dev/vndbinder");return defaultPassthroughServiceImplementation<ITest>("default", /*maxThreads*/ 6);
}
修改test/1.0/defualt/Android.dp
// FIXME: your file license if you have one
//生成一个AIDL so
cc_library_shared {// FIXME: this should only be -impl for a passthrough hal.// In most cases, to convert this to a binderized implementation, you should:// - change '-impl' to '-service' here and make it a cc_binary instead of a// cc_library_shared.// - add a *.rc file for this module.// - delete HIDL_FETCH_I* functions.// - call configureRpcThreadpool and registerAsService on the instance.// You may also want to append '-impl/-service' with a specific identifier like// '-vendor' or '-<hardware identifier>' etc to distinguish it.name: "rockchip.hardware.test@1.0-impl",relative_install_path: "hw",// FIXME: this should be 'vendor: true' for modules that will eventually be// on AOSP.proprietary: true,srcs: ["Test.cpp","TestWarpper.cpp",],shared_libs: ["libhidlbase","libutils","liblog","rockchip.hardware.test@1.0",],
}
//生成一个服务进程cc_binary {name: "rockchip.hardware.test@1.0-service",init_rc: ["rockchip.hardware.test@1.0-service.rc"],vintf_fragments: ["rockchip.hardware.test@1.0-service.xml"],defaults: ["hidl_defaults"],proprietary: true,relative_install_path: "hw",srcs: ["service.cpp",],cflags: ["-Werror", "-Wno-unused-parameter"],shared_libs: ["liblog","libhardware","libhidlbase","libutils","libcutils","rockchip.hardware.test@1.0","libbinder","libhidlmemory",],
}
device 下根据自己的代码加入编译
PRODUCT_PACKAGES += \ rockchip.hardware.test@1.0-service \rockchip.hardware.test@1.0-impl
到这里我们的AIDL添加结束
二.修改系统te,不然服务跑不起来
添加文件device\rockchip\common\manifests\frameworks\rockchip.hardware.test@1.0-service.xml
<compatibility-matrix version="1.0" type="framework"><hal format="hidl" optional="true"><name>rockchip.hardware.test</name><version>1.0</version><interface><name>ITest</name><instance>default</instance></interface></hal>
</compatibility-matrix>添加文件device\rockchip\common\sepolicy\vendor\hal_test.te
type hal_test_default, domain;
type hal_test_default_exec, exec_type, vendor_file_type, file_type;init_daemon_domain(hal_test_default)allow hal_test_default device:dir { read open};
allow hal_test_default camera_device:chr_file{ read write open ioctl };
allow hal_test_default vndbinder_device:chr_file { read write open ioctl map};
allow hal_test_default hidl_base_hwservice:hwservice_manager add;
allow hal_test_default hal_test_hwservice:hwservice_manager {add find};
allow system_app hal_test_hwservice:hwservice_manager { find };
binder_call(system_app, hal_test_default)
binder_call(hal_test_default, system_app)
#binder_call(hal_test_default, platform_app)
binder_call(hal_test_default, hwservicemanager)
#binder_call(hal_test_default, hal_camera_default)
# Client
#allow cameraserver hal_test_hwservice:hwservice_manager { find };
#allow hal_camera_default hal_test_hwservice:hwservice_manager { find };
get_prop(hal_test_default, hwservicemanager_prop)修改device\rockchip\common\sepolicy\vendor\hwservice_contexts
rockchip.hardware.tv.input::ITvInput u:object_r:hal_tv_input_hwservice:s0
+ rockchip.hardware.test::ITest u:object_r:hal_test_hwservice:s0修改device\rockchip\common\sepolicy\vendor\hwservice.te
+type hal_test_hwservice, hwservice_manager_type;
三。写一个测试demo
添加cpp和h 生成一个so
//添加packages/apps/libtest_bridge/data_bridge.h
// #include <jni.h>
#include <functional>
#include <string>
#include <time.h>void init();
void deinit();//添加packages/apps/libtest_bridge/data_bridge.cpp
#define APK_VERSION "V1.3"
#define LOG_TAG "test_bridge"
//#define LOG_NDEBUG 0#include <utils/Log.h>
#include <jni.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <utils/Log.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/ioctl.h>
#include <sys/mman.h>
#include <linux/videodev2.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <linux/v4l2-subdev.h>#include "data_bridge.h"#include <ui/Fence.h>
#include <ui/GraphicBufferMapper.h>
#include <ui/GraphicBuffer.h>
#include <ui/Rect.h>#include <utils/Singleton.h>#define VIRTUAL_CAMERA
#define RGA_PROC #include <android/hardware_buffer_jni.h>
#include <vndk/hardware_buffer.h>#include <condition_variable>#ifdef RGA_PROC
#include <RockchipRga.h>
#include <im2d_api/im2d.h>
#include "im2d_api/im2d.hpp"
#include "im2d_api/im2d_common.h"
#endif#define LOGE(msg,...) ALOGE("%s(%d): " msg ,__FUNCTION__,__LINE__,##__VA_ARGS__)
#define LOGD(msg,...) ALOGD("%s(%d): " msg ,__FUNCTION__,__LINE__,##__VA_ARGS__)
#define LOGV(msg,...) ALOGV("%s(%d): " msg ,__FUNCTION__,__LINE__,##__VA_ARGS__)using namespace std;
using namespace android;
#include <rockchip/hardware/test/1.0/ITest.h>
class FrameWarpperImp :public rockchip::hardware::test::V1_0::ITestWarpper{public:android::hardware::Return<void> GetString(GetString_cb _hidl_cb) {// TODO implement::android::hardware::hidl_string deviceId = "hellow world";_hidl_cb(deviceId);return android::hardware::Void();}~FrameWarpperImp(){ALOGD("%s",__FUNCTION__);}FrameWarpperImp(){ALOGD("%s",__FUNCTION__);}
};
FrameWarpperImp *frameWarpperImp;
void init()
{::android::hardware::hidl_string testId;ALOGD("%s",__FUNCTION__);android::sp<rockchip::hardware::test::V1_0::ITest> client = rockchip::hardware::test::V1_0::ITest::getService();if(client.get()!= nullptr){ if (frameWarpperImp != nullptr){client->setTestWarpper(nullptr);frameWarpperImp = nullptr;} frameWarpperImp = new FrameWarpperImp();ALOGD("setFrameDecorator");client->setTestWarpper(frameWarpperImp);client->GetString([&](const ::android::hardware::hidl_string &id){testId = id;ALOGD("setFrameDecorator%s",testId.c_str());});}
}
void deinit(){ALOGD("%s",__FUNCTION__);android::sp<rockchip::hardware::test::V1_0::ITest> client = rockchip::hardware::test::V1_0::ITest::getService();ALOGE("client.get():%p",client.get());if(client.get()!= nullptr){ALOGD("setFrameDecorator nullptr");client->setTestWarpper(nullptr);//delete frameWarpperImp ;frameWarpperImp = nullptr;}
}
packages/apps/libtest_bridge/Android.bp
//生成一个libtest_bridge.so
cc_library_shared {name: "libtest_bridge",srcs: ["data_bridge.cpp"],header_libs: ["jni_headers",],include_dirs: ["hardware/rockchip/librga",],shared_libs: ["libbase","libandroid","liblog","libutils","libcutils","libnativehelper","libhidlbase","libui","librga",
//rga_proc "librga","rockchip.hardware.test@1.0",],cflags: ["-Wall","-Wno-error","-Wextra","-Wno-unused-parameter",],sanitize: {scs: true,},
}
ndk_library {name: "libtest_bridge",symbol_file: "libtest_bridge.map.txt",first_version: "30",
}
prebuilt_etc {name: "public.libtest_bridge",src: "public.libtest_bridge.txt",filename_from_src: true,
}//生成一个可执行文件test1.0_service
cc_binary {name: "test1.0_service",proprietary: true,relative_install_path: "hw",srcs: ["service.cpp",],defaults: ["hidl_defaults"],vendor: true,sdk_version: "current",cflags: ["-Wall","-Wno-error","-Wextra","-Wno-unused-parameter",],shared_libs: ["liblog","libtest_bridge",],
}
添加文件packages\apps\libtest_bridge\libtest_bridge.map.txt
LIBDATA_BRIDGE {global:init;local:*;
};添加文件packages\apps\libtest_bridge\public.libtest_bridge.txt
libtest_bridge.so nopreload添加文件packages\apps\libtest_bridge\service.cpp
#define LOG_TAG "rockchip.hardware.test@1.0-service"
#include "data_bridge.h"int main() {init();
}
编译完成后将系统的test1.0_service 推到 data/loal/tmp
执行./test1.0_service
09-05 07:08:06.633 2892 2892 D data_bridge: init
09-05 07:08:06.634 2892 2892 D data_bridge: FrameWarpperImp
09-05 07:08:06.634 2892 2892 D data_bridge: setFrameDecorator
09-05 07:08:06.635 454 454 D test1.0 : @GetString,mDeviceId:
09-05 07:08:06.635 454 454 D test1.0 : @operator(),GetString mDeviceId:
09-05 07:08:06.636 2892 2892 D data_bridge: setFrameDecoratorhellow world
本文章只做自己学习记录
相关文章:

Rk3588 Android12 AIDL 开发
AIDL (Android Interface Definition Language) 和 HIDL (HAL Interface Definition Language) 都是 Android 系统中用于定义接口的工具,但它们有不同的用途和特性。 AIDL (Android Interface Definition Language) 用途: 主要用于应用程序之间的进程间…...
两个长整数字符串求和(不允许使用ES6+)
两个长整数字符串求和(不允许使用ES6), 面试手撸代码遇到到这个问题 1. 实现方式第一种 // 短整数字符串前边补 0; num需要补 0 的短整数字符串, len 长整数字符串的长度 function fillZero (num, len) {let str num.toString();if (str.length < len) {str 0.repeat(…...

11 Java 方法引用、异常处理、Java接口之函数式编程(接口知识补充Function<T,R>、BiFunction<T, U, R>和自定义泛型接口)
文章目录 前言一、Java接口之函数式编程 --- 接口知识补充1 Function<T,R>泛型接口2 BiFunction<T, U, R>泛型接口3 自定义泛型函数式编程接口4 使用lambda表达式、方法引用进行函数式编程二、方法引用1 方法引用初体验(以Array.sort()方法为例)(1)什么是方法引…...
深入探索 Go 语言的编译器与垃圾回收机制
Go 编译器 Go 编译器是通过 go 工具执行的,这个工具的功能不仅仅是生成可执行文件。你可以使用 go tool compile 命令来编译一个 Go 源文件。这个操作将生成一个目标文件,也就是 .o 后缀的文件。以下是在 macOS Mojave 系统上执行的命令和结果展示&…...

2024国赛数学建模-模拟火算法(MATLAB 实现)
模拟退火算法 1.1 算法原理 模拟退火算法的基本思想是从一给定解开始 ,从邻域 中随机产生另一个解 ,接受 Metropolis准则允许目标函数在 有限范围内变坏 ,它由一控制参数 t决定 ,其作用类似于物 理过程中的温度 T,对于控制参数的每一取值 ,算法持续进 行“产生 —判断 —接受…...
YOLOv8 只检测人 只画框不要标签
参考了这个:YOLOv8只检测人(或其他一种或者多种类别)_yolov8只检测指定类别-CSDN博客 1. 只检测人:predict的时候指定参数classes[0] 2. 只画框不要标签:plot的时候传入labelsFalse 3. 标签中去掉置信度:…...

如何将网络安全防范游戏化
组织对威胁的准备和恢复能力跟不上网络犯罪分子的进步。 一些首席执行官仍然认为网络安全需要偶尔干预,而不是持续关注。 但对于许多公司来说,情况并非如此;网络威胁准备需要协调一致的培训工作,因此网络安全团队在攻击发生时已…...

Qt QGraphicsView实现图片放缩、鼠标拖动移动、鼠标点位置放大缩小_图片查看
QtQGraphicsView实现图片放缩、鼠标拖动移动、鼠标点位置放大缩小 头文件: #ifndef TIMGWIDGET_H #define TIMGWIDGET_H#include <QGraphicsItem> #include <QMainWindow> #include <QObject> #include <QWidget>// class TImgWidget : pu…...

Percona Toolkit 神器全攻略(复制类)
Percona Toolkit 神器全攻略(复制类) Percona Toolkit 神器全攻略系列共八篇,前文回顾: 前文回顾Percona Toolkit 神器全攻略Percona Toolkit 神器全攻略(实用类)Percona Toolkit 神器全攻略(配…...
SQLite3 数据类型深入全面讲解
SQLite3,作为一款轻量级的数据库管理系统,在数据存储方面展现出了其独特的魅力。它不仅支持标准的SQL语法,还提供了丰富的数据类型供开发者选择。这些数据类型不仅涵盖了基本的数值和文本类型,还包括了日期时间、二进制数据等复杂…...
Python高效实现Trie(前缀树)及其插入和查找操作
Python高效实现Trie(前缀树)及其插入和查找操作 在Python面试中,考官通常会关注候选人的编程能力、问题解决能力以及对Python语言特性的理解。Trie(前缀树)是一种高效的数据结构,广泛应用于字符串处理、自动补全、拼写检查等场景。本文将详细介绍如何实现一个Trie,并提…...

傅里叶变换家族
禹晶、肖创柏、廖庆敏《数字图像处理(面向新工科的电工电子信息基础课程系列教材)》 禹晶、肖创柏、廖庆敏《数字图像处理》资源二维码...

深度学习——强化学习算法介绍
强化学习算法介绍 强化学习讨论的问题是一个智能体(agent) 怎么在一个复杂不确定的环境(environment)里面去极大化它能获得的奖励。 强化学习和监督学习 强化学习有这个试错探索(trial-and-error exploration),它需要通过探索环境来获取对环境的理解。强化学习 ag…...

轴承知识大全,详细介绍(附3D图纸免费下载)
轴承一般由内圈、外圈、滚动体和保持架组成。对于密封轴承,再加上润滑剂和密封圈(或防尘盖)。这就是轴承的全部组成。 根据轴承使用的工作状况来选用不同类型的轴承,才能更好的发挥轴承的功能,并延长轴承的使用寿命。我…...

【PyTorch】基础环境如何打开
前期安装可以基于这个视频,本文是为了给自己存档如何打开pycharm和jupyter notebookPyTorch深度学习快速入门教程(绝对通俗易懂!)【小土堆】_哔哩哔哩_bilibili Pycharm 配置 新建项目的时候选择解释器pytorch-gpu即可。 Jupyte…...
QT教程:QTime和QTimer的使用场景
QTime类 QTime 是一个用来表示和操作时间的类,它处理一天中的具体时间(例如小时、分钟、秒、毫秒)。通常用于计算时间间隔、记录时间戳、获取当前时间等。 特点和功能 表示时间:QTime 用来表示一天中的某个具体时间(小…...
MySQL 迁移中 explicit_defaults_for_timestamp 参数影响
前言 最近在做数据迁移的时候,使用的是云平台自带的同步工具,在预检查阶段,当时报错 explicit_defaults_for_timestamp 参数在目标端为 off 建议修改 on,有什么风险呢?在此记录下。 测试对比 MySQL 默认情况下 expl…...

树状数组记录
树状数组(Fenwick Tree)是一种用于维护数组前缀和的数据结构,支持高效的单点更新和区间查询操作。它的查询和更新时间复杂度为 O ( log n ) O(\log n) O(logn),适用于需要频繁更新和查询的场景。 树状数组的基本操作 单点更…...

客户端时间和服务器时间的区别
客户端时间: 服务器向客户端拷贝一份前端内容,客户端通过JS获取时间,这样获取的是客户端时间 服务器时间: 服务器通过java代码获取的时间传输给客户端,这样获取的是服务器时间 当有些时候需要使用客户端时间…...

已入职华为!!关于我成功拿下华为大模型算法岗经验总结
方向:大模型算法工程师 整个面试持续了1小时10分钟,能够看出面试官是典型搞技术的,问的很专业又很细,全程感觉压力好大,面完后感觉丝丝凉意,不过幸好还是成功拿下了Offer 一面: 自我介绍 简历项目深度交流 1.项目的背…...
SciencePlots——绘制论文中的图片
文章目录 安装一、风格二、1 资源 安装 # 安装最新版 pip install githttps://github.com/garrettj403/SciencePlots.git# 安装稳定版 pip install SciencePlots一、风格 简单好用的深度学习论文绘图专用工具包–Science Plot 二、 1 资源 论文绘图神器来了:一行…...

Opencv中的addweighted函数
一.addweighted函数作用 addweighted()是OpenCV库中用于图像处理的函数,主要功能是将两个输入图像(尺寸和类型相同)按照指定的权重进行加权叠加(图像融合),并添加一个标量值&#x…...
五年级数学知识边界总结思考-下册
目录 一、背景二、过程1.观察物体小学五年级下册“观察物体”知识点详解:由来、作用与意义**一、知识点核心内容****二、知识点的由来:从生活实践到数学抽象****三、知识的作用:解决实际问题的工具****四、学习的意义:培养核心素养…...

【配置 YOLOX 用于按目录分类的图片数据集】
现在的图标点选越来越多,如何一步解决,采用 YOLOX 目标检测模式则可以轻松解决 要在 YOLOX 中使用按目录分类的图片数据集(每个目录代表一个类别,目录下是该类别的所有图片),你需要进行以下配置步骤&#x…...

分布式增量爬虫实现方案
之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面,避免重复抓取,以节省资源和时间。 在分布式环境下,增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路:将增量判…...

AI,如何重构理解、匹配与决策?
AI 时代,我们如何理解消费? 作者|王彬 封面|Unplash 人们通过信息理解世界。 曾几何时,PC 与移动互联网重塑了人们的购物路径:信息变得唾手可得,商品决策变得高度依赖内容。 但 AI 时代的来…...
rnn判断string中第一次出现a的下标
# coding:utf8 import torch import torch.nn as nn import numpy as np import random import json""" 基于pytorch的网络编写 实现一个RNN网络完成多分类任务 判断字符 a 第一次出现在字符串中的位置 """class TorchModel(nn.Module):def __in…...

2025季度云服务器排行榜
在全球云服务器市场,各厂商的排名和地位并非一成不变,而是由其独特的优势、战略布局和市场适应性共同决定的。以下是根据2025年市场趋势,对主要云服务器厂商在排行榜中占据重要位置的原因和优势进行深度分析: 一、全球“三巨头”…...
MySQL 部分重点知识篇
一、数据库对象 1. 主键 定义 :主键是用于唯一标识表中每一行记录的字段或字段组合。它具有唯一性和非空性特点。 作用 :确保数据的完整性,便于数据的查询和管理。 示例 :在学生信息表中,学号可以作为主键ÿ…...
适应性Java用于现代 API:REST、GraphQL 和事件驱动
在快速发展的软件开发领域,REST、GraphQL 和事件驱动架构等新的 API 标准对于构建可扩展、高效的系统至关重要。Java 在现代 API 方面以其在企业应用中的稳定性而闻名,不断适应这些现代范式的需求。随着不断发展的生态系统,Java 在现代 API 方…...