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

在 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)

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

相关文章:

  • Flask API 项目 Swagger 版本打架不兼容
  • 02_Flask是什么?
  • Hadoop+Spark 笔记 2025/4/21
  • 【python实用小脚本系列】用Python让鼠标“动起来”:光标自动移动、自动点击、自动图象识别的小技巧
  • CSS 中实现 div 居中有以下几种常用方法
  • ADB -> pull指令推送电脑文件到手机上
  • 大数据学习(109)-Impala 和 Hive 之间的 SQL 差异
  • 机器学习 Day13 Boosting集成学习方法: Adaboosting和GBDT
  • Spark-SQL连接Hive总结及实验
  • 使用SystemWeaver生成SOME/IP ETS ARXML的完整实战指南
  • 23种设计模式-结构型模式之外观模式(Java版本)
  • C++std::map
  • 批量替换多个 Word 文档中的指定图片
  • Spark SQL核心解析:大数据时代的结构化处理利器
  • 奇怪的问题
  • 【大数据分析】Apache Doris高性能实时分析数据库:MPP架构下的多场景应用与优势分析以及部署应用
  • 第十四届蓝桥杯 2023 C/C++组 飞机降落
  • 外网如何连接内网中的mysql数据库服务器
  • 云原生--基础篇-2--云计算概述(云计算是云原生的基础,IaaS、PaaS和SaaS服务模型)
  • 微信小程序 == 倒计时验证码组件 (countdown-verify)
  • 美方因涉港问题对中国官员滥施非法单边制裁,外交部:强烈谴责,对等反制
  • 中方决定对在涉港问题上表现恶劣的美国国会议员等实施制裁
  • 深一度|奥运一年后丢冠不稀奇,但究竟谁来扛起男乒的大旗
  • 上海自然博物馆下月开启中国恐龙大展,还在筹备中国古人类大展
  • 丁薛祥:坚定发展信心,强化创新驱动,推动高质量发展行稳致远
  • 国家统计局:一季度GDP增速在全球主要经济体中名列前茅