【安卓开发】Android实现录音并播放功能

简介: 【安卓开发】Android实现录音并播放功能

官网文档

添加权限

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.RECORD_AUDIO" />

xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout 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="music.RecordActivity">
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal">
        <Button
            android:layout_width="wrap_content"
            android:layout_height="match_parent"
            android:id="@+id/bt_back"
            android:text="返回"
            ></Button>
        <Button
            android:layout_width="wrap_content"
            android:layout_height="match_parent"
            android:paddingLeft="30dp"
            android:id="@+id/bt_save"
            android:text="确定"
            ></Button>
    </LinearLayout>
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_centerInParent="true"
        android:gravity="center"
        android:orientation="vertical">
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_margin="5dp"
            android:gravity="center"
            android:orientation="horizontal">
            <ImageView
                android:id="@+id/iv_record_wave_left"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="center"
                android:layout_margin="5dp"
                android:background="@xml/record_wave_left" />
            <ImageView
                android:id="@+id/iv_microphone"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="center"
                android:layout_margin="5dp"
                android:src="@drawable/record_microphone_icon" />
            <ImageView
                android:id="@+id/iv_record_wave_right"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="center"
                android:layout_margin="5dp"
                android:background="@xml/record_wave_right" />
        </LinearLayout>
        <TextView
            android:id="@+id/tv_recordTime"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_margin="5dp"
            android:gravity="center"
            android:text="00:00:00"
            android:textColor="#499df7"
            android:textSize="20sp" />
    </LinearLayout>
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="45dp"
        android:background="@drawable/navigationbar_bg"
        android:layout_alignParentBottom="true"
        android:gravity="center"
        >
        <Button
            android:id="@+id/btn_record"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:background="@drawable/tabbar_record_start"
            />
    </LinearLayout>
