安卓无障碍脚本开发全教程

news2025/5/26 0:28:28

在这里插入图片描述

文章目录

    • 第一部分:无障碍服务基础
      • 1.1 无障碍服务概述
        • 核心功能:
      • 1.2 基本原理与架构
      • 1.3 开发环境配置
        • 所需工具:
        • 关键依赖:
    • 第二部分:创建基础无障碍服务
      • 2.1 服务声明配置
      • 2.2 服务配置文件
        • 关键属性说明:
      • 2.3 实现服务类
    • 第三部分:高级功能实现
      • 3.1 节点查找与操作
        • 常用查找方法:
        • 节点操作示例:
      • 3.2 手势模拟
      • 3.3 全局事件监听
    • 第四部分:实战案例开发
      • 4.1 自动填写表单
      • 4.2 消息自动回复
      • 4.3 游戏自动化辅助
    • 第五部分:调试与优化
      • 5.1 调试技巧
        • ADB调试命令:
        • 日志记录最佳实践:
      • 5.2 性能优化
        • 优化建议:
        • 优化示例:
    • 第六部分:发布与安全
      • 6.1 权限与隐私
        • 必要权限声明:
        • 隐私注意事项:
      • 6.2 发布流程
    • 第七部分:高级主题
      • 7.1 与其他技术的结合
        • 与Tasker集成:
        • 使用机器学习:
      • 7.2 跨版本兼容性处理
        • 版本差异处理表:
        • 兼容性代码示例:

在这里插入图片描述

第一部分:无障碍服务基础

1.1 无障碍服务概述

安卓无障碍服务(Accessibility Service)是一种特殊类型的服务,旨在帮助残障用户或需要辅助功能的用户更好地使用设备。但它的功能远不止于此,开发者可以利用它实现自动化操作、界面监控和交互等功能。

核心功能:
  • 界面内容访问:获取屏幕上的UI元素信息
  • 自动化操作:模拟点击、滑动等用户操作
  • 事件监控:监听窗口变化、通知、焦点改变等系统事件
  • 增强交互:为特定应用提供定制化辅助功能

1.2 基本原理与架构

用户操作 目标应用 无障碍服务 系统框架 触发界面变化 更新界面状态 发送AccessibilityEvent 处理事件 可选: 执行操作(点击/滑动等) 执行请求的操作 用户操作 目标应用 无障碍服务 系统框架

1.3 开发环境配置

所需工具:
  • Android Studio最新版
  • 安卓设备或模拟器(API 16+)
  • ADB调试工具
关键依赖:
dependencies {
    implementation 'androidx.core:core-ktx:1.7.0'
    implementation 'androidx.appcompat:appcompat:1.4.1'
}

第二部分:创建基础无障碍服务

2.1 服务声明配置

AndroidManifest.xml中添加服务声明:

<service
    android:name=".MyAccessibilityService"
    android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"
    android:exported="true">
    <intent-filter>
        <action android:name="android.accessibilityservice.AccessibilityService" />
    </intent-filter>
    <meta-data
        android:name="android.accessibilityservice"
        android:resource="@xml/service_config" />
</service>

2.2 服务配置文件

创建res/xml/service_config.xml

<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
    android:description="@string/accessibility_service_description"
    android:accessibilityEventTypes="typeAllMask"
    android:accessibilityFlags="flagDefault|flagRetrieveInteractiveWindows"
    android:canRetrieveWindowContent="true"
    android:settingsActivity="com.example.android.accessibility.ServiceSettingsActivity"
    android:canRequestFilterKeyEvents="true"
    android:canPerformGestures="true"
    android:notificationTimeout="100"
    android:packageNames="com.example.targetapp" />
关键属性说明:
  • accessibilityEventTypes:监听的事件类型
  • packageNames:指定监控的应用包名(可选)
  • canPerformGestures:允许执行手势操作(API 24+)

2.3 实现服务类

创建基础服务类MyAccessibilityService.kt

class MyAccessibilityService : AccessibilityService() {

