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

Android音视频开发之音频录制和播放

1.封装音频录制工具类:

public class RecorderAudioManagerUtils {private static volatile RecorderAudioManagerUtils mInstance;public static RecorderAudioManagerUtils getInstance() {if (mInstance == null) {synchronized (RecorderAudioManagerUtils.class) {if (mInstance == null) {mInstance = new RecorderAudioManagerUtils();}}}return mInstance;}}

2.音频录制方法:

    public void startRecord(WeakReference<Context> weakReference) {this.weakReference = weakReference;Log.e(TAG, "开始录音");//生成PCM文件String fileName = DateFormat.format("yyyy-MMdd-HHmmss", Calendar.getInstance(Locale.getDefault())) + ".pcm";File file = new File(Environment.getExternalStorageDirectory(), "/ACC音频/");if (!file.exists()) {file.mkdir();}String audioSaveDir = file.getAbsolutePath();Log.e(TAG, audioSaveDir);recordFile = new File(audioSaveDir, fileName);Log.e(TAG, "生成文件" + recordFile);//如果存在,就先删除再创建if (recordFile.exists()) {recordFile.delete();Log.e(TAG, "删除文件");}try {recordFile.createNewFile();Log.e(TAG, "创建文件");} catch (IOException e) {Log.e(TAG, "未能创建");throw new IllegalStateException("未能创建" + recordFile.toString());}if (filePathList.size() == 2) {filePathList.clear();}filePathList.add(recordFile);try {//输出流OutputStream os = new FileOutputStream(recordFile);BufferedOutputStream bos = new BufferedOutputStream(os);DataOutputStream dos = new DataOutputStream(bos);int bufferSize = AudioRecord.getMinBufferSize(sampleRateInHz, AudioFormat.CHANNEL_IN_STEREO, audioEncoding);if (ActivityCompat.checkSelfPermission(weakReference.get(), Manifest.permission.RECORD_AUDIO) != PackageManager.PERMISSION_GRANTED) {
​return;}audioRecord = new AudioRecord(MediaRecorder.AudioSource.MIC, sampleRateInHz, AudioFormat.CHANNEL_IN_STEREO, audioEncoding, bufferSize);
​short[] buffer = new short[bufferSize];audioRecord.startRecording();Log.e(TAG, "开始录音");isRecording = true;while (isRecording) {int bufferReadResult = audioRecord.read(buffer, 0, bufferSize);for (int i = 0; i < bufferReadResult; i++) {dos.writeShort(buffer[i]);}}audioRecord.stop();dos.close();} catch (Exception e) {e.printStackTrace();Log.e(TAG, "录音失败");showToast("录音失败");}}

3.播放音频方法:

