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

在 Android 中实现通话录音

在 Android 中实现通话录音需要处理系统权限通话状态监听音频录制等关键步骤。以下是详细实现代码及注释,注意不同 Android 版本和厂商设备的兼容性问题


1. 添加权限声明(AndroidManifest.xml)

<!-- 录制音频权限 -->
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<!-- 读取通话状态权限 -->
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<!-- Android 10+ 需要前台服务权限 -->
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<!-- 存储录音文件权限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" android:maxSdkVersion="28" /> <!-- Android 9及以下需要 -->

2. 创建通话状态监听服务(Java 代码)

import android.app.Service;
import android.content.Intent;
import android.media.MediaRecorder;
import android.os.IBinder;
import android.telephony.PhoneStateListener;
import android.telephony.TelephonyManager;
import android.util.Log;
import java.io.IOException;public class CallRecordingService extends Service {private static final String TAG = "CallRecording";private MediaRecorder mediaRecorder;private TelephonyManager telephonyManager;private PhoneStateListener phoneStateListener;@Overridepublic void onCreate() {super.onCreate();// 初始化电话状态监听telephonyManager = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);phoneStateListener = new PhoneStateListener() {@Overridepublic void onCallStateChanged(int state, String phoneNumber) {switch (state) {case TelephonyManager.CALL_STATE_OFFHOOK: // 通话开始startRecording(phoneNumber);break;case TelephonyManager.CALL_STATE_IDLE:     // 通话结束stopRecording();break;case TelephonyManager.CALL_STATE_RINGING:  // 来电响铃// 可在此处处理来电号码break;}}};// 注册监听器telephonyManager.listen(phoneStateListener, PhoneStateListener.LISTEN_CALL_STATE);}/*** 开始录音*/private void startRecording(String phoneNumber) {try {mediaRecorder = new MediaRecorder();// 设置音频源(不同设备可能支持不同配置)mediaRecorder.setAudioSource(MediaRecorder.AudioSource.VOICE_CALL); // 需要系统权限// 部分设备需改用 MIC(但无法录制对方声音):// mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);// 输出格式mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);// 音频编码mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);// 保存路径(示例路径,需适配 Android 10+ 作用域存储)String filePath = getExternalFilesDir(null) + "/call_recording_" + System.currentTimeMillis() + ".3gp";mediaRecorder.setOutputFile(filePath);mediaRecorder.prepare();mediaRecorder.start();Log.d(TAG, "开始录音: " + filePath);} catch (IOException | IllegalStateException e) {Log.e(TAG, "录音失败: " + e.getMessage());}}/*** 停止录音*/private void stopRecording() {if (mediaRecorder != null) {try {mediaRecorder.stop();mediaRecorder.reset();mediaRecorder.release();mediaRecorder = null;Log.d(TAG, "录音已停止");} catch (RuntimeException e) {Log.e(TAG, "停止录音异常: " + e.getMessage());}}}@Overridepublic void onDestroy() {super.onDestroy();// 注销监听器telephonyManager.listen(phoneStateListener, PhoneStateListener.LISTEN_NONE);stopRecording();}@Overridepublic IBinder onBind(Intent intent) {return null;}
}

3. 启动服务并检查权限(Activity 中调用)

import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Bundle;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;public class MainActivity extends AppCompatActivity {private static final int PERMISSION_REQUEST_CODE = 100;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// 检查并请求权限if (checkPermissions()) {startCallRecordingService();}}private boolean checkPermissions() {String[] requiredPermissions = {Manifest.permission.RECORD_AUDIO,Manifest.permission.READ_PHONE_STATE,Manifest.permission.WRITE_EXTERNAL_STORAGE};for (String permission : requiredPermissions) {if (ActivityCompat.checkSelfPermission(this, permission) != PackageManager.PERMISSION_GRANTED) {ActivityCompat.requestPermissions(this, requiredPermissions, PERMISSION_REQUEST_CODE);return false;}}return true;}private void startCallRecordingService() {Intent serviceIntent = new Intent(this, CallRecordingService.class);// Android 8+ 需使用前台服务if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {startForegroundService(serviceIntent);} else {startService(serviceIntent);}}@Overridepublic void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {super.onRequestPermissionsResult(requestCode, permissions, grantResults);if (requestCode == PERMISSION_REQUEST_CODE) {if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {startCallRecordingService();}}}
}

关键问题与注意事项

  1. 音频源限制

    • VOICE_CALL 音频源需要系统级权限,普通应用无法使用。

    • 大部分设备使用 MIC 仅能录制本地麦克风声音(无法录制对方声音)。

  2. 厂商兼容性

    • 华为、小米等厂商可能屏蔽通话录音功能,需申请白名单或使用特殊 API。

  3. Android 版本适配

    • Android 9+ 禁止后台应用访问麦克风,需结合前台服务(Notification)实现。

  4. 法律与隐私

    • 通话录音需明确告知用户并取得同意(部分国家/地区要求双重确认)。


替代方案(推荐)

如果无法直接录制通话音频,可通过以下方式实现:

  1. 使用系统提供的通话录音接口(如小米的 MiuiTelephony

  2. Root 设备后获取系统权限

  3. 接入第三方通话录音 SDK(如 Twilio、Agora)

建议根据实际需求选择合规方案。

http://www.xdnf.cn/news/817.html

相关文章:

  • Discuz!与DeepSeek结合:打造智能论坛,提升用户体验与运营效率
  • 华为认证是什么?
  • 【C++软件实战问题排查经验分享】UI界面卡顿 | CPU占用高 | GDI对象泄漏 | 线程堵塞 系列问题排查总结
  • [预备知识]2. PyTorch基本操作
  • [Qt]双击事件导致的问题
  • Kafka 如何理解Kafka的高可用
  • AI对话高效输入指令攻略(三):使用大忌——“AI味”
  • IDEA打不开、打开报错
  • idea远程debug调试
  • eBay自动化定价陷阱调查:价格战背后的利润黑洞与破局之道
  • redis队列 和 秒杀应用
  • 音视频学习(三十五):aud
  • 什么是AI智能音视频?小天互连即时通讯带您体验
  • ECharts散点图-散点图13,附视频讲解与代码下载
  • 多级缓存架构,让系统更快的跑起来!
  • P5670 秘籍-反复异或 Solution
  • 微软Edge浏览器字体设置
  • Matlab自学笔记五十一:(推荐)输入参数的数量和可变数量的输入
  • 前端笔记-JavaScript(下)
  • Python多进程同步全解析:从竞争条件到锁、信号量的实战应用
  • 基于javaweb的SpringBoot爱心捐赠平台系统设计与实现(源码+文档+部署讲解)
  • 免费图片软件,可矫正倾斜、调整去底效果
  • Deepseek生成图片,支持手机和PC
  • 【KWDB 创作者计划】_上位机知识篇---Arduino
  • Sharding-JDBC 系列专题 - 第二篇:广播表(Broadcast Tables)
  • docker常用命令
  • C语言究竟是一门怎样的语言
  • PyCharm使用Anaconda 中的虚拟环境
  • firewall指令
  • 存储图表数据的模板类QCPDataContainer