    override fun onServiceConnected() {
        Log.d("A11yService", "无障碍服务已连接")
        // 可以在此处进行服务配置更新
        val info = AccessibilityServiceInfo().apply {
            eventTypes = AccessibilityEvent.TYPE_VIEW_CLICKED or 
                       AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED
            feedbackType = AccessibilityServiceInfo.FEEDBACK_GENERIC
            notificationTimeout = 100
            flags = AccessibilityServiceInfo.DEFAULT or
                   AccessibilityServiceInfo.FLAG_RETRIEVE_INTERACTIVE_WINDOWS
        }
        this.serviceInfo = info
    }

    override fun onAccessibilityEvent(event: AccessibilityEvent?) {
        event ?: return
        when (event.eventType) {
            AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED -> {
                handleWindowChange(event)
            }
            AccessibilityEvent.TYPE_VIEW_CLICKED -> {
                handleViewClick(event)
            }
        }
    }

    override fun onInterrupt() {
        Log.w("A11yService", "无障碍服务被中断")
    }

    private fun handleWindowChange(event: AccessibilityEvent) {
        val rootNode = rootInActiveWindow ?: return
        Log.d("A11yService", "窗口变化: ${event.packageName}")
        // 遍历视图树
        traverseNode(rootNode)
    }

    private fun traverseNode(node: AccessibilityNodeInfo, depth: Int = 0) {
        if (node.childCount == 0) {
            Log.d("A11yTree", "${" ".repeat(depth)}${node.viewIdResourceName}")
            return
        }
        
        for (i in 0 until node.childCount) {
            node.getChild(i)?.let { child ->
                traverseNode(child, depth + 1)
                child.recycle()
            }
        }
    }
}

第三部分:高级功能实现

3.1 节点查找与操作

常用查找方法:
fun findNodes(root: AccessibilityNodeInfo) {
    // 通过文本查找
    val byText = root.findAccessibilityNodeInfosByText("搜索")
    
    // 通过View ID查找(全限定ID)
    val byId = root.findAccessibilityNodeInfosByViewId("com.example.app:id/btnSubmit")
    
    // 通过类名查找
    val editTexts = mutableListOf<AccessibilityNodeInfo>()
    val queue: Queue<AccessibilityNodeInfo> = LinkedList()
    queue.add(root)
    
    while (queue.isNotEmpty()) {
        val current = queue.poll()
        if (current.className == "android.widget.EditText") {
            editTexts.add(current)
        }
        for (i in 0 until current.childCount) {
            current.getChild(i)?.let { queue.add(it) }
        }
    }
}
节点操作示例:
fun performActions(node: AccessibilityNodeInfo) {
    // 点击操作
    if (node.isClickable) {
        node.performAction(AccessibilityNodeInfo.ACTION_CLICK)
    }
    
    // 文本输入
    val arguments = Bundle().apply {
        putCharSequence(AccessibilityNodeInfo.ACTION_ARGUMENT_SET_TEXT_CHARSEQUENCE, "Hello")
    }
    node.performAction(AccessibilityNodeInfo.ACTION_SET_TEXT, arguments)
    
    // 焦点控制
    node.performAction(AccessibilityNodeInfo.ACTION_FOCUS)
    
    // 滚动操作
    node.performAction(AccessibilityNodeInfo.ACTION_SCROLL_FORWARD)
}

3.2 手势模拟

Android支持通过无障碍服务模拟复杂手势:

fun performGesture(service: AccessibilityService) {
    val path = Path().apply {
        moveTo(100f, 100f)  // 起点
        lineTo(500f, 100f)   // 移动到右侧
        lineTo(500f, 500f)   // 向下移动
        lineTo(100f, 500f)   // 向左移动
        close()              // 闭合路径
    }
    
    val gestureBuilder = GestureDescription.Builder()
        .addStroke(GestureDescription.StrokeDescription(
            path, 
            0L,  // 开始时间
            1000L,  // 持续时间(毫秒)
            false  // 是否持续
        ))
    
    service.dispatchGesture(gestureBuilder.build(), object : AccessibilityService.GestureResultCallback() {
        override fun onCompleted(gestureDescription: GestureDescription?) {
            Log.d("Gesture", "手势完成")
        }
        
        override onCancelled(gestureDescription: GestureDescription?) {
            Log.w("Gesture", "手势取消")
        }
    }, null)
}

