本文转载自:http://blog.csdn.net/u014737138/article/details/49738827

这个问题在网上看到了太多的答案,一直提示说按照官网的api的顺序来,其实解决问题的方法不是这样的,那样没法解决问题,照着那个顺序来也米有用

 

我们得知道为什么它停止不了,为什么停止闪退了,

 

这里面有个结论就是:闪退必然是出现了控制值的错误,在Java中就是java.lang.NullXXException的错误

 

好像没有其他的原因导致闪退把,

 

stop 就删除,是因为stop的对象不存在,这个懂把,

stop对象不存在,说明对象new失败,这个懂吧,

stop的对象new失败,那就需要看new执行了哪些操作了:

  1. <strong><span style="font-size:18px;">mr=new MediaRecorder();
  2. mr.setAudioSource(MediaRecorder.AudioSource.MIC);
  3. mr.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
  4. mr.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
  5. mr.setMaxDuration(10000);
  6. mr.setOutputFile(Environment.getExternalStorageDirectory().getPath()+File.separator+"fisii77s.3gp");
  7. mr.prepare();
  8. mr.start();</span></strong>


我相信大家都是这样的做法,关键是我们前面包了一层if(mr==null)的判断了

我在代码中报错之前是这样写的:

  1. <strong><span style="font-size:18px;">  public void start(Context context, String name) {
  2. if (!Environment.getExternalStorageState().equals(
  3. android.os.Environment.MEDIA_MOUNTED)) {
  4. return;
  5. }
  6. if (mRecorder == null) {
  7. mRecorder = new MediaRecorder();
  8. mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
  9. mRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
  10. mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
  11. mRecorder.setOutputFile(UtilsForChat.getMusicFilePath(context, name));
  12. try {
  13. mRecorder.prepare();
  14. mRecorder.start();
  15. mEMA = 0.0;
  16. } catch (IllegalStateException e) {
  17. System.out.print(e.getMessage());
  18. } catch (IOException e) {
  19. System.out.print(e.getMessage());
  20. }
  21. }
  22. }</span></strong>

也就是说如果对象存在,没有被销毁掉,我就不用创建了,按照常理说,确实应该这样判断对吧,可是常理一般都是错误的,

在java中 判断 一个对象是否为空确实是 == null  关键是 ==null是有谁来判断的  这个好像是说面试的时候经常这样考   我这里还是讲讲我一直那么理解的东西啊,本人还在读书,希望对以后找工作面试的时候有用 哈哈

== 这个判断是地址值比较的,equals是内容值比较的

== 这个地址值是指堆内存的,equals这个值是指在栈内存中的

 

在我们知道null判断的机制的了之后,我们就要去考虑,如果这个对象还存在,我们是否能直接用这个对象呢,经过测试答案是不行的,

 

接下来再接着说,如果我们第一次创建一个MediaRecorder对象,当我们录音结束之后肯定是停止并且释放了的,否则录音文件就不成功了,

停止,停止的是jni对象,释放,释放的jni里面的对象,同时也释放了java对象里面的栈内存的值,堆内存还保留着呢,引用为空   这里搞明白了,原因也就出来了,

关于这里面jni机制如果不懂,可以看我上篇文章,讲解的很详细

释放之后,如果马上执行第二次录音,这个时候上面的if代码就需要判断了,很明显if里面的语句是不会被执行的,因为堆内存没有被释放掉啊,但是jni里面的对象全部被释放掉了

 

这个时候如果你还用这个对象去录音,那么结果就是你对着空气说话,录的音也成了空气,然后你松开手指也就是录音对象停止的时候,无法停止了,因为jni对象是空的,不存在,怎么去停止呢,程序直接闪退,原因就是这样的了

 

解决办法就是这个时候如果你的对象不为空,你需要再重新创建一次,主要是保证你录音不会成为空气,

 

  1. <strong><span style="font-size:18px;">  public void start(Context context, String name) {
  2. if (!Environment.getExternalStorageState().equals(
  3. android.os.Environment.MEDIA_MOUNTED)) {
  4. return;
  5. }
  6. if (mRecorder == null) {
  7. mRecorder = new MediaRecorder();
  8. mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
  9. mRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
  10. mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
  11. mRecorder.setOutputFile(UtilsForChat.getMusicFilePath(context, name));
  12. try {
  13. mRecorder.prepare();
  14. mRecorder.start();
  15. mEMA = 0.0;
  16. } catch (IllegalStateException e) {
  17. System.out.print(e.getMessage());
  18. } catch (IOException e) {
  19. System.out.print(e.getMessage());
  20. }
  21. }else{
  22. stop();
  23. mRecorder = new MediaRecorder();
  24. mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
  25. mRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
  26. mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
  27. mRecorder.setOutputFile(UtilsForChat.getMusicFilePath(context, name));
  28. try {
  29. mRecorder.prepare();
  30. } catch (IllegalStateException | IOException e) {
  31. // TODO Auto-generated catch block
  32. e.printStackTrace();
  33. }
  34. mRecorder.start();
  35. }
  36. }</span></strong>


代码就变成这样了,其实,这里面更简单点就是每次不判断,进来直接把原来的停止,然后再创建 可以省掉很多代码了

 