</RelativeLayout>
package music;
import androidx.annotation.RequiresApi;
import androidx.appcompat.app.AppCompatActivity;
import android.content.Intent;
import android.graphics.drawable.AnimationDrawable;
import android.icu.text.SimpleDateFormat;
import android.media.MediaPlayer;
import android.media.MediaRecorder;
import android.os.Build;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.view.Window;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;
import com.example.myapp_b.R;
import java.io.File;
import java.io.IOException;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
public class RecordActivity extends AppCompatActivity {
    private Button btn_record;
    private ImageView iv_microphone;
    private TextView tv_recordTime;
    private ImageView iv_record_wave_left,iv_record_wave_right;
    private AnimationDrawable ad_left,ad_right;
    private int isRecording = 0;
    private int isPlaying = 0;
    private Timer mTimer;
    //语音操作对象
    private MediaPlayer mPlayer = null;
    private MediaRecorder mRecorder = null;
    //语音保存路径
    private String FilePath = null;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_record);
        Button btn_save = (Button)findViewById(R.id.bt_save);
        btn_save.setOnClickListener((View.OnClickListener) new ClickEvent());
        Button btn_back = (Button)findViewById(R.id.bt_back);
        btn_back.setOnClickListener((View.OnClickListener) new ClickEvent());
        btn_record = (Button)findViewById(R.id.btn_record);
        btn_record.setOnClickListener((View.OnClickListener) new ClickEvent());
        iv_microphone = (ImageView)findViewById(R.id.iv_microphone);
        iv_microphone.setOnClickListener((View.OnClickListener) new ClickEvent());
        iv_record_wave_left = (ImageView)findViewById(R.id.iv_record_wave_left);
        iv_record_wave_right = (ImageView)findViewById(R.id.iv_record_wave_right);
        ad_left = (AnimationDrawable)iv_record_wave_left.getBackground();
        //ad_left = (AnimationDrawable)iv_record_wave_left.getDrawable();
        ad_right = (AnimationDrawable)iv_record_wave_right.getBackground();
        //ad_right = (AnimationDrawable)iv_record_wave_right.getDrawable();
        tv_recordTime = (TextView)findViewById(R.id.tv_recordTime);
    }
    final Handler handler = new Handler(){
        public void handleMessage(Message msg) {
            switch(msg.what){
                case 1 :
                    String time[] = tv_recordTime.getText().toString().split(":");
                    int hour = Integer.parseInt(time[0]);
                    int minute = Integer.parseInt(time[1]);
                    int second = Integer.parseInt(time[2]);
                    if(second < 59){
                        second++;
                    }
                    else if(second == 59 && minute < 59){
                        minute++;
                        second = 0;
                    }
                    if(second == 59 && minute == 59 && hour < 98){
                        hour++;
                        minute = 0;
                        second = 0;
                    }
                    time[0] = hour + "";
                    time[1] = minute + "";
                    time[2] = second + "";
                    //调整格式显示到屏幕上
                    if(second < 10)
                        time[2] = "0" + second;
                    if(minute < 10)
                        time[1] = "0" + minute;
                    if(hour < 10)
                        time[0] = "0" + hour;
                    //显示在TextView中
                    tv_recordTime.setText(time[0]+":"+time[1]+":"+time[2]);
                    break;
            }
        }
    };
    class ClickEvent implements View.OnClickListener {
        @RequiresApi(api = Build.VERSION_CODES.N)
        @Override
        public void onClick(View v) {
            switch(v.getId()){
                //点击的是开始录音按钮
                case  R.id.btn_record :
                    //开始录音
                    if(isRecording == 0){
                        //每一次调用录音,可以录音多次,至多满意为至,最后只将最后一次的录音文件保存,其他的删除
                        if(FilePath != null){
                            File oldFile = new File(FilePath);
                            oldFile.delete();
                        }
                        //获得系统当前时间,并以该时间作为文件名
                        SimpleDateFormat formatter   =   new   SimpleDateFormat   ("yyyyMMddHHmmss");
                        Date   curDate   =   new Date(System.currentTimeMillis());//获取当前时间
                        String   str   =   formatter.format(curDate);
                        str = str + "record.amr";
                        File dir = new File("/sdcard/notes/");
                        File file = new File("/sdcard/notes/",str);
                        if (!dir.exists()) {
                            dir.mkdir();
                        }
                        else{
                            if(file.exists()){
                                file.delete();
                            }
                        }
                        FilePath = dir.getPath() +"/"+ str;
                        //计时器
                        mTimer = new Timer();
                        //将麦克图标设置成不可点击,
                        iv_microphone.setClickable(false);
                        //将显示的时间设置为00:00:00
                        tv_recordTime.setText("00:00:00");
                        //将按钮换成停止录音
                        isRecording = 1;
                        btn_record.setBackgroundResource(R.drawable.tabbar_record_stop);
                        mRecorder = new MediaRecorder();
                        mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
                        mRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
                        mRecorder.setOutputFile(FilePath);
                        mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
                        try {
                            mRecorder.prepare();
                        } catch (IllegalStateException e) {
                            e.printStackTrace();
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                        mRecorder.start();
                        mTimer.schedule(new TimerTask() {
                            @Override
                            public void run() {
                                Message message = new Message();
                                message.what = 1;
                                handler.sendMessage(message);
                            }
                        },1000, 1000);
                        //播放动画
                        ad_left.start();
                        ad_right.start();
                    }
                    //停止录音
                    else{
                        //将按钮换成开始录音
                        isRecording = 0;
                        btn_record.setBackgroundResource(R.drawable.tabbar_record_start);
                        mRecorder.stop();
                        mTimer.cancel();
                        mTimer = null;
                        mRecorder.release();
                        mRecorder = null;
                        //将麦克图标设置成可点击,
                        iv_microphone.setClickable(true);
                        //停止动画
                        ad_left.stop();
                        ad_right.stop();
                        Toast.makeText(RecordActivity.this, "单击麦克图标试听,再次点击结束试听", Toast.LENGTH_LONG).show();
                    }
                    break;
                //如果单击的是麦克图标,则可以是进入试听模式,再次点击,停止播放
                case R.id.iv_microphone :
                    if(FilePath == null)
                        Toast.makeText(RecordActivity.this, "没有录音广播可以播放,请先录音", Toast.LENGTH_LONG).show();
                    else{
                        //试听
                        if(isPlaying == 0){
                            isPlaying = 1;
                            mPlayer = new MediaPlayer();
                            tv_recordTime.setText("00:00:00");
                            mTimer = new Timer();
                            mPlayer.setOnCompletionListener((MediaPlayer.OnCompletionListener) new MediaCompletion());
                            try {
                                mPlayer.setDataSource(FilePath);
                                mPlayer.prepare();
                                mPlayer.start();
                            } catch (IllegalArgumentException e) {
                                // TODO Auto-generated catch block
                                e.printStackTrace();
                            } catch (SecurityException e) {
                                // TODO Auto-generated catch block
                                e.printStackTrace();
                            } catch (IllegalStateException e) {
                                // TODO Auto-generated catch block
                                e.printStackTrace();
                            } catch (IOException e) {
                                // TODO Auto-generated catch block
                                e.printStackTrace();
                            }
                            mTimer.schedule(new TimerTask() {
                                @Override
                                public void run() {
                                    Message message = new Message();
                                    message.what = 1;
                                    handler.sendMessage(message);
                                }
                            }, 1000,1000);
                            //播放动画
                            ad_left.start();
                            ad_right.start();
                        }
                        //结束试听
                        else{
                            isPlaying = 0;
                            mPlayer.stop();
                            mPlayer.release();
                            mPlayer = null;
                            mTimer.cancel();
                            mTimer = null;
                            //停止动画
                            ad_left.stop();
                            ad_right.stop();
                        }
                    }
                    break;
                //点击确定按钮
                case R.id.bt_save :
                    //将最终的录音文件的路径返回
                    Intent intent = getIntent();
                    Bundle b = new Bundle();
                    b.putString("audio", FilePath);
                    System.out.println(FilePath);
                    intent.putExtras(b);
                    setResult(RESULT_OK, intent);
                   // RecordActivity.this.finish();
                    break;
                case R.id.bt_back :
                    //返回前将录音的文件删除
                    if(FilePath != null){
                        File oldFile = new File(FilePath);
                        oldFile.delete();
                    }
                  //  ActivityRecord.this.finish();
                    break;
            }
        }
    }
    class MediaCompletion implements MediaPlayer.OnCompletionListener {
        @Override
        public void onCompletion(MediaPlayer mp) {
            mTimer.cancel();
            mTimer = null;
            isPlaying = 0;
            //停止动画
            ad_left.stop();
            ad_right.stop();
            Toast.makeText(RecordActivity.this, "播放完毕", Toast.LENGTH_LONG).show();
            tv_recordTime.setText("00:00:00");
        }
    }
}
目录
相关文章
|
6天前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
11天前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
13天前
|
Android开发 Swift iOS开发
探索安卓与iOS开发的差异和挑战
【10月更文挑战第37天】在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统扮演着主角。它们各自拥有独特的特性、优势以及面临的开发挑战。本文将深入探讨这两个平台在开发过程中的主要差异,从编程语言到用户界面设计,再到市场分布的不同影响,旨在为开发者提供一个全面的视角,帮助他们更好地理解并应对在不同平台上进行应用开发时可能遇到的难题和机遇。
|
15天前
|
XML 存储 Java
探索安卓开发之旅:从新手到专家
【10月更文挑战第35天】在数字化时代,安卓应用的开发成为了一个热门话题。本文旨在通过浅显易懂的语言,带领初学者了解安卓开发的基础知识,同时为有一定经验的开发者提供进阶技巧。我们将一起探讨如何从零开始构建第一个安卓应用,并逐步深入到性能优化和高级功能的实现。无论你是编程新手还是希望提升技能的开发者,这篇文章都将为你提供有价值的指导和灵感。
|
13天前
|
存储 API 开发工具
探索安卓开发:从基础到进阶
【10月更文挑战第37天】在这篇文章中,我们将一起探索安卓开发的奥秘。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的信息和建议。我们将从安卓开发的基础开始,逐步深入到更复杂的主题,如自定义组件、性能优化等。最后,我们将通过一个代码示例来展示如何实现一个简单的安卓应用。让我们一起开始吧!
|
14天前
|
存储 XML JSON
探索安卓开发:从新手到专家的旅程
【10月更文挑战第36天】在这篇文章中,我们将一起踏上一段激动人心的旅程,从零基础开始,逐步深入安卓开发的奥秘。无论你是编程新手,还是希望扩展技能的老手,这里都有适合你的知识宝藏等待发掘。通过实际的代码示例和深入浅出的解释,我们将解锁安卓开发的关键技能,让你能够构建自己的应用程序,甚至贡献于开源社区。准备好了吗?让我们开始吧!
25 2
|
15天前
|
Android开发
布谷语音软件开发:android端语音软件搭建开发教程
语音软件搭建android端语音软件开发教程!
|
20天前
|
JSON Java Android开发
探索安卓开发之旅:打造你的第一个天气应用
【10月更文挑战第30天】在这个数字时代,掌握移动应用开发技能无疑是进入IT行业的敲门砖。本文将引导你开启安卓开发的奇妙之旅,通过构建一个简易的天气应用来实践你的编程技能。无论你是初学者还是有一定经验的开发者,这篇文章都将成为你宝贵的学习资源。我们将一步步地深入到安卓开发的世界中,从搭建开发环境到实现核心功能,每个环节都充满了发现和创造的乐趣。让我们开始吧,一起在代码的海洋中航行!
|
21天前
|
缓存 数据库 Android开发
安卓开发中的性能优化技巧
【10月更文挑战第29天】在移动应用的海洋中,性能是船只能否破浪前行的关键。本文将深入探讨安卓开发中的性能优化策略,从代码层面到系统层面,揭示如何让应用运行得更快、更流畅。我们将以实际案例和最佳实践为灯塔,引领开发者避开性能瓶颈的暗礁。
43 3
|
18天前
|
移动开发 Java Android开发
探索Android与iOS开发的差异性与互联性
【10月更文挑战第32天】在移动开发的大潮中,Android和iOS两大平台各领风骚。本文将深入浅出地探讨这两个平台的开发差异,并通过实际代码示例,展示如何在各自平台上实现相似的功能。我们将从开发环境、编程语言、用户界面设计、性能优化等多个角度进行对比分析,旨在为开发者提供跨平台开发的实用指南。
38 0
下一篇
无影云桌面