3.3 全局事件监听

监听系统级事件:

override fun onAccessibilityEvent(event: AccessibilityEvent) {
    when (event.eventType) {
        AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED -> {
            val notificationText = event.text.joinToString()
            Log.d("Notification", "新通知: $notificationText")
        }
        AccessibilityEvent.TYPE_ANNOUNCEMENT -> {
            Log.d("Announcement", "系统公告: ${event.text}")
        }
        AccessibilityEvent.TYPE_TOUCH_EXPLORATION_GESTURE_START -> {
            Log.d("Touch", "触摸探索开始")
        }
    }
}

第四部分:实战案例开发

4.1 自动填写表单

class FormFillerService : AccessibilityService() {

    private val formData = mapOf(
        "username" to "testuser",
        "password" to "secure123",
        "email" to "test@example.com"
    )

    override fun onAccessibilityEvent(event: AccessibilityEvent) {
        if (event.eventType != AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED) return
        
        val rootNode = rootInActiveWindow ?: return
        
        formData.forEach { (fieldName, value) ->
            val nodes = rootNode.findAccessibilityNodeInfosByViewId("com.example.app:id/$fieldName")
            nodes.firstOrNull()?.let { field ->
                if (field.className == "android.widget.EditText") {
                    val args = Bundle().apply {
                        putCharSequence(AccessibilityNodeInfo.ACTION_ARGUMENT_SET_TEXT_CHARSEQUENCE, value)
                    }
                    field.performAction(AccessibilityNodeInfo.ACTION_SET_TEXT, args)
                }
            }
        }
        
        // 自动提交表单
        rootNode.findAccessibilityNodeInfosByViewId("com.example.app:id/submit")
            .firstOrNull()
            ?.performAction(AccessibilityNodeInfo.ACTION_CLICK)
    }
}

4.2 消息自动回复

class AutoReplyService : AccessibilityService() {

    private val replyMessages = listOf(
        "我正在开会,稍后回复您",
        "好的,收到",
        "谢谢通知"
    )

    override fun onAccessibilityEvent(event: AccessibilityEvent) {
        if (event.packageName != "com.whatsapp") return
        
        when (event.eventType) {
            AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED -> {
                // 处理通知事件
                val messages = event.text.filter { it.contains("发来消息") }
                if (messages.isNotEmpty()) {
                    replyToLatestMessage()
                }
            }
            AccessibilityEvent.TYPE_VIEW_TEXT_CHANGED -> {
                // 处理界面文本变化
                if (isChatOpen()) {
                    autoReplyInChat()
                }
            }
        }
    }
    
    private fun replyToLatestMessage() {
        // 实现打开聊天界面并回复的逻辑
    }
    
    private fun isChatOpen(): Boolean {
        // 检测当前是否在聊天界面
    }
    
    private fun autoReplyInChat() {
        val root = rootInActiveWindow ?: return
        val messageNodes = root.findAccessibilityNodeInfosByViewId("com.whatsapp:id/message_text")
        
        // 获取最后一条消息
        val lastMessage = messageNodes.lastOrNull()?.text ?: return
        
        // 随机选择回复内容
        val randomReply = replyMessages.random()
        
        // 找到输入框并发送
        root.findAccessibilityNodeInfosByViewId("com.whatsapp:id/entry")
            .firstOrNull()
            ?.let { input ->
                val args = Bundle().apply {
                    putCharSequence(AccessibilityNodeInfo.ACTION_ARGUMENT_SET_TEXT_CHARSEQUENCE, randomReply)
                }
                input.performAction(AccessibilityNodeInfo.ACTION_SET_TEXT, args)
                
                // 发送消息
                root.findAccessibilityNodeInfosByViewId("com.whatsapp:id/send")
                    .firstOrNull()
                    ?.performAction(AccessibilityNodeInfo.ACTION_CLICK)
            }
    }
}