public void playPcm(boolean isChecked) {try {if (isChecked) {//两首一起播放for (File recordFiles : filePathList) {threadPoolExecutor.execute(() -> playPcmData(recordFiles));}} else {//只播放最后一次录音playPcmData(recordFile);}}catch (Exception e){e.printStackTrace();}
}

4.播放pcm流边录边播:

/*** 播放Pcm流,边读取边播*/
public void playPcmData(File recordFiles) {Log.e(TAG, "打印线程" + Thread.currentThread().getName());try {DataInputStream dis = new DataInputStream(new BufferedInputStream(new FileInputStream(recordFiles)));//最小缓存区int bufferSizeInBytes = AudioTrack.getMinBufferSize(sampleRateInHz, AudioFormat.CHANNEL_OUT_STEREO, audioEncoding);AudioTrack player = new AudioTrack(AudioManager.STREAM_MUSIC, sampleRateInHz, AudioFormat.CHANNEL_OUT_STEREO, audioEncoding, bufferSizeInBytes, AudioTrack.MODE_STREAM);short[] data = new short[bufferSizeInBytes];//开始播放player.play();while (true) {int i = 0;while (dis.available() > 0 && i < data.length) {data[i] = dis.readShort();i++;}player.write(data, 0, data.length);//表示读取完了if (i != bufferSizeInBytes) {player.stop();//停止播放player.release();//释放资源dis.close();showToast("播放完成了!!!");break;}}} catch (Exception e) {Log.e(TAG, "播放异常: " + e.getMessage());showToast("播放异常!!!!");e.printStackTrace();}
}

5.播放所有音频方法:

    public void playAllRecord() {if (recordFile == null) {return;}//读取文件int musicLength = (int) (recordFile.length() / 2);short[] music = new short[musicLength];try {InputStream is = new FileInputStream(recordFile);BufferedInputStream bis = new BufferedInputStream(is);DataInputStream dis = new DataInputStream(bis);int i = 0;while (dis.available() > 0) {music[i] = dis.readShort();i++;}dis.close();AudioTrack audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, sampleRateInHz, channelConfiguration, audioEncoding, musicLength * 2, AudioTrack.MODE_STREAM);audioTrack.play();audioTrack.write(music, 0, musicLength);audioTrack.stop();} catch (Throwable t) {Log.e(TAG, "播放失败");showToast("播放失败");}}

如需完整版 Android 音视频从入门到高级进阶学习笔记 请点击此处免费领取

6.设置是否录音方法:

public void setRecord(boolean isRecording) {this.isRecording = isRecording;
}

7.暂停播放音频方法:

    public void pauseAudio(){if(audioRecord != null ){audioRecord.stop();}}

8.回收资源和播放器方法:

    public void releaseAudio(){if(audioRecord != null){audioRecord.release();}if(handler != null){handler.removeCallbacksAndMessages(null);}if(threadPoolExecutor != null){threadPoolExecutor.shutdown();}}

9.音频播放、文件读写权限申请:

    private void afterPermissions() {// Marshmallow开始才用申请运行时权限if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {for (int i = 0; i < permissions.length; i++) {if (ContextCompat.checkSelfPermission(this, permissions[i]) !=PackageManager.PERMISSION_GRANTED) {mPermissionList.add(permissions[i]);}}if (!mPermissionList.isEmpty()) {String[] permissions = mPermissionList.toArray(new String[mPermissionList.size()]);ActivityCompat.requestPermissions(this, permissions, MY_PERMISSIONS_REQUEST);}}}

10.调用开始录音:

       btnRecord.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {buttonEnabled(false, true, true);Toast.makeText(MainActivity.this, "开始录音", Toast.LENGTH_SHORT).show();threadPoolExecutor.execute(() -> {RecorderAudioManagerUtils.getInstance().startRecord(new WeakReference<>(getApplicationContext()));});}});

11.调用播放音频:

       btnPlay.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {RecorderAudioManagerUtils.getInstance().playPcm(true);buttonEnabled(false, false, true);}});

12.调用停止录音:

        btnStop.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {new Handler().post(new Runnable() {@Overridepublic void run() {buttonEnabled(true, true, false);Toast.makeText(MainActivity.this, "停止录音", Toast.LENGTH_SHORT).show();RecorderAudioManagerUtils.getInstance().pauseAudio();}});}});

13.设置录音、播放、停止按钮状态:

private void buttonEnabled(boolean record, boolean play, boolean stop) {btnRecord.setEnabled(record);btnPlay.setEnabled(play);btnStop.setEnabled(stop);
}

14.布局文件代码如下:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".MainActivity"><Buttonandroid:id="@+id/btn_record"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="录制音频"app:layout_constraintBottom_toBottomOf="parent"app:layout_constraintEnd_toStartOf="@id/btn_play"app:layout_constraintStart_toStartOf="parent"app:layout_constraintTop_toTopOf="parent" /><Buttonandroid:id="@+id/btn_play"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="播放音频"app:layout_constraintBottom_toBottomOf="parent"app:layout_constraintEnd_toStartOf="@id/btn_stop"app:layout_constraintStart_toEndOf="@id/btn_record"app:layout_constraintTop_toTopOf="parent" /><Buttonandroid:id="@+id/btn_stop"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="停止录制"app:layout_constraintBottom_toBottomOf="parent"app:layout_constraintEnd_toEndOf="parent"app:layout_constraintStart_toEndOf="@id/btn_play"app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

15.布局预览效果如下:

在这里插入图片描述

在这里插入图片描述

16.完整MainActivity代码:

