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

AudioRecord 录制pcm转wav

pcm转wav

    • PCM 格式校验
    • pcm 添加 wav 头信息
    • WAV
    • WAV 格式检验
    • 小端序?
  • 参考地址

PCM 格式校验

/*** 专业PCM文件验证(支持动态参数与多格式)* @param silenceThreshold 静音检测阈值(0.0~1.0),默认90%零值为静音* @return false表示文件无效(自动打印错误日志)*/
fun validatePcmFile(file: File,sampleRate: Int,channelConfig: Int,audioFormat: Int,silenceThreshold: Float = 0.9f
): Boolean {// 基础参数校验require(silenceThreshold in 0.0f..1.0f) { "静音阈值必须在0~1之间" }require(audioFormat == AudioFormat.ENCODING_PCM_8BIT || audioFormat == AudioFormat.ENCODING_PCM_16BIT || audioFormat == AudioFormat.ENCODING_PCM_FLOAT) {"不支持的音频格式: $audioFormat"}// 基础检查if (!file.exists()) {Log.e(TAG, "PCM文件不存在: ${file.absolutePath}")return false}if (file.length() == 0L) {Log.e(TAG, "PCM文件为空: ${file.absolutePath}")return false}// 调试日志Log.d(TAG, "开始校验PCM文件: ${file.name} [大小=${file.length()} bytes]")try {// 计算位深度和字节对齐val bytesPerSample = when (audioFormat) {AudioFormat.ENCODING_PCM_8BIT -> 1AudioFormat.ENCODING_PCM_16BIT -> 2AudioFormat.ENCODING_PCM_FLOAT -> 4else -> 2 // 不会执行(已校验)}// 1. 最小帧检查val minFrameSize = AudioRecord.getMinBufferSize(sampleRate, channelConfig, audioFormat)val minRequiredSize = minFrameSize * MIN_FRAME_MULTIPLIERif (file.length() < minRequiredSize) {Log.e(TAG, "PCM文件过小: ${file.length()} bytes (至少需要 $minRequiredSize bytes)")return false}// 2. 数据有效性检查val buffer = ByteArray(1024)var zeroCount = 0var totalSamples = 0FileInputStream(file).use { stream ->var bytesRead: Intwhile (true) {bytesRead = stream.read(buffer)if (bytesRead == -1) breakval samples = bytesRead / bytesPerSampleval byteBuffer = ByteBuffer.wrap(buffer, 0, bytesRead).apply {order(ByteOrder.LITTLE_ENDIAN)}when (bytesPerSample) {1 -> { // 8-bit unsignedfor (i in 0 until samples) {val value = byteBuffer.get().toInt() and 0xFFif (value == 128) zeroCount++ // 静音中点为128}}2 -> { // 16-bit signedfor (i in 0 until samples) {if (byteBuffer.short == 0.toShort()) zeroCount++}}4 -> { // 32-bit floatfor (i in 0 until samples) {if (abs(byteBuffer.float - 0.0f) < 1e-6) zeroCount++}}}totalSamples += samples}}Log.d(TAG, "样本统计: 总数=$totalSamples, 零值=$zeroCount")// 静音检测if (totalSamples > 0) {val zeroRatio = zeroCount.toFloat() / totalSamplesif (zeroRatio > silenceThreshold) {Log.e(TAG, "静音数据过多: ${"%.1f%%".format(zeroRatio * 100)} ≥ ${"%.0f%%".format(silenceThreshold * 100)}")return false}} else {Log.e(TAG, "文件无有效样本数据")return false}Log.i(TAG, "PCM文件验证通过")return true} catch (e: Exception) {Log.e(TAG, "验证异常: ${e.javaClass.simpleName} - ${e.message}", e)return false}
}