4.3 游戏自动化辅助

class GameHelperService : AccessibilityService() {

    private var isRunning = false
    private val handler = Handler(Looper.getMainLooper())
    private val clickRunnable = object : Runnable {
        override fun run() {
            performAutoClick()
            if (isRunning) {
                handler.postDelayed(this, 1000) // 每秒点击一次
            }
        }
    }

    override fun onServiceConnected() {
        val info = AccessibilityServiceInfo().apply {
            eventTypes = AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED
            feedbackType = AccessibilityServiceInfo.FEEDBACK_GENERIC
            flags = AccessibilityServiceInfo.FLAG_REPORT_VIEW_IDS
        }
        serviceInfo = info
    }

    override fun onAccessibilityEvent(event: AccessibilityEvent) {
        if (event.packageName != "com.game.package") return
        
        when (event.eventType) {
            AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED -> {
                checkGameState()
            }
        }
    }
    
    private fun checkGameState() {
        val root = rootInActiveWindow ?: return
        val battleNode = root.findAccessibilityNodeInfosByViewId("com.game.package:id/battle_indicator")
        
        if (battleNode.isNotEmpty()) {
            startAutoClicking()
        } else {
            stopAutoClicking()
        }
    }
    
    private fun startAutoClicking() {
        if (!isRunning) {
            isRunning = true
            handler.post(clickRunnable)
        }
    }
    
    private fun stopAutoClicking() {
        isRunning = false
        handler.removeCallbacks(clickRunnable)
    }
    
    private fun performAutoClick() {
        val root = rootInActiveWindow ?: return
        val attackBtn = root.findAccessibilityNodeInfosByViewId("com.game.package:id/attack_button")
            .firstOrNull()
        
        attackBtn?.performAction(AccessibilityNodeInfo.ACTION_CLICK)
        
        // 随机位置点击,避免被检测为机器人
        if (Math.random() < 0.3) {
            val randomX = (100..500).random()
            val randomY = (200..800).random()
            dispatchGesture(createClickGesture(randomX, randomY), null, null)
        }
    }
    
    private fun createClickGesture(x: Int, y: Int): GestureDescription {
        val clickPath = Path().apply {
            moveTo(x.toFloat(), y.toFloat())
        }
        
        return GestureDescription.Builder()
            .addStroke(GestureDescription.StrokeDescription(
                clickPath, 0, 50))
            .build()
    }
}

第五部分:调试与优化

5.1 调试技巧

ADB调试命令:
# 查看已启用的无障碍服务
adb shell settings get secure enabled_accessibility_services

# 启用服务
adb shell settings put secure enabled_accessibility_services com.example.pkg/.MyAccessibilityService

# 查看无障碍事件日志
adb shell logcat -s AccessibilityEvent
日志记录最佳实践:
fun logNodeInfo(node: AccessibilityNodeInfo) {
    val sb = StringBuilder().apply {
        append("View ID: ${node.viewIdResourceName}\n")
        append("Text: ${node.text}\n")
        append("Class: ${node.className}\n")
        append("Bounds: ${node.boundsInScreen}\n")
        append("Actions: ${node.actionList.joinToString()}\n")
        append("ChildCount: ${node.childCount}\n")
    }
    Log.d("NodeInfo", sb.toString())
}

5.2 性能优化

优化建议:
  1. 减少遍历深度:只查找必要的节点层级
  2. 及时回收节点:调用recycle()释放资源
  3. 事件过滤:只监听必要的事件类型
  4. 延迟处理:对频繁事件使用防抖
  5. 后台处理:将耗时操作移到工作线程
优化示例:
class OptimizedService : AccessibilityService() {

    private val eventQueue = LinkedBlockingQueue<AccessibilityEvent>()
    private val workerThread = HandlerThread("EventProcessor").apply { start() }
    private val workerHandler = Handler(workerThread.looper)
    
    private val eventProcessor = object : Runnable {
        override fun run() {
            while (true) {
                val event = eventQueue.take()
                processEvent(event)
            }
        }
    }
    