public class MainActivity extends AppCompatActivity {private Button btnRecord,btnPlay,btnStop;ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(3, 5,1, TimeUnit.MINUTES,new LinkedBlockingDeque<>(10),Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());private String[] permissions = new String[]{android.Manifest.permission.RECORD_AUDIO,Manifest.permission.WRITE_EXTERNAL_STORAGE};/*** 被用户拒绝的权限列表*/private List<String> mPermissionList = new ArrayList<>();private static final int MY_PERMISSIONS_REQUEST = 1001;private final String TAG = MainActivity.this.getClass().getSimpleName();@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);afterPermissions();initView();}private void afterPermissions() {// Marshmallow开始才用申请运行时权限if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {for (int i = 0; i < permissions.length; i++) {if (ContextCompat.checkSelfPermission(this, permissions[i]) !=PackageManager.PERMISSION_GRANTED) {mPermissionList.add(permissions[i]);}}if (!mPermissionList.isEmpty()) {String[] permissions = mPermissionList.toArray(new String[mPermissionList.size()]);ActivityCompat.requestPermissions(this, permissions, MY_PERMISSIONS_REQUEST);}}}private void initView() {btnRecord = findViewById(R.id.btn_record);btnPlay = findViewById(R.id.btn_play);btnStop = findViewById(R.id.btn_stop);btnRecord.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {buttonEnabled(false, true, true);Toast.makeText(MainActivity.this, "开始录音", Toast.LENGTH_SHORT).show();threadPoolExecutor.execute(() -> {RecorderAudioManagerUtils.getInstance().startRecord(new WeakReference<>(getApplicationContext()));});}});btnPlay.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {RecorderAudioManagerUtils.getInstance().playPcm(true);buttonEnabled(false, false, true);}});btnStop.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {new Handler().post(new Runnable() {@Overridepublic void run() {buttonEnabled(true, true, false);Toast.makeText(MainActivity.this, "停止录音", Toast.LENGTH_SHORT).show();RecorderAudioManagerUtils.getInstance().pauseAudio();}});}});}private void buttonEnabled(boolean record, boolean play, boolean stop) {btnRecord.setEnabled(record);btnPlay.setEnabled(play);btnStop.setEnabled(stop);}@Overrideprotected void onStop() {super.onStop();RecorderAudioManagerUtils.getInstance().pauseAudio();}@Overrideprotected void onDestroy() {super.onDestroy();RecorderAudioManagerUtils.getInstance().releaseAudio();}@Overridepublic void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {super.onRequestPermissionsResult(requestCode, permissions, grantResults);boolean allGranted = true;for (int results : grantResults) {allGranted &= results == PackageManager.PERMISSION_GRANTED;}if (allGranted) {afterPermissions();} else {Toast.makeText(this, "请打开特殊权限", Toast.LENGTH_LONG).show();}}}

17.工具类完整代码如下:

package com.example.audiorecorddemo.utils;import android.Manifest;
import android.content.Context;
import android.content.pm.PackageManager;
import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioRecord;
import android.media.AudioTrack;
import android.media.MediaRecorder;
import android.os.Environment;
import android.os.Handler;
import android.os.Looper;
import android.text.format.DateFormat;
import android.util.Log;
import android.widget.Toast;import androidx.core.app.ActivityCompat;import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.List;
import java.util.Locale;
import java.util.concurrent.LinkedBlockingDeque;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;/*** @author: njb* @date: 2023/2/25 18:29* @desc:*/
public class RecorderAudioManagerUtils {private static final String TAG = "PlayManagerUtils";private WeakReference<Context> weakReference;private File recordFile;private boolean isRecording;/*** 最多只能存2条记录*/private final List<File> filePathList = new ArrayList<>(2);/*** 16K采集率*/int sampleRateInHz = 16000;/*** 格式*/int channelConfiguration = AudioFormat.CHANNEL_OUT_STEREO;/*** 16Bit*/int audioEncoding = AudioFormat.ENCODING_PCM_16BIT;private static volatile RecorderAudioManagerUtils mInstance;private final Handler handler = new Handler(Looper.getMainLooper());AudioRecord audioRecord;ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(3, 5, 1, TimeUnit.MINUTES, new LinkedBlockingDeque<>(10));public static RecorderAudioManagerUtils getInstance() {if (mInstance == null) {synchronized (RecorderAudioManagerUtils.class) {if (mInstance == null) {mInstance = new RecorderAudioManagerUtils();}}}return mInstance;}public void startRecord(WeakReference<Context> weakReference) {this.weakReference = weakReference;Log.e(TAG, "开始录音");//生成PCM文件String fileName = DateFormat.format("yyyy-MMdd-HHmmss", Calendar.getInstance(Locale.getDefault())) + ".pcm";File file = new File(Environment.getExternalStorageDirectory(), "/ACC音频/");if (!file.exists()) {file.mkdir();}String audioSaveDir = file.getAbsolutePath();Log.e(TAG, audioSaveDir);recordFile = new File(audioSaveDir, fileName);Log.e(TAG, "生成文件" + recordFile);//如果存在,就先删除再创建if (recordFile.exists()) {recordFile.delete();Log.e(TAG, "删除文件");}try {recordFile.createNewFile();Log.e(TAG, "创建文件");} catch (IOException e) {Log.e(TAG, "未能创建");throw new IllegalStateException("未能创建" + recordFile.toString());}if (filePathList.size() == 2) {filePathList.clear();}filePathList.add(recordFile);try {//输出流OutputStream os = new FileOutputStream(recordFile);BufferedOutputStream bos = new BufferedOutputStream(os);DataOutputStream dos = new DataOutputStream(bos);int bufferSize = AudioRecord.getMinBufferSize(sampleRateInHz, AudioFormat.CHANNEL_IN_STEREO, audioEncoding);if (ActivityCompat.checkSelfPermission(weakReference.get(), Manifest.permission.RECORD_AUDIO) != PackageManager.PERMISSION_GRANTED) {return;}audioRecord = new AudioRecord(MediaRecorder.AudioSource.MIC, sampleRateInHz, AudioFormat.CHANNEL_IN_STEREO, audioEncoding, bufferSize);short[] buffer = new short[bufferSize];audioRecord.startRecording();Log.e(TAG, "开始录音");isRecording = true;while (isRecording) {int bufferReadResult = audioRecord.read(buffer, 0, bufferSize);for (int i = 0; i < bufferReadResult; i++) {dos.writeShort(buffer[i]);}}audioRecord.stop();dos.close();} catch (Exception e) {e.printStackTrace();Log.e(TAG, "录音失败");showToast("录音失败");}}/*** 播放pcm流的方法,一次性读取所有Pcm流,读完后在开始播放*/public void playAllRecord() {if (recordFile == null) {return;}//读取文件int musicLength = (int) (recordFile.length() / 2);short[] music = new short[musicLength];try {InputStream is = new FileInputStream(recordFile);BufferedInputStream bis = new BufferedInputStream(is);DataInputStream dis = new DataInputStream(bis);int i = 0;while (dis.available() > 0) {music[i] = dis.readShort();i++;}dis.close();AudioTrack audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, sampleRateInHz, channelConfiguration, audioEncoding, musicLength * 2, AudioTrack.MODE_STREAM);audioTrack.play();audioTrack.write(music, 0, musicLength);audioTrack.stop();} catch (Throwable t) {Log.e(TAG, "播放失败");showToast("播放失败");}}public void playPcm(boolean isChecked) {try {if (isChecked) {//两首一起播放for (File recordFiles : filePathList) {threadPoolExecutor.execute(() -> playPcmData(recordFiles));}} else {//只播放最后一次录音playPcmData(recordFile);}}catch (Exception e){e.printStackTrace();}}/*** 播放Pcm流,边读取边播*/public void playPcmData(File recordFiles) {Log.e(TAG, "打印线程" + Thread.currentThread().getName());try {DataInputStream dis = new DataInputStream(new BufferedInputStream(new FileInputStream(recordFiles)));//最小缓存区int bufferSizeInBytes = AudioTrack.getMinBufferSize(sampleRateInHz, AudioFormat.CHANNEL_OUT_STEREO, audioEncoding);AudioTrack player = new AudioTrack(AudioManager.STREAM_MUSIC, sampleRateInHz, AudioFormat.CHANNEL_OUT_STEREO, audioEncoding, bufferSizeInBytes, AudioTrack.MODE_STREAM);short[] data = new short[bufferSizeInBytes];//开始播放player.play();while (true) {int i = 0;while (dis.available() > 0 && i < data.length) {data[i] = dis.readShort();i++;}player.write(data, 0, data.length);//表示读取完了if (i != bufferSizeInBytes) {player.stop();//停止播放player.release();//释放资源dis.close();showToast("播放完成了!!!");break;}}} catch (Exception e) {Log.e(TAG, "播放异常: " + e.getMessage());showToast("播放异常!!!!");e.printStackTrace();}}public File getRecordFile() {return recordFile;}public void setRecord(boolean isRecording) {this.isRecording = isRecording;}public void pauseAudio(){if(audioRecord != null ){audioRecord.stop();}}public void releaseAudio(){if(audioRecord != null){audioRecord.release();}if(handler != null){handler.removeCallbacksAndMessages(null);}if(threadPoolExecutor != null){threadPoolExecutor.shutdown();}}private void showToast(String msg) {if(weakReference.get() != null){handler.post(() -> Toast.makeText(weakReference.get(), msg, Toast.LENGTH_LONG).show());}}}

相关文章:

Android音视频开发之音频录制和播放

1.封装音频录制工具类&#xff1a; public class RecorderAudioManagerUtils {private static volatile RecorderAudioManagerUtils mInstance;public static RecorderAudioManagerUtils getInstance() {if (mInstance null) {synchronized (RecorderAudioManagerUtils.class…...

Java之单例模式

目录 一.上节内容 1.什么是线程安全 2.线程不安全的原因 3.JMM(Java内存模型) 4.synchronized锁 5.锁对象 6.volatile关键字 7.wait()和notify() 8.Java中线程安全的类 二.单例模式 1.什么是单例 2.怎么设计一个单例 1.口头约定 2.使用编程语言的特性 三.饿汉模式…...

【分组码系列】线性分组码的网格图和维特比译码

线性分组码的网格图 由于码字的比特位是统计独立的,所以编码过程可以利用有限状态机来描述,它能精确地确定初始和最终状态。可以利用网格图进一步描述编码过程[36],采用维特比算法进行最大似然译码. 在GF(2)上定义线性分组码(n,k)。相应的(n-k)Xn维校验阵可以写成 令码字为系…...

代码命名规范是真优雅呀!代码如诗

日常编码中&#xff0c;代码的命名是个大的学问。能快速的看懂开源软件的代码结构和意图&#xff0c;也是一项必备的能力。那它们有什么规律呢&#xff1f; Java项目的代码结构&#xff0c;能够体现它的设计理念。Java采用长命名的方式来规范类的命名&#xff0c;能够自己表达…...

你不知道的自动化?使用自动化测试在项目中创造高业务价值...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 脱离数据支撑谈价…...

通过实现一个简单的 JavaScript 猜数字大小的游戏,介绍如何进行布局样式处理

JavaScript 猜数字大小是一个非常简单、却又经典的游戏&#xff0c;可以锻炼玩家的逻辑思维能力。在这个游戏中&#xff0c;电脑会随机生成一个数字&#xff0c;玩家需要根据提示逐步猜出正确的数字。接下来&#xff0c;我们将通过实现一个简单的 JavaScript 猜数字大小游戏来介…...

Java设计模式(二十二)策略模式

一、概述 策略模式是一种行为型设计模式&#xff0c;它允许在运行时选择算法的行为。策略模式通过将算法封装成独立的策略类&#xff0c;使得它们可以相互替换&#xff0c;而不影响使用算法的客户端。这样可以使客户端代码与具体算法的实现细节解耦&#xff0c;提高了代码的可…...

【沐风老师】一步一步教你在3dMax中进行UVW贴图和展开UVW的方法

将简单或程序材质应用于对象并不难。但是当表面需要在其上显示某种纹理时&#xff0c;它会变得更加复杂。任何纹理贴图都放在材质的 Diffuse 插槽中&#xff0c;但渲染的结果可能无法预测。这就是为什么我们需要了解 3DMAX 如何将纹理应用于 3D 对象&#xff0c;什么是 UVW 贴图…...

Redis主从复制(搭建集群的一种方式)【故障转移,内存,回收】

做一个伪集群 配置文件&#xff1a; daemonize yes port 7777 logfile .redis-7777.log dir ./ bind 0.0.0.0启动6666 and 7777 现在设置主从表 但是有个问题我把服务器停掉 关系就会解除 还可以手动解除 slaveof no one 命令 配置Sentinel&#xff08;哨兵&#…...

专业专注,极致体验,高端隐形智能晾衣机品牌邦先生官宣浙江卫视知名主持人沈涛为品牌代言人

5月11日&#xff0c;高端隐形晾衣架领导品牌邦先生正式宣布&#xff0c;浙江卫视知名主持人沈涛为品牌代言人&#xff0c;以更高标准的晾晒&#xff0c;共同迎接智能晾晒大时代&#xff0c;用科技力量创造美好智慧家居生活。 专业实力品牌邦先生王牌主持沈涛 作为浙江卫视的“王…...

SpringCloud使用SkyWalking实现分布式链路追踪1

文章目录 一、MicrometerTracingBrave(Sleuth)链路追踪1、MicrometerTracingBrave和Zipkin的概论2、Docker搭建Zipkin服务3、MicrometerTracingBrave和Zipkin实现链路追踪 二、SkyWaking服务的安装与使用1、SkyWalking的概论2、Java探针的环境搭建3、Java探针实现日志监控4、Sk…...

【牛客刷题专栏】0x28:JZ30 包含min函数的栈(C语言编程题)

前言 个人推荐在牛客网刷题(点击可以跳转)&#xff0c;它登陆后会保存刷题记录进度&#xff0c;重新登录时写过的题目代码不会丢失。个人刷题练习系列专栏&#xff1a;个人CSDN牛客刷题专栏。 题目来自&#xff1a;牛客/题库 / 在线编程 / 剑指offer&#xff1a; 目录 前言问…...

聚焦丨酷雷曼荣列XRMA联盟成员单位

自“元宇宙”概念兴起之初&#xff0c;酷雷曼VR所属北京同创蓝天云科技有限公司就积极布局、探索和实践。2022年12月&#xff0c;酷雷曼VR成功加入虚拟现实与元宇宙产业联盟&#xff08;XRMA&#xff09;&#xff0c;正式被接纳为联盟成员单位&#xff0c;意味着酷雷曼公司将进…...

物联网架构和技术:如何实现物物互联和智能化控制

第一章&#xff1a;引言 物联网是一种新兴的技术领域&#xff0c;通过将物理设备、传感器和软件等连接起来&#xff0c;可以实现设备之间的互联互通&#xff0c;让各种设备可以进行数据交换和智能化控制。在这个数字化时代&#xff0c;物联网已经成为了连接万物的关键技术之一…...

Linux系统查看CPU信息命令cat /proc/cpuinfo详细说明

Linux操作系统服务器如何查看CPU处理器信息&#xff1f;使用命令cat /proc/cpuinfo可以查看CPU详细信息&#xff0c;包括CPU核数、逻辑CPU、物理CPU个数、CPU是否启用超线程等&#xff0c;阿里云服务器网分享Linux服务器查看CPU信息命令&#xff1a; 目录 Linux服务器查看CPU…...

RK3588旗舰32T人工智能多网口边缘智能网关交换机

32T边缘智能网关发布&#xff0c;助力多行业数字化升级&#xff0c;运维降本增效&#xff0c;搭载RK3588旗舰芯 搭载瑞芯微RK3588芯片的边缘智能网关XM-RK3588&#xff0c;算力可扩展至32T&#xff0c;适用于电力能源、智慧交通、智慧城市、智慧安防、智慧医疗、工业互联网等领…...

一行代码绘制高分SCI火山图

一、概述 在近半年中&#xff0c;我读了很多的高分SCI文章&#xff0c;很多文章中都有多种不同的火山图&#xff0c;包括「普通的火山图、渐变火山图、以及包含GO通路信息的火山图」&#xff01; 经过一段时间的文献阅读和资料查询&#xff0c;终于找到了一个好用而且简单的包…...

chmod是什么?cron是什么?

chmod 和 cron 是 Unix 和类 Unix 系统&#xff08;如 Linux&#xff09;的常用命令。 chmod&#xff1a;这是一个命令行工具&#xff0c;用于更改文件或目录的权限。在 Unix 和类 Unix 系统中&#xff0c;每个文件和目录都有一个访问权限集&#xff0c;该集定义了哪些用户可以…...

励志长篇小说《周兴和》书连载之三 十五岁时做父亲

十五岁时做父亲 周兴和的父亲一天天更衰老了。 他母亲身体也越来越是消瘦。近一两年来&#xff0c;她常常感到全身无力、胸口发堵、心慌气紧、吞咽困难&#xff0c;做什么事都力不从心了。 这时&#xff0c;他母亲不知是心血来潮&#xff0c;还是她预感到了什么&#xff0c;出…...

文件一直处于修改状态 git checkout 无法还原的问题解决方法

问题描述 最近在 RT-Thread 时&#xff0c;使用 Git 回退版本验证问题&#xff0c;后来 git pull 拉取最新代码后&#xff0c;发现里面有几个文件&#xff0c;一直为【修改】状态&#xff0c;并且无法还原&#xff0c;git checkout xxx git reset --hard 都用了&#xff0c;依旧…...

Phi-4-mini-reasoning应用场景:AI编程教练中算法题逻辑拆解与反馈生成

Phi-4-mini-reasoning应用场景&#xff1a;AI编程教练中算法题逻辑拆解与反馈生成 1. 模型介绍 Phi-4-mini-reasoning是一款专注于推理任务的文本生成模型&#xff0c;特别擅长处理需要多步逻辑分析的场景。与通用聊天模型不同&#xff0c;它被设计用来解决数学题、逻辑题等需…...

浏览器插件:让Markdown预览效率提升300%的秘密武器

浏览器插件&#xff1a;让Markdown预览效率提升300%的秘密武器 【免费下载链接】markdown-viewer Markdown Viewer / Browser Extension 项目地址: https://gitcode.com/gh_mirrors/ma/markdown-viewer 作为开发者、学生或技术写作者&#xff0c;你是否经常遇到这些困扰…...

OpenCore Legacy Patcher技术指南:让老旧Mac焕发新生的系统扩展方案

OpenCore Legacy Patcher技术指南&#xff1a;让老旧Mac焕发新生的系统扩展方案 【免费下载链接】OpenCore-Legacy-Patcher Experience macOS just like before 项目地址: https://gitcode.com/GitHub_Trending/op/OpenCore-Legacy-Patcher 当您的Mac设备因苹果官方停止…...

原神帧率解锁终极指南:3步轻松突破60FPS限制,享受极致流畅体验

原神帧率解锁终极指南&#xff1a;3步轻松突破60FPS限制&#xff0c;享受极致流畅体验 【免费下载链接】genshin-fps-unlock unlocks the 60 fps cap 项目地址: https://gitcode.com/gh_mirrors/ge/genshin-fps-unlock 还在为原神60帧限制而苦恼吗&#xff1f;高端显卡却…...

Optick与虚幻引擎集成教程:打造专业级游戏性能分析环境

Optick与虚幻引擎集成教程&#xff1a;打造专业级游戏性能分析环境 【免费下载链接】optick C Profiler For Games 项目地址: https://gitcode.com/gh_mirrors/op/optick 作为游戏开发者&#xff0c;你是否曾经为性能瓶颈而苦恼&#xff1f;想要深入了解游戏运行时的性能…...

Wangle客户端开发实战:从零开始构建高效网络应用

Wangle客户端开发实战&#xff1a;从零开始构建高效网络应用 【免费下载链接】wangle Wangle is a framework providing a set of common client/server abstractions for building services in a consistent, modular, and composable way. 项目地址: https://gitcode.com/g…...

Android 11文件权限避坑指南:为什么你的APP无法修改原文件?

Android 11存储权限深度解析&#xff1a;从沙盒机制到实战解决方案 在去年的一次应用升级中&#xff0c;我们团队遇到了一个棘手的问题&#xff1a;用户反馈图片编辑后无法保存到原位置。经过排查&#xff0c;发现这是Android 11引入的存储权限机制变化导致的。作为开发者&…...

【DexGraspNet与多指手抓取算法详解】第六章 运动规划与轨迹优化

目录 第六章 运动规划与轨迹优化 6.1 从静态姿态到动态轨迹 6.1.1 抓取前运动规划 6.1.1.1 快速扩展随机树 (RRT) 6.1.1.1.1 状态空间采样 6.1.1.1.2 碰撞检测机制 6.1.1.2 轨迹平滑处理 6.1.1.2.1 B样条插值 6.1.1.2.2 速度与加速度约束 6.2 基于优化的轨迹生成 6.…...

springboot+vue基于web的蛋糕商城论坛交流系统的设计系统

目录同行可拿货,招校园代理 ,本人源头供货商系统功能模块分析核心功能模块特色功能实现技术难点解决方案性能优化措施项目技术支持源码获取详细视频演示 &#xff1a;文章底部获取博主联系方式&#xff01;同行可合作同行可拿货,招校园代理 ,本人源头供货商 系统功能模块分析 …...

LPDDR4X引脚功能详解:从CK到DQS,这些信号线你都用对了吗?

LPDDR4X引脚功能深度解析&#xff1a;信号完整性设计与实战避坑指南 在移动设备和高性能嵌入式系统中&#xff0c;LPDDR4X内存已成为主流选择。但许多硬件工程师在实际设计中常陷入"信号连通即可"的误区&#xff0c;导致系统稳定性问题频发。本文将带您深入理解每个…...