pcm 添加 wav 头信息

 /*** 头部信息共44字节* @param sampleRate* @param channels* @param bitDepth* @param dataSize* @return* @throws IOException*/fun getWavHeader(sampleRate: Int, channels: Int, bitDepth: Int, dataSize: Long): ByteArray {val header = ByteArray(44)// ChunkID,RIFF标识header[0] = 'R'.code.toByte()header[1] = 'I'.code.toByte()header[2] = 'F'.code.toByte()header[3] = 'F'.code.toByte()// ChunkSize,文件长度val totalSize = dataSize + 36header[4] = (totalSize and 0xffL).toByte()header[5] = ((totalSize shr 8) and 0xffL).toByte()header[6] = ((totalSize shr 16) and 0xffL).toByte()header[7] = ((totalSize shr 24) and 0xffL).toByte()// Format,WAVE标识header[8] = 'W'.code.toByte()header[9] = 'A'.code.toByte()header[10] = 'V'.code.toByte()header[11] = 'E'.code.toByte()// Subchunk1ID,fmt标识header[12] = 'f'.code.toByte()header[13] = 'm'.code.toByte()header[14] = 't'.code.toByte()header[15] = ' '.code.toByte()// Subchunk1Size,格式信息长度header[16] = 16header[17] = 0header[18] = 0header[19] = 0// AudioFormat,音频格式(PCM为1)header[20] = 1header[21] = 0// NumChannels,声道数header[22] = channels.toByte()header[23] = 0// SampleRate,采样率header[24] = (sampleRate and 0xff).toByte()header[25] = ((sampleRate shr 8) and 0xff).toByte()header[26] = ((sampleRate shr 16) and 0xff).toByte()header[27] = ((sampleRate shr 24) and 0xff).toByte()// ByteRate,比特率val byteRate = sampleRate * channels * bitDepth / 8header[28] = (byteRate and 0xff).toByte()header[29] = ((byteRate shr 8) and 0xff).toByte()header[30] = ((byteRate shr 16) and 0xff).toByte()header[31] = ((byteRate shr 24) and 0xff).toByte()// BlockAlign,块对齐val blockAlign = channels * bitDepth / 8header[32] = blockAlign.toByte()header[33] = 0// BitsPerSample,采样位深度header[34] = bitDepth.toByte()header[35] = 0// Subchunk2ID,data标识header[36] = 'd'.code.toByte()header[37] = 'a'.code.toByte()header[38] = 't'.code.toByte()header[39] = 'a'.code.toByte()// Subchunk2Size,音频数据长度 dataHdrLengthheader[40] = (dataSize and 0xffL).toByte()header[41] = ((dataSize shr 8) and 0xffL).toByte()header[42] = ((dataSize shr 16) and 0xffL).toByte()header[43] = ((dataSize shr 24) and 0xffL).toByte()return header}

WAV

DEPSEK

偏移地址字段名称说明
00-03ChunkId固定值 "RIFF" (ASCII编码)
04-07ChunkSize文件总字节数 - 8(从地址08开始到文件末尾的总字节数,小端存储)
08-11fccType固定值 "WAVE" (ASCII编码)
12-15SubChunkId1固定值 "fmt "(包含末尾空格)
16-19SubChunkSize1fmt块数据大小(标准PCM为16,扩展格式可能为1840,小端存储)
20-21FormatTag编码格式:1=PCM,3=IEEE浮点(小端存储)
22-23Channels声道数:1=单声道,2=双声道(小端存储)
24-27SamplesPerSec采样率(Hz,如44100,小端存储)
28-31BytesPerSec字节率 = 采样率 × 声道数 × 位深/8(小端存储)
32-33BlockAlign每帧字节数 = 声道数 × 位深/8(小端存储)
34-35BitsPerSample位深:8/16/24/32(小端存储)
36-39SubChunkId2固定值 "data"
40-43SubChunkSize2音频数据长度(字节数 = 采样总数 × 声道数 × 位深/8,小端存储)
44-…Data音频原始数据(二进制格式,与FormatTag和BitsPerSample匹配)

WAV 格式检验

/*** 专业WAV文件验证(增强版)* 注意:WAV_HEADER_SIZE常量在此版本中已不再需要,因采用动态块遍历机制* @return false表示文件无效(自动打印错误日志)*/
fun validateWavFile(file: File): Boolean {// 基础文件检查if (!file.exists()) {Log.e(TAG, "❌ WAV文件不存在: ${file.absolutePath}")return false}if (file.length() < MIN_WAV_FILE_SIZE) { // 至少需要包含RIFF头、WAVE标识和一个子块Log.e(TAG, "❌ 文件过小: ${file.length()} bytes (至少需要 $MIN_WAV_FILE_SIZE bytes)")return false}try {RandomAccessFile(file, "r").use { raf ->/* ------------------------- RIFF头验证 ------------------------- */// 读取前4字节验证RIFF标识val riffHeader = ByteArray(4).apply { raf.read(this) }if (String(riffHeader) != "RIFF") {Log.e(TAG, "❌ 无效的RIFF头: ${String(riffHeader)} (应为\"RIFF\")")return false}Log.d(TAG, "✅ RIFF头验证通过")/* ----------------------- RIFF块大小验证 ----------------------- */// 读取小端序的RIFF块大小(文件总大小-8)val riffChunkSize = raf.readLittleEndianInt()val expectedRiffSize = file.length() - 8if (riffChunkSize != expectedRiffSize.toInt()) {Log.e(TAG, "❌ RIFF大小不匹配 | 声明:$riffChunkSize | 实际:$expectedRiffSize")return false}Log.d(TAG, "✅ RIFF块大小验证通过 (${riffChunkSize}bytes)")/* ----------------------- WAVE标识验证 ------------------------ */val waveHeader = ByteArray(4).apply { raf.read(this) }if (String(waveHeader) != "WAVE") {Log.e(TAG, "❌ 无效的WAVE标识: ${String(waveHeader)} (应为\"WAVE\")")return false}Log.d(TAG, "✅ WAVE标识验证通过")/* --------------------- 子块遍历验证 --------------------- */var fmtVerified = falsevar dataSize = 0LchunkLoop@ while (raf.filePointer < file.length()) {// 读取当前块头信息val chunkId = ByteArray(4).apply { raf.read(this) }.toString(Charsets.US_ASCII)val chunkSize = raf.readLittleEndianInt().toLong() and 0xFFFFFFFFL // 转为无符号when (chunkId) {"fmt " -> { // 格式块验证/* --------------- 基本格式块验证 --------------- */if (chunkSize < 16) {Log.e(TAG, "❌ fmt块过小: $chunkSize bytes (至少需要16 bytes)")return false}// 读取音频格式(1=PCM)val formatTag = raf.readLittleEndianShort()if (formatTag != 1.toShort()) {Log.e(TAG, "❌ 非PCM格式 | 格式码:$formatTag (应为1)")return false}Log.d(TAG, "✅ PCM格式验证通过")// 跳过声道数、采样率等参数(此处不验证具体音频参数)raf.skipBytes(6) // 跳过2(short)+4(int)// 验证位深是否为整数字节val bitsPerSample = raf.readLittleEndianShort()if (bitsPerSample % 8 != 0) {Log.e(TAG, "❌ 非常规位深: ${bitsPerSample}bits (应为8的倍数)")return false}Log.d(TAG, "✅ 位深验证通过 (${bitsPerSample}bits)")fmtVerified = trueraf.skipBytes(chunkSize.toInt() - 16) // 跳过剩余格式数据}"data" -> { // 数据块验证/* --------------- 数据块大小验证 --------------- */dataSize = chunkSizeval declaredDataEnd = raf.filePointer + chunkSizeval actualDataEnd = file.length()if (declaredDataEnd > actualDataEnd) {Log.e(TAG, "❌ 数据块越界 | 声明结束位置:$declaredDataEnd | 文件大小:$actualDataEnd")return false}Log.d(TAG, "✅ 数据块大小验证通过 (${chunkSize}bytes)")break@chunkLoop // 找到数据块后终止遍历}else -> { // 其他块处理Log.d(TAG, "⏭ 跳过块: $chunkId (${chunkSize}bytes)")raf.skipBytes(chunkSize.toInt())}}}/* ------------------- 最终完整性检查 ------------------- */if (!fmtVerified) {Log.e(TAG, "❌ 缺少必需的fmt块")return false}if (dataSize == 0L) {Log.e(TAG, "❌ 缺少data块")return false}return true}} catch (e: Exception) {Log.e(TAG, "❌ 文件解析异常: ${e.javaClass.simpleName} - ${e.message}")return false}
}/* ------------------------- 扩展函数:小端序读取 ------------------------- */
private fun RandomAccessFile.readLittleEndianInt(): Int {return ByteArray(4).apply { read(this) }.let {(it[3].toInt() shl 24) or (it[2].toInt() shl 16) or (it[1].toInt() shl 8) or it[0].toInt()}
}private fun RandomAccessFile.readLittleEndianShort(): Short {return ByteArray(2).apply { read(this) }.let {((it[1].toInt() shl 8) or it[0].toInt()).toShort()}
}companion object {private const val TAG = "WavValidator"private const val MIN_WAV_FILE_SIZE = 44L // RIFF头(12) + fmt块(24) + data块头(8)
}

小端序?

在 Android 中,AudioRecord 录制的音频数据默认是 PCM 格式,且字节序(Endianness)为 小端序(Little-Endian)。这是 Android 音频系统的默认行为,与大多数移动设备和 x86/ARM 平台的处理器架构一致。

大 2 小

    /**************** 字节序转换实现 ****************/private fun convertEndian(inputFile: File): File? {return try {val outputFile = createTempPcmFile("converted_")FileInputStream(inputFile).use { input ->FileOutputStream(outputFile).use { output ->val buffer = ByteArray(4096) // 4KB缓冲区var bytesRead: Intwhile (input.read(buffer).also { bytesRead = it } != -1) {// 确保读取的是完整shortval validLength = bytesRead - (bytesRead % 2)if (validLength == 0) continue// 转换字节序convertByteOrder(buffer, validLength)output.write(buffer, 0, validLength)}}}outputFile} catch (e: Exception) {Log.e(TAG, "Endian conversion failed", e)null}}private fun convertByteOrder(data: ByteArray, length: Int) {val byteBuffer = ByteBuffer.wrap(data, 0, length)val shortBuffer = byteBuffer.order(ByteOrder.BIG_ENDIAN).asShortBuffer()val shorts = ShortArray(shortBuffer.remaining())shortBuffer.get(shorts)ByteBuffer.wrap(data).order(ByteOrder.LITTLE_ENDIAN).asShortBuffer().put(shorts)}

参考地址

PCM 2 WAV:
https://blog.csdn.net/qq_36451275/article/details/135057683

PCM 2 WAV:
https://blog.csdn.net/m0_54198552/article/details/145653031

depsek

转:Android音频开发(4):PCM转WAV格式音频
https://www.jianshu.com/p/90c77197f1d4

相关文章:

AudioRecord 录制pcm转wav

pcm转wav PCM 格式校验pcm 添加 wav 头信息WAVWAV 格式检验小端序&#xff1f; 参考地址 PCM 格式校验 /*** 专业PCM文件验证&#xff08;支持动态参数与多格式&#xff09;* param silenceThreshold 静音检测阈值&#xff08;0.0~1.0&#xff09;&#xff0c;默认90%零值为静…...

625SJBH网上便利店的设计与实现

1前 言 目前&#xff0c;网络正以一种前所未有的冲击力在影响着人类的活动&#xff0c;包括人类的生产和日常生活。网络的诞生和发展&#xff0c;颠覆了传统的信息传播方式&#xff0c;冲破了存在于传统交流方式中时间和空间的种种壁垒&#xff0c;极大地改变了人类从物质到精…...

如何开发英语在线训练小程序:从0到1的详细步骤

在数字化学习的浪潮下&#xff0c;英语在线训练小程序凭借便捷、高效的学习模式&#xff0c;成为众多英语学习者的得力助手。如果你也想开发一款独具特色的英语在线训练小程序&#xff0c;不妨参考以下步骤&#xff0c;开启你的小程序开发之旅。 一、前期规划 &#xff08;…...

java设计模式-装饰者模式

装饰者模式(Decorator) 定义 1、动态的将新功能附加到对象上&#xff0c;在对象功能扩展方面&#xff0c;他比继承更有弹性&#xff0c;也体现了开闭原则(OCP) 2、这里提到的动态的将新功能附加到对象和OCP原则&#xff0c;在后面应用实际上会以代码的形式体现。 //饮料 // 饮…...

我提了一个 Androidx IssueTracker

问题 在运行 gradle plugin 插件的 transform R8 阶段出现了报错 Caused by: com.android.tools.r8.internal.xk: java.lang.NullPointerException: Cannot invoke “String.length()” because “” is null 报错日志 FAILURE: Build failed with an exception.* What went w…...

spring mvc @ResponseBody 注解转换为 JSON 的原理与实现详解

ResponseBody 注解转换为 JSON 的原理与实现详解 1. 核心作用 ResponseBody 是 Spring MVC 的一个注解&#xff0c;用于将方法返回的对象直接序列化为 HTTP 响应体&#xff08;如 JSON 或 XML&#xff09;&#xff0c;而不是通过视图解析器渲染为视图&#xff08;如 HTML&…...

RK3588芯片NPU的使用:Windows11 Docker中运行MobileNet模型以及部署到开发板进行目标检测

本文的目标 本文将在RKNN Docker环境&#xff08;见本系列的第二篇文章&#xff09;中练习MobileNet图像分类示例&#xff0c;并通过adb工具部署到RK3588开发板。 MobileNet简介请参考上一篇文章。 开发环境说明 主机系统&#xff1a;Windows11目标设备&#xff1a;搭载RK35…...

智能仓储数字孪生Demo(Unity实现)

一、项目背景与行业痛点 医药流通行业仓储管理面临三大核心挑战&#xff1a; 合规性风险&#xff1a;GSP&#xff08;药品经营质量管理规范&#xff09;对温湿度、药品批次追溯的严苛要求&#xff0c;传统人工记录易出错效率瓶颈&#xff1a;库区布局复杂&#xff0c;人工巡检…...

Qt上hook钩子的使用,监测键盘和鼠标。

演示平台&#xff1a;windows。 编译环境&#xff1a;Qt5.12.2 MinGW 64-bit Windows API&#xff1a; ///加载钩子 /*** SetWindowsHookEx 函数解释* int idHook 所监控的挂钩类型* HOOKPROC lpfn 监控信息的处理函数* HINSTANCEhMod 监控信息的动态链接位置 nullptr则与本线…...

Android12源码编译之预置Android Studio项目Android.mk文件编写

1、在AndroidManifest.xml文件中添加package"com.sprd.silentinstalldemo"属性&#xff0c;因为新版本的Android Studio默认生成的AndroidManifest.xml是没有这个属性值的 <?xml version"1.0" encoding"utf-8"?> <manifest xmlns:an…...

微服务注册中心选择指南:Eureka vs Consul vs Zookeeper vs Nacos

文章目录 引言微服务注册中心概述什么是服务注册与发现选择注册中心的标准 常见的微服务注册中心1. Eureka1.1 理论基础1.2 特点1.3 示例代码 2. Consul2.1 理论基础2.2 特点2.3 示例代码 3. Zookeeper3.1 理论基础3.2 特点3.3 示例代码 4. Nacos4.1 理论基础4.2 特点4.3 示例代…...

pg_waldump无法定位WAL文件问题

目录 排查pg_waldump无法定位WAL文件问题的步骤1. 确认WAL文件路径配置2. 检查WAL文件名格式3. 验证文件存在性4. 检查文件权限5. 时间线历史文件检查6. 使用pg_controldata验证状态7. 尝试指定完整路径 典型错误场景及解决方案 排查pg_waldump无法定位WAL文件问题的步骤 1. 确…...

Mysql安装

Mysql安装 1. windows安装1.1 官网下载1.2 安装 1. windows安装 1.1 官网下载 官网下载 选择对于版本&#xff0c;然后跳转到下载页 1.2 安装...

Windows版-RabbitMQ自动化部署

一键完成Erlang环境变量配置&#xff08;ERLANG_HOME系统变量&#xff09;‌ 一键完成RabbitMQ环境变量配置&#xff08;RabbitMQ系统变量&#xff09;‌ 实现快速安装部署RabbitMQ PS&#xff1a; 需提前下载安装&#xff1a; - otp_win64_25.0.exe (Erlang) - rabbit…...

spring mvc的拦截器HandlerInterceptor 接口详解

HandlerInterceptor 接口详解 1. 接口方法说明 方法作用执行时机返回值/注意事项preHandle请求处理前拦截在控制器方法执行前调用返回 false 中断后续流程&#xff1b;返回 true 继续执行postHandle控制器方法执行后拦截在控制器方法返回结果后&#xff0c;视图渲染前调用无返…...

Linux平台内存泄漏检测工具介绍: ASan vs Valgrind

目录&#xff1a; 前言Valgrind 介绍在Ubuntu上安装Valgrind 核心主要功能Valgrind 基本用法1. --leak-checkfull2. --show-leak-kindsall3. --track-originsyes4. 其他常用选项--tool<name>--log-file<filename>-v / --verbose--error-exitcode<n> 示例命令…...

c# 数据结构 链表篇 有关单链表的一切

本人能力有限,本文仅作学习交流与参考,如有不足还请斧正 目录 0.单链表好处 0.5.单链表分类 1.无虚拟头节点情况 图示: 代码: 头插/尾插 删除 搜索 遍历全部 测试代码: 全部代码 2.有尾指针情况 尾插 全部代码 3.有虚拟头节点情况 全部代码 4.循环单链表 几个…...

二叉树层平均值:层序遍历+队列解法详解

给定一个非空二叉树的根节点 root , 以数组的形式返回每一层节点的平均值。与实际答案相差 10-5 以内的答案可以被接受。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;[3.00000,14.50000,11.00000] 解释&#xff1a;第 0 层的平均值为 …...

解决 Docker Swarm 集群节点故障:从问题剖析到修复实战

解决 Docker Swarm 集群节点故障&#xff1a;从问题剖析到修复实战 在使用 Docker Swarm 构建容器编排集群时&#xff0c;可能会遭遇各种难题。本文将分享一次处理 Docker Swarm 集群节点故障的实战经历&#xff0c;涵盖问题出现的缘由、详细剖析以及完整的解决步骤&#xff0…...

【Java中级】11章、注解、元注解介绍、快速入门,了解java注解的基本使用方式【2】

文章内容 JDK内置的基本注释类型 Override DeprecatedSuppressWarnings 元注解 对注释进行注解 ❤️内容涉及注解的定义&#xff0c;快速入门&#xff0c;注意事项 &#x1f308; 跟着B站一位老师学习的内部类内容&#xff0c;现写这篇文章为学习内部类的小伙伴提供思路支持&…...

Qt中自定义插件和库(1)

Qt中自定义插件和库(1) 在Qt中自定义插件和库的方法有两种&#xff1a; 1.提升法。 2.自定义Qt Designer 插件法。 下面就以《Qt 5.9 C开发指南》一书中的例子来讲解。下面先讲提升法。 一、提升法 提升法(Promotion)是Qt Designer中重用自定义控件的一种方法&#xff0c…...

RK3568下QT实现视频播放器

在开发多媒体应用时,视频播放器是常见的项目。QT 作为一款跨平台的 C++ 应用程序开发框架,凭借丰富的类库和工具,让开发视频播放器变得简单。本文将介绍如何使用 QT 的QMediaPlayer和QVideoWidget类,实现一个简单的视频播放器,并逐步添加打开视频、播放、暂停、停止以及进…...

Shell脚本核心要点总结

刷题&#xff1a; Shell脚本核心要点总结 一、Shell基础 定义&#xff1a;Shell是用户与内核交互的接口&#xff0c;本质是多个指令的集合&#xff0c;需遵循逻辑关系。类型&#xff1a; 编译型语言&#xff08;如C&#xff09;&#xff1a;需编译器&#xff08;如gcc&#xf…...

C++-FFmpeg-(5)-1-ffmpeg原理-ffmpeg编码接口-AVFrame-AVPacket-最简单demo

1.视频编码原理 2.FFMpeg编码接口和AVPacket结构体详解 2.1ffmpeg编码接口 -编码器上下文 2.2AVPacket结构体 2.3AVFrame结构体 3.视频播放最简单demo 3.1FFMpeg编码器获取和上下文打开 3.2视频帧创建和测试 1.视频编码原理 1.1 流程&#xff1a;像素格式转换-&…...

Opencv计算机视觉编程攻略-第十二节 处理视频序列

视频由一系列图像构成&#xff0c;这些图像称为帧&#xff0c;帧是以固定时间间隔获取的&#xff08;称为帧速率&#xff0c;通常用帧/秒表示&#xff0c;例如大疆无人机抽取每一帧&#xff09;&#xff0c;本文将介绍如何读取、处理和存储视频序列。如果从视频序列中提取出独立…...

浮点许可优化管理软件 - 智能许可管理专家

为什么选择浮点许可优化管理软件? 在当今数字化时代&#xff0c;企业软件许可支出持续攀升&#xff0c;如何实现许可资源的最优配置成为一大挑战。浮点许可优化管理软件通过先进的算法和自动化技术&#xff0c;帮助企业实现许可资源的精准管理和成本优化。 革命性的智能化功能…...

Spring Boot接口返回Long类型的数据时丢失精度的全局处理

1、问题 当实体类中的字段为Long类型时&#xff0c;通过Ajax请求返回给前段&#xff0c;在js中数据会丢失精度 直接通过postman请求或通过浏览器请求&#xff0c;看下响应则不会丢失精度 2、处理方式 1、使用JsonSerialize注解 JsonSerialize(using ToStringSerializer.…...

量子计算入门:开启未来计算的次元之门

在科幻电影中&#xff0c;我们常看到“量子计算机”被描绘成无所不能的黑科技——破解密码、模拟宇宙、瞬间完成超算百年的任务。但现实中&#xff0c;量子计算究竟是什么&#xff1f;它真的能颠覆传统计算机吗&#xff1f; 一、从“硬币”到“薛定谔的猫”&#xff1a;量子世界…...

学习日志37—基于变分量子电路的量子机器学习算法综述

基于变分量子电路的量子机器学习算法综述 论文原链接参考&#xff1a;https://crad.ict.ac.cn/article/cstr/32373.14.issn1000-1239.202330979 这篇综述的核心内容是基于变分量子电路&#xff08;VQCs&#xff09;的量子机器学习算法的研究现状、应用进展以及面临的挑战和未…...

初入Web网页开发

1、网页哪些内容 1.1 三个核心文件的作用 index.html&#xff1a;网页的骨架&#xff0c;用HTML编写网页结构和内容。 script.js&#xff1a;网页的行为&#xff0c;用JavaScript实现交互功能&#xff08;如按钮点击事件&#xff09;。 styles.css&#xff1a;网页的外观&…...

Vue进行前端开发流程

一、创建vue项目 创建vue项目&#xff1a;先进入要操作的目录下&#xff0c;注意本项目是用vue2开发的。 vue create vue项目名 二、项目开发 1.创建项目结构 2.开发功能模块 主入口App.vue <template><div class"boss-app"><Header /><m…...

【深度学习:实战篇】--PyTorch+Transformer谣言检测系统

任务&#xff1a;构建一个多模态谣言检测模型。 数据集描述如下&#xff1a; 数据集包含以下模态&#xff1a; 谣言文本&#xff1a;谣言的核心文本信息。2. 配图&#xff1a;与谣言文本相关的图像数据&#xff1b;3. OCR 文本&#xff1a;可以通过 PaddleOCR 从配图中提取的…...

PostGreSQL/openGauss表膨胀处理

如果面试官问你&#xff0c;Oracle与PG/OG最大的区别是什么&#xff1f;你要是没回答出MVCC机制&#xff0c;表膨胀&#xff0c;那你多半挂了。 在PG/OG数据库中&#xff0c;命令vacuum full&#xff0c;插件pg_repack用于处理表膨胀&#xff0c;但是别高兴得太早&#xff0c;如…...

视频融合平台EasyCVR搭建智慧粮仓系统:为粮仓管理赋能新优势

一、项目背景 当前粮仓管理大多仍处于原始人力监管或初步信息化监管阶段。部分地区虽采用了简单的传感监测设备&#xff0c;仍需大量人力的配合&#xff0c;这不仅难以全面监控粮仓复杂的环境&#xff0c;还容易出现管理 “盲区”&#xff0c;无法实现精细化的管理。而一套先进…...

基于 Node.js 和 Spring Boot 的 RSA 加密登录实践

在当今的互联网应用开发中&#xff0c;用户数据的安全性至关重要。登录功能作为用户进入系统的第一道防线&#xff0c;其安全性更是不容忽视。本文将介绍一种基于 RSA 加密的登录方案&#xff0c;前端使用 Node.js 的 node-forge 库对密码进行公钥加密&#xff0c;后端使用 Spr…...

jupyter在Pycharm中遇到的一个问题

jupyter比较简洁&#xff0c;可以分块执行&#xff0c;下面显示结果&#xff0c;还能用Markdown写注释&#xff0c;总体来说来还是比较好用的。 但是遇到了一个奇怪的问题&#xff0c;从一个py文件中导入一个函数&#xff0c;结果输出为None。但是如果直接把这个函数的内容复制…...

十二、buildroot系统 adb登录权限设置

4.6.4、adb权限设置 ​ android-adbd 是 ADB&#xff08;Android Debug Bridge&#xff09;的守护进程&#xff0c;允许开发者远程访问和调试设备。它通常用于 Android 设备&#xff0c;但在嵌入式 Linux上&#xff0c;也可以用来提供远程 shell、文件传输和应用调试功能。 ​…...

MySQL、Oracle 和 PostgreSQL 是三种主流的关系型数据库的主要原理性差异分析

MySQL、Oracle 和 PostgreSQL 是三种主流的关系型数据库&#xff0c;它们在底层原理和设计哲学上存在显著差异&#xff0c;尤其在存储引擎、事务处理、并发控制、索引结构、复制机制等方面。以下是它们的主要原理性差异分析&#xff1a; 1. 存储引擎与架构设计 MySQL 多存储引…...

【AI开源大模型工具链ModelEngine】【01】应用框架-源码编译运行

ModelEngine提供从数据处理、知识生成&#xff0c;到模型微调和部署&#xff0c;以及RAG&#xff08;Retrieval Augmented Generation&#xff09;应用开发的AI训推全流程工具链。 GitCode开源地址&#xff1a;https://gitcode.com/ModelEngineGitee开源地址&#xff1a;https…...

一文掌握 google浏览器插件爬虫 的制作

更多内容请见: 爬虫和逆向教程-专栏介绍和目录 文章目录 一、技术栈1. 前端技术(浏览器插件开发)2. 爬虫技术3. 后端(可选)4. 整体技术栈组成二、开发步骤1. 创建 Chrome 插件基础结构(1)`manifest.json` 配置(2)前端页面(`popup.html`)(3)前端逻辑(`popup.js`)…...

【leetcode 100】贪心Java版本

划分字母区间 题目 我的思路&#xff1a;第一次没有一点思路&#xff0c;第二次看了官网思路后&#xff0c;写的以下答案&#xff0c;没有搞明白循环遍历&#xff0c; //是不对的以下&#xff1a; class Solution {public List<Integer> partitionLabels(String s) {Li…...

Linux用Wireshark进行Thread网络抓包关键步骤

用Nordic nRF52840 Dongle作为RCP配合Wireshark进行Thread网络抓包是debug Thread网络的有效工具之一&#xff0c;主要流程在这里&#xff0c;不再赘述&#xff1a;官方流程 但是按官方流程会卡在一个地方&#xff0c;第一次费劲解决后&#xff0c;今天在另一台机器重新配的时…...

项目管理中客户拒绝签字验收?如何处理和预防

客户拒绝签字验收&#xff1f;如何处理和预防&#xff1f;核心在于&#xff1a;正面沟通、证据留存、灵活应对、合同条款明确、阶段验收机制。其中正面沟通格外关键&#xff0c;如果在发现客户迟迟不愿签字时能够主动沟通&#xff0c;了解其顾虑或不满并迅速针对性解决&#xf…...

docker 修改镜像源教程

当在拉取镜像时报以下错误时&#xff0c;可以通过更换镜像源解决 rootlocalhost:/etc/docker# docker pull mysql Using default tag: latest Error response from daemon: Get "https://registry-1.docker.io/v2/": net/http: request canceled while waiting for …...

【JAVA】十、基础知识“类和对象”干货分享~(三)

目录 1. 封装 1.1 封装的概念 1.2 访问限定符 public&#xff08;公开访问&#xff09; private&#xff08;私有访问&#xff09; 1.3 包 1.3.1 包的概念 1.3.2 导入包中的类 1.3.3 自定义包 2. static成员 2.1 static变量&#xff08;类变量&#xff09; 2.1.1 sta…...

Open GL ES -> SurfaceView + 自定义EGL实现OpenGL渲染框架

SurfaceView 自定义EGL实现OpenGL渲染 在Android开发中&#xff0c;当需要灵活控制OpenGL渲染或在多个Surface间共享EGL上下文时&#xff0c;自定义EGL环境是必要的选择 核心实现流程 -------------------- -------------------- -------------------- | 1. 创建Su…...

Solidity入门实战—web3

项目介绍 在这个项目中&#xff0c;我们建立一个小型智能合约应用&#xff0c;他允许用户向合约地址捐赠&#xff0c;允许合约所有者从合约中提取余额&#xff1b;并且还设定了捐赠的金额门槛&#xff1b;针对直接对地址进行的捐赠行为&#xff0c;我们也予以记录 源代码 ht…...

Open Scene Graph动画系统

OSG 提供了强大的动画功能&#xff0c;支持多种动画实现方式&#xff0c;从简单的变换动画到复杂的骨骼动画。以下是 OSG 动画系统的全面介绍&#xff1a; 1. 基本动画类型 1.1 变换动画 (Transform Animation) // 创建动画路径 osg::AnimationPath* createAnimationPath(co…...

无需libpacp库,BPF指令高效捕获指定数据包

【环境】无libpacp库的Linux服务器 【要求】高效率读取数据包&#xff0c;并过滤指定端口和ip 目前遇到两个问题 一是手写BPF&#xff0c;难以兼容&#xff0c;有些无法正常过滤二是性能消耗问题&#xff0c;尽可能控制到1% 大方向&#xff1a;过滤数据包要在内核层处理&…...

重回全面发展亲自操刀

项目场景&#xff1a; 今年工作变动&#xff0c;优化后在一家做国有项目的私人公司安顿下来了。公司环境不如以前&#xff0c;但是好在瑞欣依然可以每天方便的买到。人文氛围挺好&#xff0c;就是工时感觉有点紧&#xff0c;可能长期从事产品迭代开发&#xff0c;一下子转变做项…...