    override fun onCreate() {
        super.onCreate()
        workerHandler.post(eventProcessor)
    }
    
    override fun onAccessibilityEvent(event: AccessibilityEvent) {
        // 快速将事件加入队列,避免阻塞主线程
        eventQueue.put(event)
    }
    
    private fun processEvent(event: AccessibilityEvent) {
        // 实际处理逻辑
    }
    
    override fun onDestroy() {
        workerThread.quitSafely()
        super.onDestroy()
    }
}

第六部分:发布与安全

6.1 权限与隐私

必要权限声明:
<uses-permission android:name="android.permission.BIND_ACCESSIBILITY_SERVICE" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
隐私注意事项:
  1. 明确告知用户:在隐私政策中说明数据收集范围
  2. 最小权限原则:只请求必要的权限
  3. 敏感数据处理:避免收集密码等敏感信息
  4. 数据加密:对存储的日志和数据进行加密

6.2 发布流程

  1. 测试阶段

    • 在不同安卓版本上测试
    • 在各种品牌设备上测试(特别是国产ROM)
    • 测试电池消耗情况
  2. 应用商店要求

    • 明确说明是无障碍辅助工具
    • 提供详细的使用说明视频
    • 如果是自动化工具,需遵守各商店政策
  3. 持续更新

    • 定期适配新安卓版本
    • 针对流行应用的特殊适配
    • 根据用户反馈优化功能

第七部分:高级主题

7.1 与其他技术的结合

与Tasker集成:
// 接收Tasker的广播意图
private val taskerReceiver = object : BroadcastReceiver() {
    override fun onReceive(context: Context, intent: Intent) {
        if (intent.action == "net.dinglisch.android.tasker.ACTION_TRIGGER") {
            val task = intent.getStringExtra("task")
            when (task) {
                "start_automation" -> startAutomation()
                "stop_automation" -> stopAutomation()
            }
        }
    }
}

override fun onCreate() {
    super.onCreate()
    registerReceiver(taskerReceiver, IntentFilter("net.dinglisch.android.tasker.ACTION_TRIGGER"))
}
使用机器学习:
// 使用ML Kit识别屏幕内容
fun detectTextFromScreen(bitmap: Bitmap): String {
    val recognizer = TextRecognition.getClient(TextRecognizerOptions.DEFAULT_OPTIONS)
    val image = InputImage.fromBitmap(bitmap, 0)
    
    return try {
        val result = recognizer.process(image).await()
        result.text
    } catch (e: Exception) {
        Log.e("ML", "识别失败", e)
        ""
    }
}

// 截图并处理
fun captureAndAnalyze() {
    val projection = MediaProjectionManager.createScreenCaptureIntent()
    // 需要先获取用户授权...
    
    val imageReader = ImageReader.newInstance(
        screenWidth, screenHeight, 
        PixelFormat.RGBA_8888, 2
    )
    
    imageReader.setOnImageAvailableListener({ reader ->
        val image = reader.acquireLatestImage()
        // 转换为Bitmap并传递给识别器
        val text = detectTextFromScreen(convertImageToBitmap(image))
        Log.d("ScreenText", "识别结果: $text")
        image.close()
    }, handler)
}

7.2 跨版本兼容性处理

版本差异处理表:
功能API 16-22API 23-28API 29+
节点信息获取基本支持增强支持受限
手势模拟不支持部分支持完全支持
隐私限制部分严格
后台服务允许限制严格限制
兼容性代码示例:
fun performActionCompat(node: AccessibilityNodeInfo, action: Int, args: Bundle? = null): Boolean {
    return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        node.performAction(action, args)
    } else {
        node.performAction(action)
    }
}

fun getNodeTextCompat(node: AccessibilityNodeInfo): String {
    return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        node.text?.toString() ?: ""
    } else {
        node.text ?: ""
    }
}

在这里插入图片描述

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

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

相关文章

SOC-ESP32S3部分:10-GPIO中断按键中断实现

