Android Build Variants(构建变体)详解
Android Build Variants(构建变体)是 Android 开发中用于生成不同版本应用程序的一种机制。它允许开发者根据不同的需求,如不同的应用市场、不同的功能模块、不同的环境配置等,从同一个代码库中生成多个不同的 APK。
组成部分
- Build Types: 构建类型主要用于区分不同的构建环境,如开发环境、测试环境和生产环境等。常见的构建类型有 debug 和 release。
- debug: 主要用于开发过程中,它通常包含更多的调试信息,并且不进行代码混淆和压缩等优化操作,方便开发者进行调试。
- release: 用于发布到应用商店供用户下载安装的版本,会进行代码混淆、压缩等优化操作,以减小 APK 体积和提高应用的安全性和性能。
- Product Flavors: 产品风味可以用来区分不同的应用变体,比如针对不同的应用市场、不同的客户群体或不同的功能特性等。例如,一个应用可能有免费版和付费版两种产品风味,它们在功能上有所不同;或者针对不同的应用商店(如 Google Play 和华为应用市场)有不同的产品风味,可能会在一些配置或资源上有所差异。
作用
-
定制化应用版本: 通过不同的 Build Variants 组合,可以轻松地为不同的场景和需求生成定制化的应用版本。例如,为某个特定客户定制具有特殊功能的应用版本,或者为不同的应用市场生成包含不同渠道信息的版本。
-
优化应用性能和大小: 对于不同的构建变体,可以根据其特点进行针对性的优化。比如,在 release 版本中进行代码混淆和资源压缩,去除不必要的代码和资源,以减小 APK 的大小,提高应用的加载速度和运行性能。
-
方便开发和测试: 在开发过程中,使用 debug 构建类型可以方便地进行调试,快速定位和解决问题。同时,通过不同的 Product Flavors 可以模拟不同的应用场景和用户需求,便于进行全面的测试。
使用示例
在 Android Studio 中配置 Build Variants 主要涉及在 build.gradle 文件里定义构建类型(Build Types)和产品风味(Product Flavors),以下为你详细介绍配置步骤:
1. 定义构建类型
构建类型通常用于区分不同的构建环境,像开发环境和发布环境等。默认情况下,Android 项目有 debug 和 release 两种构建类型。你也能自定义新的构建类型。
在项目的 app/build.gradle 文件里添加或修改 buildTypes 部分,示例如下:
android {buildTypes {// 调试版本配置debug {// 可调试debuggable true// 开启 JNI 调试jniDebugBuild true}// 发布版本配置release {// 开启代码混淆minifyEnabled true// 指定混淆规则文件proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'}// 自定义构建类型staging {initWith debug// 应用 ID 后缀applicationIdSuffix ".staging"// 版本名后缀versionNameSuffix "-staging"}}
}
2. 定义产品风味
产品风味可用来区分不同的应用变体,例如免费版和付费版,或者针对不同应用商店的版本。
在 app/build.gradle 文件中添加 productFlavors 部分,示例如下:
android {productFlavors {// 免费版风味free {// 应用 IDapplicationId "com.example.app.free"// 资源字符串值resValue "string", "app_name", "Free App"}// 付费版风味paid {applicationId "com.example.app.paid"resValue "string", "app_name", "Paid App"}}
}
3. 组合构建变体
构建类型和产品风味会组合形成不同的构建变体。例如,在上述配置中,会产生 freeDebug、freeRelease、freeStaging、paidDebug、paidRelease 和 paidStaging 这些构建变体。
4. 选择构建变体
在 Android Studio 里,你可以通过以下方式选择要使用的构建变体:
- 打开 Android Studio 底部的 Build Variants 面板。
- 在 Build Variants 面板中,为每个模块选择所需的构建变体。
完整配置示例
下面是一个完整的 app/build.gradle 文件示例,其中包含了构建类型和产品风味的配置:
apply plugin: 'com.android.application'android {compileSdkVersion 33buildToolsVersion "33.0.0"defaultConfig {applicationId "com.example.app"minSdkVersion 21targetSdkVersion 33versionCode 1versionName "1.0"testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"}buildTypes {debug {debuggable truejniDebugBuild true}release {minifyEnabled trueproguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'}staging {initWith debugapplicationIdSuffix ".staging"versionNameSuffix "-staging"}}productFlavors {free {applicationId "com.example.app.free"resValue "string", "app_name", "Free App"}paid {applicationId "com.example.app.paid"resValue "string", "app_name", "Paid App"}}
}dependencies {implementation fileTree(dir: 'libs', include: ['*.jar'])implementation 'androidx.appcompat:appcompat:1.4.1'implementation 'androidx.constraintlayout:constraintlayout:2.1.3'testImplementation 'junit:junit:4.13.2'androidTestImplementation 'androidx.test.ext:junit:1.1.3'androidTestImplementation 'androidx.test.espresso:espresso-core:3.4.0'
}
按照上述步骤操作,你就能在 Android Studio 中完成 Build Variants 的配置,进而生成不同的应用版本。
不同的构建变体资源加载
在 Android 应用中,可根据不同的构建变体加载不同资源,以下是具体实现方式:
1. 为不同构建变体创建资源目录
在 src 目录下,为每种构建变体(产品风味和构建类型的组合)创建对应的资源目录。目录命名遵循 src/<flavorName><buildTypeName> 的格式。
例如,若你有 free 和 paid 两种产品风味,以及 debug 和 release 两种构建类型,可创建如下目录结构:
app/
├── src/
│ ├── freeDebug/
│ │ └── res/
│ │ ├── drawable/
│ │ ├── layout/
│ │ └── values/
│ ├── freeRelease/
│ │ └── res/
│ ├── paidDebug/
│ │ └── res/
│ └── paidRelease/
│ └── res/
│ ├── main/
│ │ └── res/
- main 目录中的资源是所有构建变体共享的。
- 特定构建变体目录中的资源会覆盖 main 目录中同名的资源。
2. 在不同资源目录中放置不同资源
在上述创建的目录中,为不同构建变体放置不同的资源。例如,在 freeDebug/res/values/strings.xml 中可以定义免费调试版的字符串资源:
<resources><string name="app_name">Free App (Debug)</string>
</resources>
而在 paidRelease/res/values/strings.xml 中定义付费发布版的字符串资源:
<resources><string name="app_name">Paid App (Release)</string>
</resources>
3. 在代码中加载资源
在代码里加载资源时,Android 系统会自动根据当前的构建变体选择合适的资源。例如,在 Activity 中获取应用名称:
import android.os.Bundle
import androidx.appcompat.app.AppCompatActivityclass MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)// 获取应用名称val appName = getString(R.string.app_name)// 打印应用名称println("App Name: $appName")}
}
4. 注意事项
- 资源优先级:资源加载时,特定构建变体的资源优先级最高,然后是产品风味的资源,最后是 main 目录中的资源。例如,freeDebug 目录中的资源优先级高于 free 目录中的资源,而 free 目录中的资源优先级高于 main 目录中的资源。
- 资源命名:确保不同构建变体的资源命名一致,这样才能正确覆盖。
通过以上步骤,你就能在 Android 应用中依据不同的构建变体加载不同的资源。
不同的构建变体代码逻辑区分
Android 应用里,你可以依据不同的构建变体加载不同的代码逻辑。以下是几种常见的实现方式:
1. 使用条件编译
借助 Gradle 的构建配置,在代码里通过条件判断来依据不同的构建变体执行不同逻辑。
步骤
在 build.gradle 文件中定义构建类型或产品风味的标志:
android {buildTypes {debug {buildConfigField "boolean", "IS_DEBUG", "true"}release {buildConfigField "boolean", "IS_DEBUG", "false"}}productFlavors {free {buildConfigField "boolean", "IS_FREE_VERSION", "true"}paid {buildConfigField "boolean", "IS_FREE_VERSION", "false"}}
}
上述代码中,在 debug 和 release 构建类型里定义了 IS_DEBUG 标志,在 free 和 paid 产品风味中定义了 IS_FREE_VERSION 标志。
在代码中使用这些标志进行条件判断
import android.os.Bundle
import androidx.appcompat.app.AppCompatActivityclass MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)if (BuildConfig.IS_DEBUG) {// 调试版本的代码逻辑println("Debug version logic")} else {// 发布版本的代码逻辑println("Release version logic")}if (BuildConfig.IS_FREE_VERSION) {// 免费版的代码逻辑println("Free version logic")} else {// 付费版的代码逻辑println("Paid version logic")}}
}
2. 为不同构建变体创建不同的类文件
可以为不同的构建变体创建对应的类文件,让 Android 系统在运行时根据当前构建变体自动加载合适的类。
步骤
- 创建不同构建变体的目录结构:
app/
├── src/
│ ├── free/
│ │ └── java/
│ │ └── com/
│ │ └── example/
│ │ └── app/
│ │ └── Feature.java
│ ├── paid/
│ │ └── java/
│ │ └── com/
│ │ └── example/
│ │ └── app/
│ │ └── Feature.java
│ └── main/
│ └── java/
│ └── com/
│ └── example/
│ └── app/
│ └── MainActivity.java
- 在不同的 Feature.java 文件中实现不同的逻辑:
free/java/com/example/app/Feature.kt:
package com.example.appclass Feature {fun doSomething() {println("Free version feature")}
}
paid/java/com/example/app/Feature.kt:
package com.example.appclass Feature {fun doSomething() {println("Paid version feature")}
}
在 MainActivity 中使用 Feature 类:
import android.os.Bundle
import androidx.appcompat.app.AppCompatActivityclass MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)val feature = Feature()feature.doSomething()}
}
通过以上两种方法,你就能在 Android 应用中依据不同的构建变体加载不同的代码逻辑。
总结
Android 构建变体(Build Variants)是强大且灵活的机制,由构建类型(如 debug、release)和产品风味(如免费版、付费版)组合而成。借助它,开发者能从同一代码库生成多个定制化的 APK 版本。在配置方面,可在 build.gradle 文件中定义构建类型和产品风味,同时能设置不同的属性和资源。通过构建变体,不仅能针对不同应用市场、客户群体、功能特性等定制应用,还能优化应用性能和大小,方便开发和测试。在实际应用里,能依据不同的构建变体加载不同资源和代码逻辑,甚至结合设备类型等其他条件,进一步实现复杂的定制化需求。
相关文章:
Android Build Variants(构建变体)详解
Android Build Variants(构建变体)是 Android 开发中用于生成不同版本应用程序的一种机制。它允许开发者根据不同的需求,如不同的应用市场、不同的功能模块、不同的环境配置等,从同一个代码库中生成多个不同的 APK。 组成部分 B…...
看一看 中间件Middleware
中间件(Middleware)是介于操作系统与应用程序之间的一层软件 它为分布式应用程序提供了统一的通信、数据交换、服务调用、消息传递等能力。 它的作用就像“胶水”,连接各个系统组件。 文章目录 消息队列缓存中间件数据库中间件服务注册与发现…...
ai人才需要掌握什么
在人工智能(AI)技术重塑全球产业格局的今天,AI人才的核心竞争力已超越单一技术维度,演变为“技术深度+人文广度+伦理自觉”的三维能力模型。本文将从技术能力体系、跨学科思维、伦理与治理三个层面,解析AI人才的核心能力框架,并针对技术局限性提出系统性应对策略。 一、…...
RISC-V低功耗MCU动态时钟门控技术详解
我来分享一下RISC-V核低功耗MCU的动态时钟门控技术实现: 这款MCU通过硬件级时钟门控电路实现了模块级的功耗管理。当外设(如UART、SPI)处于闲置状态时,系统会自动切断其时钟信号,减少无效翻转功耗。同时支持多电压域协…...
第十天 Shader编程:编写简单表面着色器 Addressable资源管理系统 DOTS(面向数据技术栈)入门
前言 作为Unity初学者,在实现复杂场景时经常会遇到性能瓶颈。本文将带你通过四个关键技术的实战学习,掌握现代Unity开发的核心优化方案: Shader编程 - 编写表面着色器控制物体渲染Addressable系统 - 实现高效资源管理DOTS技术栈 - 解锁百万…...
把dll模块注入到游戏进程的方法_插APC注入
一、概述 APC是异步过程调用,系统创建线程的时候会为线程创建一个APC队列,当线程调用SleepEx,WaitSingleObjectEx等函数时,并把线程状态被设置为可提醒状态时,线程并不会睡眠,而是检查APC队列是否为空,如果不为空,转去执行APC队列中的每一项,因此给目标进程中的线程插…...
【C++】内存管理:内存划分、动态内存管理(new、delete用法)
文章目录 一、C/C中的内存划分二、C语言中动态内存管理方式三、C中动态内存管理方式1、new、delete基本用法(1)、内置类型(2)、自定义类型 2、operator new与operator delete函数3、new和delete的实现原理(1)内置类型(2)自定义类型…...
【k8s】KubeProxy 的三种工作模式——Userspace、iptables 、 IPVS
在 Kubernetes 集群中,KubeProxy 是实现 Service 的网络代理和负载均衡功能的关键组件。它支持三种工作模式:Userspace 模式、iptables 模式和 IPVS 模式。每种模式都有其特点和适用场景,以下是详细介绍: Userspace 模式 工作原…...
密码学的hash函数,哈希碰撞, collision resistance, BTC用到的SHA-256简介
密码学中的哈希函数、哈希碰撞、抗碰撞性(collision resistance)以及比特币中使用的 SHA-256 的简明介绍: 🧩 一、哈希函数(Hash Function) 定义: 哈希函数是一种将任意长度的输入(…...
通过监督微调(SFT)提升AI Agent效果的完整指南
一、SFT技术深度剖析 1.1 核心概念 监督微调(Supervised Fine-Tuning)是在大规模预训练语言模型(如LLaMA、GPT系列)的基础上,使用特定任务标注数据进行二次训练的过程。其本质是通过有监督学习调整模型参数ÿ…...
TCP基础题:音乐播放列表管理系统
需求描述 服务器端 创建一个 TCP 服务器,监听本地的 9999 端口,支持多个客户端连接。维护一个音乐播放列表,每个音乐条目包含歌曲名称、歌手、时长等信息。能够处理客户端的以下请求: 添加音乐到播放列表:接收客户端发…...
碰一碰发视频源码文案功能,支持OEM
在数字化营销与内容传播领域,碰一碰发视频文案功能凭借 NFC 技术的便捷性,实现了视频与文案的快速传递。本文将围绕该功能的开发流程,从需求分析、技术架构设计到核心功能实现,详细解析其开发要点,为开发者提供实践参考…...
如何在 Vue 3 中实现一个自定义的 `v-html` 组件
引言 在 Vue.js 中,v-html 是一个非常有用的指令,用于将 HTML 字符串渲染到元素中。然而,由于它直接插入 HTML 内容,存在潜在的安全风险(如 XSS 攻击)。因此,了解其工作原理并实现一个安全的替…...
差分对的返回电流-信号完整性分析
差分对的返回电流: 单端线的返回电流集中在参考平面,差分对的返回电流是怎样分布的?有一种观点认为,对于差分对这种耦合传输结构,其中一条走线是另一条走线的返回路径。甚至更进一步得出差分对不需要参考平面的结论。是否真的如此?要想了解…...
html单页业务介绍源码
源码介绍 html单页业务介绍源码,源码由HTMLCSSJS组成,记事本打开源码文件可以进行内容文字之类的修改,双击html文件可以本地运行 效果预览 源码免费获取 html单页业务介绍源码...
IDEA导入并启动若依项目步骤(SpringBoot+Vue3)
1.下载后端项目 项目地址:https://gitee.com/y_project/RuoYi-Vue 复制命令,将项目克隆到本地 git clone gitgitee.com:y_project/RuoYi-Vue.git使用IDEA打开(会有点慢,耐心等待),打开后如下 2.Mysql导…...
零基础上手Python数据分析 (22)案例实战]之利用 Matplotlib Seaborn 进行电商销售数据可视化分析
写在前面 —— 图表为刃,洞察先行!综合运用 Pandas、Matplotlib 与 Seaborn,点亮数据价值 本篇通过一个完整的案例实战,体验如何将数据分析与数据可视化紧密结合,让冰冷的数据转化为生动、直观、富有洞察力的视觉故事! 案例目标: 本篇博客将延续我们在第 17 篇案例中…...
图像预处理-霍夫变换
一.概念 霍夫变换是图像处理的一种技术,主要用于检测图像中的直线、圆等几何形状。基本思想就是将图像空间(直角坐标系或极坐标系)中的点映射到(霍夫空间)参数空间中,通过在参数空间中寻找累计最大值实现对特定形状的检测。 接下…...
逐步了解蓝牙 LE 配对(物联网网络安全)
配对是蓝牙 LE 中的一个重要概念。让我们来了解一下蓝牙 LE 配对的基础知识,概述 LE 设备如何在可信设备之间安全地共享密钥。 在之前的一篇文章]中,我们讨论了蓝牙 LE 的安全密钥。我们了解到,蓝牙 LE 设备可以创建并共享三个不同的安全密钥:一个用于数据加密,一个用于创…...
嵌入模型(Embedding Models)原理详解:从Word2Vec到BERT的技术演进
一、什么是嵌入模型? 嵌入模型(Embedding Models) 是一种将高维离散数据(如文本、图像)转换为低维连续向量表示的技术。这些向量能够反映数据的语义关系,使得“语义相近的实体在向量空间中距离更近”。例如…...
解析塔能科技:绿色低碳智慧节能一站式破局之匙
在能源问题日益凸显的当下,绿色低碳、高效节能成为全球发展的重要课题。对各类节能方案进行深入剖析后,可以发现塔能科技的绿色低碳智慧节能一站式解决方案极具创新性与实用性,切实为众多行业面临的能源困境提供了有效解决路径。 直面行业痛点…...
el-menu箭头改为右下
问: el-menu箭头改为右下 回答: :deep(.el-menu){.el-sub-menu .el-sub-menu__icon-arrow{transition: transform 0.3stransform: rotateZ(-90deg) !important;}/* 展开状态:箭头向下 */.el-sub-menu.is-opened .el-sub-menu__icon-arrow,…...
[特殊字符][特殊字符] HarmonyOS相关实现原理聊聊![特殊字符][特殊字符]
Hey小伙伴们~ 今天咱们来聊聊HarmonyOS的实现原理吧!🤔💡 虽然直接看源代码是最直接的方式,但是OpenHarmony项目的开源进度有点让人捉急呢… 不过没关系,我们可以通过已经开放的SDK、IDE、开发示例和编译产物来一探究竟…...
RTSP播放器实现回调RGB|YUV给视觉算法,然后二次编码推送到RTMP服务
引言 在本文中,我们将介绍如何基于大牛直播SDK构建一个功能强大的RTSP|RTMP播放器,该播放器利用自定义SDK解码视频、处理RGB帧,并将其推送到RTMP流中进行直播。这个解决方案非常适合需要在实时视频流中集成视觉算法的场景,在处理…...
补题【Darkness+Different Billing+Dice Game】
文章目录 1.Darkness2.Different Billing3.Dice Game 1.Darkness 题目来源:Darkness I 这题不难想,通过作图我们发现 当nm时直接取对角线就好 当n!m时,取m,n的最小值,那么最小值的这个正方形都可以被填为黑色,剩下的…...
卷积神经网络:视觉炼金术士的数学魔法
🎮 “你以为《赛博朋克2077》的夜之城是画出来的?不,是卷积神经网络’卷’出来的!” 🎮 一、卷积:像素世界的连连看大师 想象你拿着一张《蒙娜丽莎》的拼图——卷积核就像你手中的拼图碎片,在画…...
Python:简介,Python解释器安装,第一个Python程序,开发环境(PyCharm安装和配置、Sublime安装和配置)
目录 Python简介 Python解释器的安装(Windows) 下载和安装Python安装包(解释器) 验证Python安装情况 第一个Python程序 单行运行 多行运行 Python开发环境(安装和配置PyCharm工具) 安装PyCharm Py…...
CE第二次作业
实验要求: 1.配置ssh实现A,B主机互相免密登录 2.配置nginx服务,通过多ip区分多网站 一,配置ssh实现A,B主机互相免密登录 步骤 在主机 A 上生成 SSH 密钥对。把主机 A 的公钥复制到主机 B。在主机 B 上重复上述步骤…...
【阿里云大模型高级工程师ACP习题集】2.4 自动化评测答疑机器人的表现(⭐️⭐️⭐️ 重点章节!!!)
习题集: 【单选题】在使用Ragas评估RAG应用时,Answer Correctness指标计算中,语义相似度是通过以下哪种方式得到的?( ) A. 大模型直接判断 B. 计算文本向量的余弦相似度 C. 对比文本词汇重合度 D. 统计文本字数差异 【多选题】当Context recall指标得分较低时,可采取的优…...
多头注意力
Multi-Head Attention 论文地址 https://arxiv.org/pdf/1706.03762 多头注意力介绍 多头注意力是Transformer模型的关键创新,通过并行执行多个独立的注意力计算单元,使模型能够同时关注来自不同表示子空间的信息。每个注意力头学习不同的语义特征&#x…...
【leetcode100】目标和
1、题目描述 给你一个非负整数数组 nums 和一个整数 target 。 向数组中的每个整数前添加 或 - ,然后串联起所有整数,可以构造一个 表达式 : 例如,nums [2, 1] ,可以在 2 之前添加 ,在 1 之前添加 - …...
动态哈希映射深度指南:从基础到高阶实现与优化
哈希表是计算机科学中最高效的数据结构之一,而动态哈希映射通过智能扩容机制,在实时系统中展现出极强的适应性。本文将深入探讨其实现细节,结合主流框架源码解析,并给出可落地的性能优化方案。 一、动态哈希的数学本质 1. 哈希函…...
leetcode 2799. 统计完全子数组的数目 中等
给你一个由 正 整数组成的数组 nums 。 如果数组中的某个子数组满足下述条件,则称之为 完全子数组 : 子数组中 不同 元素的数目等于整个数组不同元素的数目。 返回数组中 完全子数组 的数目。 子数组 是数组中的一个连续非空序列。 示例 1ÿ…...
使用RabbitMQ实现判题功能
这次主要选用RabbitMQ消息队列来对判题服务和题目服务解耦,题目服务只需要向消息队列发送消息,判题服务从消息队列中取信息去执行判题,然后异步更新数据库即可。 五一宝宝请快点跑~~~~~ 先回顾一下RabbitMQ (1)引入依…...
无过拟合的记忆:分析大语言模型的训练动态
Kushal Tirumala⇤ Aram H. Markosyan⇤ Luke Zettlemoyer Armen Aghajanyan Meta AI 研究 {ktirumala,amarkos,lsz,armenag}fb.com 原文链接:[2210.09262] Physics-Driven Convolutional Autoencoder Approach for CFD Data Compressions 摘要 尽管超大语言模型…...
【Java面试笔记:进阶】16.synchronized底层如何实现?什么是锁的升级、降级?
在 Java 中,synchronized 关键字的底层实现依赖于 对象头(Object Header) 和 监视器锁(Monitor) 机制,并通过 锁的状态升级(Lock Escalation) 来优化同步性能。 1. synchronized 的底层实现 synchronized 的同步机制基于 Monitor 对象,它是同步的基本实现单元。 通过…...
python可视化:北方城市人口流动趋势分析1
python可视化:北方城市人口流动趋势分析1 斑点鱼在做销售数据分析时发现北京天津的同比下滑明显,客流下滑明显。而山东保定的客流同比上升。引起了斑点鱼对于北方人口流动的好奇。 所以本文将分析2025年北方地区(北京、天津、河北、山东、山西、辽宁等)…...
wps excel 常用操作
数据分列 对于有分隔规律的内容,可以通过分隔符将该内容进行分列 例如,以下字符串,可使用Excel对包含IP地址、数据库类型、环境、负责人和日期的字符串进行分列: 192.168.175.211-MySQL 数据库-DEV-李华-2025.06.30 将以上字符串…...
云智融合普惠大模型AI,政务服务重构数智化路径
2025年是“十四五”收官之年,数字政府和政务数智化作为“数字中国”建设的重点,已经取得了显著成效。根据《联合国电子政务调查报告2024》,我国电子政务发展指数全球排名第35位,与2022年相比提升8个名次;其中ÿ…...
全行业软件定制:APP/小程序/系统开发与物联网解决方案
在数字化浪潮席卷全球的今天,软件已经渗透到我们生活的方方面面,成为推动社会进步的重要力量。作为一家专注于专业软件定制开发的公司,哲科软件深知每一个行业、每一个企业都有其独特的需求和痛点。因此,我们致力于提供个性化软件…...
Java虚拟机(JVM)家族发展史及版本对比
Java虚拟机(JVM)家族发展史及版本对比 一、JVM家族发展史 1. 早期阶段(1996-2000) Classic VM(Java 1.0-1.1): 厂商:Sun Microsystems(Oracle前身)。特点&…...
电脑怎么强制退出程序回到桌面 详细操作步骤
电脑日常使用过程中,我们有时会遇到程序无响应或卡死的情况,这时需要采取措施强制关闭这些程序才能保持电脑的正常工作和运行。那么,电脑如何强制退出程序呢?其实方法有很多种,下面便为大家介绍几种电脑强制关闭程序的…...
蓝牙 LE:安全模式和程序说明(蓝牙中的网络安全)
在蓝牙低功耗 (BLE) 中,安全性是一个多方面的难题。了解 BLE 的三种主要安全模式以及五个关键的 BLE 安全程序。 毫无疑问,低功耗蓝牙 (BLE) 技术的迅猛发展为我们的生活带来了更多便利。然而,随着低功耗蓝牙设备的普及,人们对其安全性的担忧也日益加剧。 与普遍看法相反…...
低代码平台开发胎压监测APP
项目介绍 该项目是一个利用Flutter框架和蓝牙技术实现轮胎压力实时监测的应用。 主要功能如下: 用于接收蓝牙模块传输的胎压数据,并实时显示胎压值。APP对接收到的胎压数据进行处理,如单位转换、数据滤波等,然后将处理后的胎压值…...
GNOME扩展入门:日期时间
Getting Started | GNOME JavaScript 1.扩展路径 ~/.local/share/gnome-shell/extensions/ 2.新建文件夹 datetimesonichy 3.metadata.json {"uuid": "datetimesonichy","name": "datetime","description": "Dis…...
NLP高频面试题(五十二)——深度学习优化器详解
在深度学习的训练过程中,各种基于梯度的优化器肩负着寻找损失函数最优解的重任。最基础的梯度下降法通过沿着损失函数负梯度方向迭代更新参数,实现对模型参数的优化;而随机梯度下降(SGD)则以更高的计算效率和内存利用率在大规模数据集上大放异彩,但也因更新噪声大、易陷入…...
SLAM常用地图对比示例
序号地图类型概述1格栅地图将现实环境栅格化,每一个栅格用 0 和 1 分别表示空闲和占据状态,初始化为未知状态 0.52特征地图以点、线、面等几何特征来描绘周围环境,将采集的信息进行筛选和提取得到关键几何特征3拓扑地图将重要部分抽象为地图&…...
Web常见攻击方式及防御措施
一、常见Web攻击方式 1. 跨站脚本攻击(XSS) 攻击原理:攻击者向网页注入恶意脚本,在用户浏览器执行 存储型XSS:恶意脚本存储在服务器(如评论区) 反射型XSS:恶意脚本通过URL参数反射给用户 DOM型XSS&…...
java.lang.IllegalArgumentException: URI is not hierarchical报错
java.lang.IllegalArgumentException: URI is not hierarchical Thread.currentThread().getContextClassLoader("类的全路径").getClass().newInstance()一个类的静态块初始化异常了,后面调用这个类创建对象会报错吗? 是的,如果一…...
118. 杨辉三角
目录 一、问题描述 二、解题思路 三、代码 四、复杂度分析 一、问题描述 给定一个非负整数 numRows,生成「杨辉三角」的前 numRows 行。 在「杨辉三角」中,每个数是它左上方和右上方的数的和。 二、解题思路 每一行的第一个和最后一个元素是 1&…...