然后就是stop函数,一定要写对,否则还是报错的哦:

 

  1. <strong><span style="font-size:18px;">  public void stop() {
  2. if (mRecorder != null) {
  3. try {
  4. mRecorder.stop();
  5. } catch (IllegalStateException e) {
  6. // TODO 如果当前java状态和jni里面的状态不一致,
  7. //e.printStackTrace();
  8. mRecorder = null;
  9. mRecorder = new MediaRecorder();
  10. }
  11. mRecorder.release();
  12. mRecorder = null;
  13. }
  14. }</span></strong>


这里面有人肯定问,你这里面stop已经搞了异常抛出,为什么上面的创建还加上else那样的代码呢,

 

这里面也是机型的原因然后导致我最终还是保留了这段代码   因为要保证你说的话,录音的音不会成为空气啊  这也是很重要的bug啊

Android 使用MediaRecorder录音调用stop()方法的时候报错【转】的更多相关文章

  1. java调用本地方法的时候报错 could not find the main class:xx.program will exit

    如图所示,当在java调用本地方法的时候报错 我的解决办法是把dll文件放到System.out.println(System.getProperty("java.library.path& ...

  2. Android和JavaScript相互调用的方法

    转载地址:http://www.jb51.net/article/77206.htm 这篇文章主要介绍了Android和JavaScript相互调用的方法,实例分析了Android的WebView执行 ...

  3. ajax 调用 .net core WebAPI,报错 400 (Bad Request) Unexpected character encountered while parsing value

    此文由博主前两天的提问及 dudu 的回答整理,地址:https://q.cnblogs.com/list/myquestion 情况说明 基于 .net core 写了一个 Web API,用 po ...

  4. Android 使用MediaRecorder录音

    package com.example.HyyRecord; import android.app.Activity; import android.content.Intent; import an ...

  5. Android JNI中C调用Java方法

    背景需求 我们需要在JNI的C代码调用Java代码.实现原理:使用JNI提供的反射借口来反射得到Java方法,进行调用. JNI关键方法讲解. 1. 在同一个类中,调用其他方法 JNIEXPORT v ...

  6. Android Studio ERROR: x86 emulation currently requires hardware acceleration!报错解决傻瓜教程~

    很早之前就碰到过Android Studio模拟器无法启动的问题,今天终于尝试去解决了下,下面将我解决的方法记录下. 模拟器报错信息为: emulator: ERROR: x86 emulation ...

  7. vc调用BCB的dll 参数传递 报错

    可能原因: 调用方式约定不一致. 函数调用约定如下: 1. __cdecl:C 和 C++ 程序的缺省调用规范. 2. __stdcall:标准调用约定(即WINAPI调用约定),也就是pascal调 ...

  8. 执行发送邮件Send方法时,报错:邮箱不可用。 服务器响应为: 5.7.1 Unable to relay for xxx@xxx.com

    .net代码在执行发送邮件Send方法时,往往出现这个的报错: 邮箱不可用. 服务器响应为: 5.7.1 Unable to relay for xxx@xxx.com 这个问题应该是smtp的设置问 ...

  9. fetch get方法的时候报错

    fetch 报错 Failed to execute 'fetch' on 'Window': Request with GET/HEAD method cannot have body. 翻译过来就 ...

随机推荐

  1. swift protocol 与类继承结合时的bug

    protocol CommonTrait: class { func commonBehavior() -> String } extension CommonTrait { func comm ...

  2. 并发编程学习笔记(5)----AbstractQueuedSynchronizer(AQS)原理及使用

    (一)什么是AQS? 阅读java文档可以知道,AbstractQueuedSynchronizer是实现依赖于先进先出 (FIFO) 等待队列的阻塞锁和相关同步器(信号量.事件,等等)提供一个框架, ...

  3. gym101343J. Husam and the Broken Present 2 (状压DP)

    题意:给定n个串 每个串长度不超过100 找到一个新串 使得这n个串都是它的字串 输出这个新串的最小长度 题解:n是15 n的阶乘的复杂度肯定不行 就想到了2的15次方的复杂度 想到了状压但是不知道怎 ...

  4. Docker 安装并定制 Nginx 服务器

    安装并定制 Nginx 1.查阅对应的官方文档,首先下载镜像文件: [spider@izwz9d74k4cznxtxjeeur9z local]$ sudo docker pull nginx [su ...

  5. PS切图基本操作

    PS切图基本操作 2016-05-11 20:56:46|  分类: PhotoShop|字号 订阅     下载LOFTER我的照片书  |     1首先在“文件”中打开一张图片.   2点击“移 ...

  6. css3文字渐变无效果的解决方案

    现在css3越来月流行了,为了实现一些高大上的效果,我们会用一些渐变的特效,请看文字渐变的特效代码: .title { font-size: 60px; line-height: 80px; text ...

  7. iframe子页面操作父页面并实现屏蔽页面弹出层效果

  8. buf.writeDoubleBE()函数详解

    buf.writeDoubleBE(value, offset[, noAssert]) buf.writeDoubleLE(value, offset[, noAssert]) value {Num ...

  9. Network----轮询

    轮询: 定时每隔多长时间刷新一次,但是,7X24的对服务器的压力会过大,因为在夜间或者是流量低峰期时,他还要持续工作. 客户端发一次请求,服务器就要相应一次. 长轮询: 和轮询的模式不同,长轮询是一次 ...

  10. day21 01 包的初识

    day21 01包的初识 包:把解决一类问题的模块放在同一个文件夹里面-----包(一个包里面通常会含有_init_.py文件(python2里面必须有),但是后面的就没有要求一定要有了) 同样导入的 ...