飞书文档https://x509p6c8to.feishu.cn/wiki/W4Wlw45P2izk5PkfXEaceMAunKg 学习了GPIO输入和输出功能后&#xff0c;参考示例工程&#xff0c;我们再来看看GPIO中断&#xff0c;IO中断的配置分为三步 配置中断触发类型安装中断服务注册中断回调函数 ESP32-S3的所有通用GPIO…

战略-2.1 -战略分析(PEST/五力模型/成功关键因素)

战略分析路径&#xff0c;先宏观&#xff08;PEST&#xff09;、再产业&#xff08;产品生命周期、五力模型、成功关键因素&#xff09;、再竞争对手分析、最后企业内部分析。 本文介绍&#xff1a;PEST、产品生命周期、五力模型、成功关键因素、产业内的战略群组 一、宏观环境…

python第三方库安装错位

问题所在 今天在安装我的django库时&#xff0c;我的库安装到了python3.13版本。我本意是想安装到python3.11版本的。我的pycharm右下角也设置了python3.11 但是太可恶了&#xff0c;我在pycharm的项目终端执行安装命令的时候还是给我安装到了python3.13的位置。 解决方法 我…

如何把vue项目部署在nginx上

1&#xff1a;在vscode中把vue项目打包会出现dist文件夹 按照图示内容即可把vue项目部署在nginx上

Vue3集成Element Plus完整指南:从安装到主题定制下-实现后台管理系统框架搭建

本文将详细介绍如何使用 Vue 3 构建一个综合管理系统&#xff0c;包括路由配置、页面布局以及常用组件集成。 一、路由配置 首先&#xff0c;我们来看系统的路由配置&#xff0c;这是整个应用的基础架构&#xff1a; import {createRouter, createWebHistory} from vue-rout…

SpringBoot项目配置文件、yml配置文件

一. 配置文件格式 1. SpringBoot项目提供了多种属性配置方式(properties、yaml、yml)。 二. yml配置文件 1. 格式&#xff1a; (1) 数值前边必须有空格&#xff0c;作为分隔符。 (2) 使用缩进表示层级关系&#xff0c;缩进时&#xff0c;不允许使用Tab键&#xff0c;只能使用空…

windows11 安装 jupyter lab

1、安装python环境 略 2、安装jupyterlab pip install jupyterlab 3、将jupyterlab的目录配置到path pip show jupyterlab 看到location的值&#xff0c;那么 jupyterlab就安装在与之同级的Scripts下&#xff0c;将Scripts目录设置在Path即可。

【算法】:动态规划--背包问题

背包问题 引言 什么是背包问题&#xff1f; 背包问题就是一个有限的背包&#xff0c;给出一定的物品&#xff0c;如何合理的装入物品使得背包中的物品的价值最大&#xff1f; 01背包 01背包&#xff0c;顾名思义就是每一种给定的物品要么选择&#xff0c;要么不选&#xff…

Nginx核心功能

目录 前言一. 正向代理1.配置正向代理&#xff08;1&#xff09;添加正向代理&#xff08;2&#xff09;验证正向代理 二. 反向代理1.配置nginx七层代理&#xff08;1&#xff09;环境安装&#xff08;2&#xff09;配置nginx七层代理转发&#xff08;3&#xff09;测试 2. 配置…

upload-labs通关笔记-第15关 文件上传之图片马getimagesize绕过

系列目录 upload-labs通关笔记-第1关 文件上传之前端绕过&#xff08;3种渗透方法&#xff09; upload-labs通关笔记-第2关 文件上传之MIME绕过-CSDN博客 upload-labs通关笔记-第3关 文件上传之黑名单绕过-CSDN博客 upload-labs通关笔记-第4关 文件上传之.htacess绕过-CSDN…

【游戏设计】游戏玩法与游戏机制

在游戏设计中&#xff0c;“玩法”&#xff08;Gameplay&#xff09;和“机制”&#xff08;Game Mechanic&#xff09;是两个频繁出现但容易混淆的概念。许多新手开发者、设计师甚至玩家常常将两者混为一谈。本文将通过定义、对比和案例解析的方式&#xff0c;清晰地阐明二者的…

Spring的资源Resource和ResourceLoader

两者区别和联系 Resource 和ResourceLoader 都是 Spring 框架中用于资源访问的接口 Resource 是“资源本身”&#xff0c;ResourceLoader 是“资源工厂/加载器”&#xff0c;负责创建 Resource。 ​ Resource:Spring 统一抽象的“资源”对象,可以表示文件、类路径下的文件、U…

【AI实战】从“苦AI”到“爽AI”:Magentic-UI 把“人类-多智能体协作”玩明白了!

Hello&#xff0c;亲爱的小伙伴们&#xff01;你是否曾经在深夜里&#xff0c;为了自动化点外卖、筛机票、抓网页数据焦头烂额&#xff1f;有没有幻想过哪天能出个“贴心AI管家”&#xff0c;一键点菜、搞定事务、自动操作网页&#xff0c;比你还懂你&#xff1f;更关键——还让…

Linux之 SPI 驱动框架- spi-mem 框架

一、框架变更的历程 1.1 旧框架图 1.2 新框架图 那么问题来了&#xff0c; 为什么要开发新的 SPI 存储器接口&#xff1f; 有了这个新的框架&#xff0c; SPI NOR 和SPI NAND 都可以基于相同的SPI控制器驱动进行支持了。m25p80 驱动将被修改成&#xff0c;使用spi-mem 接口&a…

振动分析 - 献个宝

1.一个自制的振动能量分析工具 这个分析工具似乎真的定位到了故障的具体位置。 1.1对一组实验室虚拟信号的分析结果: 1.2 对现场真实数据的分析结果 依照边频带的调制,和边频的缝隙宽度,基本定位到问题。 追加几份待看的文档: 齿轮结构的频谱特征 - 知乎使用 FFT 获得…

【论文阅读】——D^3-Human: Dynamic Disentangled Digital Human from Monocular Vi

文章目录 摘要1 引言2 相关工作3 方法3.1 HmSDF 表示3.2 区域聚合3.3. 变形场3.4. 遮挡感知可微分渲染3.5 训练3.5.1 训练策略3.5.2 重建损失3.5.3 正则化限制 4. 实验4.1 定量评估4.2 定性评价4.3 消融研究4.4 应用程序 5 结论 摘要 我们介绍 D 3 D^{3} D3人&#xff0c;一种…

高分辨率北半球多年冻土数据集(2000-2016)

关键数据集分类&#xff1a;冰冻圈数据集时间分辨率&#xff1a;10 year < x < 100 year空间分辨率&#xff1a;1km - 10km共享方式&#xff1a;开放获取数据大小&#xff1a;339.79 MB数据时间范围&#xff1a;2000-01-01 — 2016-12-31元数据更新时间&#xff1a;2022-…

基于开源AI智能名片链动2+1模式S2B2C商城小程序的管理与运营策略研究

摘要&#xff1a;本文通过分析开源AI智能名片链动21模式S2B2C商城小程序的技术架构与商业逻辑&#xff0c;探讨其在企业管理与运营中的实践价值。结合案例研究&#xff0c;论证该模式如何通过清晰的目标设定、动态反馈机制和资源整合能力&#xff0c;提升团队执行力与客户粘性。…

储能电站:风光储一体化能源中心数字孪生

在 “双碳” 目标引领下&#xff0c;我国能源产业加速向清洁低碳、绿色化转型&#xff0c;风能、太阳能等可再生能源的开发利用成为关键。然而&#xff0c;风能和太阳能的波动性、间歇性与随机性&#xff0c;给大规模接入电网带来挑战。储能技术的兴起&#xff0c;为解决这一难…

9. 现代循环神经网络

文章目录 9.1. 门控循环单元&#xff08;GRU&#xff09;9.1.1. 门控隐状态9.1.1.1. 重置门和更新门9.1.1.2. 候选隐状态9.1.1.3. 隐状态 9.1.2. 从零开始实现9.1.2.1. 初始化模型参数9.1.2.2. 定义模型 9.1.3. 简洁实现9.1.4. 小结 9.2. 长短期记忆网络&#xff08;LSTM&#…