经过一番折腾,终于找到了一种第三方库来转换android与iOS录音播放格式不兼容的问题。
思路:android/iOS手机录音传给服务器,参数(录音数据+手机端口类型),然后等到服务器广播数据给玩家,判断端口是android,将数据保存为为.amr格式,否则保存为.aac格式。不同手机端口调用相应端口函数。
1、设置端口录音格式:
android–>.amr

public void record(){
        if (isRecording == true ) return;    

        recondpath = Environment.getExternalStorageDirectory().getAbsolutePath(); 
        recondpath += "/ione1.amr"; //录音格式

        File dirs = new File(recondpath);
        if (dirs.exists()){
           dirs.delete();
        }

        stopRecorder();         
        isRecording = true; 

        mRecorder = new MediaRecorder();          
        //设置音源为Micphone 
        mRecorder.setAudioSource(MediaRecorder.AudioSource.CAMCORDER); 
        //设置封装格式
        mRecorder.setoutputFormat(MediaRecorder.OutputFormat.AMR_NB); 
        //设置编码格式 
        mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);         
        mRecorder. setAudioEncodingBitRate(8);//设置音频编码录音比特率
        mRecorder.setAudioChannels(1);//设置录制的音频通道数
        mRecorder.setAudioSamplingRate(8000); //设置音频采样率记录 
        mRecorder.setoutputFile(recondpath);    

        try {    
            mRecorder.prepare();    
        } catch (IOException e) {    
            Log.e(TAG,"prepare() Failed");
        }
        //录音 
        mRecorder.start();        
    }
    public boolean stopRecorder() {
         if( mRecorder == null ) return false;

        try{
            mRecorder.stop();    
            mRecorder.reset();
            mRecorder.release();    
            mRecorder = null;   
        }catch ( IllegalStateException e){
            e.printstacktrace();
        }

        return false;  
    }

ios–>.aac

-(void)startAudioRecording
{
    if(!isRecording)
    {
        [self init];

        isRecording = YES;
        NSLog(@"正在录音");

        NSMutableDictionary *dicM=[NSMutableDictionary dictionary];
        //设置录音格式
        [dicM setobject:@(kAudioFormatMPEG4AAC) forKey:AVFormatIDKey];//录音格式
        //设置录音采样率,8000是电话采样率,对于一般录音已经够了
        [dicM setobject:@(1600) forKey:AVSampleRateKey];
        //设置通道,这里采用单声道
        [dicM setobject:@(1) forKey:AVNumberOfChannelsKey];
        //每个采样点位数,分为8、16、24、32
        [dicM setobject:@(8) forKey:AVLinearPCMBitDepthKey];
        //录音的质量
        [dicM setValue:[NSNumber numberWithInt:AVAudioQualityMin] forKey:AVEncoderAudioQualityKey];
        //是否使用浮点数采样
        [dicM setobject:@(YES) forKey:AVLinearPCMIsFloatKey];

        recorder = [[AVAudioRecorder alloc] initWithURL:recordedFile settings:dicM error:nil];

        [[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayAndRecord error:nil];
        [[AVAudioSession sharedInstance] setActive:YES error:nil];
        [self setSoundSession];
        [recorder peakPowerForChannel:0];

        [recorder preparetoRecord];
        [recorder record];
    }
}

2、接收服务器传来的需要播放的录音数据,根据不同端口录的音保存为数据相同录音格式:
android–>.amr
ios–>.aac

//服务器传来录音数据,保存在表voiceTab中 //voiceTab[index] = event.data
function MainScene:startPlayVoice()
    if self.isstartRecond  == false then -- 没有在录音
        if device.platform == "android" or device.platform == "ios" then --播放录音 
            local len = #voiceTab
            if len >= 1 then
                voicePlay( data )
            else
                print(" 录音播放完了 ")
            end
        end
    end
end
function voicePlay(data)
    if device.platform == "android" then
        local function callback(result)
            -- print("录音播放完成 " ) 
            ReturnRecordingPlayChange(result)
        end

        local path = writefileCheckRecond( data.voicebin,data.content )
        print("开始播放了 voicePlay ")
        local args = { 1,path,callback }
        local sigs = "(ILjava/lang/String;I)I"
        local luaj = require "cocos.cocos2d.luaj"
        local className = "com/cocos2dx/sample/LuaJavaBridge"
        local ok,ret  = luaj.callStaticmethod(className,"sendLuaToJavaAudioRecorPlay",args,sigs)
        if not ok then
            print("luaj error:",ret)
        else
            print("The ret is:",ret)
        end
    elseif device.platform == "ios" then
        local path = writefileCheckRecond( data.voicebin,data.content )
        print("开始播放了 voicePlay ",path)
        local i = iosAudioStartPlay( path ) 
    end
end
function writefileCheckRecond( data,platform )
    local path = device.writablePath.."netSprite/" --获取本地存储目录
    if not io.exists(path) then
        lfs.mkdir(path) --目录不存在,创建此目录
    end

    if platform == "android" then
        path = path.."recond.amr"
    elseif platform == "ios" then
        path = path.."recond.aac"
    else
        return;
    end

    print("写入完成:"..path)

    return path;
end

3、分析两端口录音互相播放的情况(A端口录音–>B端口播放)
android–>android:android玩家录音为.amr传给服务器,android玩家接收录音二进制数据保存的也是.amr格式,播放OK。

public void startPlay(String filePath,int luaFunc){
        if( isPlay == true)
        {
            Log.d("tag","正在播放中");
            return;
        }

        try {
            stopPlay();  

            isPlay = true;
            playLuaFun = luaFunc; 

            mPlayer = MediaPlayer.create(Cocos2dxActivity.getInstance(),Uri.parse(filePath));
            mPlayer.start();// 开始播放 

            mPlayer.setonCompletionListener(new MediaPlayer.OnCompletionListener(){
                @Override 
                public void onCompletion(MediaPlayer m) { 
                    Log.d("tag","播放完毕");                       
                    LuaJavaBridge.callbackLuaFun("1","ReturnRecordingPlayChange",index++);   
                    buySuccessJava(1);
                    Log.d(TAG,"playLuaFun = " + playLuaFun);

                    isPlay = false;
                    stopPlay();
                }
            });

        }catch(Exception e){  
            Log.e(TAG,"prepare() Failed");
        }
    }

android–>iOS:android玩家录音为.amr传给服务器,iOS玩家接收录音二进制数据保存为.amr格式,调用第三方库将格式.amr转.mav格式就播放OK。

-(void)playAudio
{
    if(!isPlay)
    {
        if([player isPlaying])
        {
            NSLog(@"停止录音播放");
            [player pause];
        }
        else
        {
            Nsstring *documentsDirectory= [NSHomeDirectory() stringByAppendingPathComponent:@"Documents"];
            audioRecoderSavePath=[Nsstring stringWithFormat:@"%@/%@",documentsDirectory,@"netSprite/"];
            recoderName= [Nsstring stringWithFormat:@"%@",@"recond.amr"];
            tempRecoderPath=[Nsstring stringWithFormat:@"%@%@",audioRecoderSavePath,recoderName];

            fm = [NSFileManager defaultManager]; //创建文件管理对象
            if([fm fileExistsAtPath:tempRecoderPath] == NO)//判断文件是否存在
            {
                NSLog(@"文件不存在");

                recoderName= [Nsstring stringWithFormat:@"%@",@"recond.aac"];
                tempRecoderPath=[Nsstring stringWithFormat:@"%@%@",recoderName];
                isIos = true;
                [self startPlayAudio];
                return;
            }
            else
            {
                isIos = false;
                NSLog(@"文件存在");
            }

            Nsstring *recoderWavName= [Nsstring stringWithFormat:@"%@",@"recond.wav"];
            tempWavRecoderPath =[Nsstring stringWithFormat:@"%@%@",recoderWavName];
            //第三方库AudioConverter
            [AudioConverter convertAmrToWavAtPath:tempRecoderPath wavSavePath:tempWavRecoderPath asynchronize:YES completion:^(BOOL success,Nsstring * _Nullable resultPath) {
                if (success) {
                    NSLog(@"amr转wav成功!");

                    _wavFilePath = resultPath;

                    [self startPlayAudio];
                } else {
                   NSLog(@"amr转wav失败!");
                }
            }];
        }
    }
}

iOS –>android:ios玩家录音为.aac格式传给服务器,android玩家接收录音二进制数据保存为.aac格式,
android玩家播iOS录的.aac格式OK。

iOS–>iOS:ios玩家录音为.aac格式传给服务器,ios玩家接收录音二进制数据保存为.aac格式,ios玩家播iOS录的.aac格式OK。

参考:
http://www.jb51.cc/article/p-mcdpwusb-bpt.html
http://blog.csdn.net/adalu1986/article/details/50502387
http://www.jianshu.com/p/7dc01b48f8fc

cocos2dx之 android/ios语音交互二的更多相关文章

  1. html5 canvas合成海报所遇问题及解决方案总结

    这篇文章主要介绍了html5 canvas合成海报所遇问题及解决方案总结,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  2. Html5 video标签视频的最佳实践

    这篇文章主要介绍了Html5 video标签视频的最佳实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  3. HTML5在微信内置浏览器下右上角菜单的调整字体导致页面显示错乱的问题

    HTML5在微信内置浏览器下,在右上角菜单的调整字体导致页面显示错乱的问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

  4. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  5. ios – containerURLForSecurityApplicationGroupIdentifier:在iPhone和Watch模拟器上给出不同的结果

    我使用默认的XCode模板创建了一个WatchKit应用程序.我向iOSTarget,WatchkitAppTarget和WatchkitAppExtensionTarget添加了应用程序组权利.(这是应用程序组名称:group.com.lombax.fiveminutes)然后,我尝试使用iOSApp和WatchKitExtension访问共享文件夹URL:延期:iOS应用:但是,测试NSURL

  6. ios – Testflight无法安装应用程序

    我有几个测试人员注册了testflight并连接了他们的设备……他们有不同的ios型号……但是所有这些都有同样的问题.当他们从“safari”或“testflight”应用程序本身单击应用程序的安装按钮时……达到约90%并出现错误消息…

  7. ibm-mobilefirst – 在iOS 7.1上获取“无法安装应用程序,因为证书无效”错误

    当我的客户端将他们的设备更新到iOS7.1,然后尝试从AppCenter更新我们的应用程序时,我收到了上述错误.经过一番搜索,我找到了一个类似问题的帖子here.但是后来因为我在客户端使用AppCenter更新应用程序的环境中,我无法使用USB插件并为他们安装应用程序.在发布支持之前,是否有通过AppCenter进行下载的解决方法?

  8. ios – 视图的简单拖放?

    我正在学习iOS,但我找不到如何向UIView添加拖放行为.我试过了:它说“UIView没有可见的接口声明选择器addTarget”此外,我尝试添加平移手势识别器,但不确定这是否是我需要的它被称为,但不知道如何获得事件的坐标.在iOS中注册移动事件回调/拖放操作的标准简单方法是什么?

  9. ios – 什么控制iTunes中iPhone应用程序支持的语言列表?

    什么控制iPhone应用程序的iTunes页面中支持的语言?

  10. ios – 获得APNs响应BadDeviceToken或Unregistered的可能原因是什么?

    我知道设备令牌在某些时候是有效的.用户如何使其设备令牌变坏?从关于“未注册”的文档:Thedevicetokenisinactiveforthespecifiedtopic.这是否意味着应用程序已被删除?.您应该看到四种分发方法:如果您选择AppStore或Enterprise,您将在后面的对话框中看到Xcode将APNS权利更改为生产:如果选择AdHoc或Development,则aps-environment下的文本将是开发,然后应与后端的配置匹配.

随机推荐

  1. 【cocos2d-x 3.x 学习笔记】对象内存管理

    Cocos2d-x的内存管理cocos2d-x中使用的是上面的引用计数来管理内存,但是又增加了一些自己的特色。cocos2d-x中通过Ref类来实现引用计数,所有需要实现内存自动回收的类都应该继承自Ref类。下面是Ref类的定义:在cocos2d-x中创建对象通常有两种方式:这两中方式的差异可以参见我另一篇博文“对象创建方式讨论”。在cocos2d-x中提倡使用第二种方式,为了避免误用第一种方式,一般将构造函数设为protected或private。参考资料:[1]cocos2d-x高级开发教程2.3节[

  2. 利用cocos2dx 3.2开发消灭星星六如何在cocos2dx中显示中文

    由于编码的不同,在cocos2dx中的Label控件中如果放入中文字,往往会出现乱码。为了方便使用,我把这个从文档中获取中文字的方法放在一个头文件里面Chinese.h这里的tex_vec是cocos2dx提供的一个保存文档内容的一个容器。这里给出ChineseWords,xml的格式再看看ChineseWord的实现Chinese.cpp就这样,以后在需要用到中文字的地方,就先include这个头文件然后调用ChineseWord函数,获取一串中文字符串。

  3. 利用cocos2dx 3.2开发消灭星星七关于星星的算法

    在前面,我们已经在GameLayer中利用随机数初始化了一个StarMatrix,如果还不知道怎么创建星星矩阵请回去看看而且我们也讲了整个游戏的触摸事件的派发了。

  4. cocos2dx3.x 新手打包APK注意事项!

    这个在编译的时候就可以发现了比较好弄这只是我遇到的,其他的以后遇到再补充吧。。。以前被这两个问题坑了好久

  5. 利用cocos2dx 3.2开发消灭星星八游戏的结束判断与数据控制

    如果你看完之前的,那么你基本已经拥有一个消灭星星游戏的雏形。开始把剩下的两两互不相连的星星消去。那么如何判断是GameOver还是进入下一关呢。。其实游戏数据贯穿整个游戏,包括星星消除的时候要加到获得分数上,消去剩下两两不相连的星星的时候的加分政策等,因此如果前面没有做这一块的,最好回去搞一搞。

  6. 利用cocos2dx 3.2开发消灭星星九为游戏添加一些特效

    needClear是一个flag,当游戏判断不能再继续后,这个flag变为true,开始消除剩下的星星clearSumTime是一个累加器ONE_CLEAR_TIME就是每颗星星消除的时间2.连击加分信息一般消除一次星星都会有连击信息和加多少分的信息。其实这些combo标签就是一张图片,也是通过控制其属性或者runAction来实现。源码ComboEffect.hComboEffect.cpp4.消除星星粒子效果消除星星时,为了实现星星爆裂散落的效果,使用了cocos2d提供的粒子特效引擎对于粒子特效不了

  7. 02 Cocos2D-x引擎win7环境搭建及创建项目

    官网有搭建的文章,直接转载记录。环境搭建:本文介绍如何搭建Cocos2d-x3.2版本的开发环境。项目创建:一、通过命令创建项目前面搭建好环境后,怎样创建自己的Cocos2d-x项目呢?先来看看Cocos2d-x3.2的目录吧这就是Cocos2d-x3.2的目录。输入cocosnew项目名–p包名–lcpp–d路径回车就创建成功了例如:成功后,找到这个项目打开proj.win32目录下的Hello.slnF5成功了。

  8. 利用cocos2dx 3.2开发消灭星星十为游戏添加音效项目源码分享

    一个游戏,声音也是非常的重要,其实cocos2dx里面的简单音效引擎的使用是非常简单的。我这里只不过是用一个类对所有的音效进行管理罢了。Audio.hAudio.cpp好了,本系列教程到此结束,第一次写教程如有不对请见谅或指教,谢谢大家。最后附上整个项目的源代码点击打开链接

  9. 03 Helloworld

    程序都有一个入口点,在C++就是main函数了,打开main.cpp,代码如下:123456789101112131415161718#include"main.h"#include"AppDelegate.h"#include"cocos2d.h"USING_NS_CC;intAPIENTRY_tWinMain{UNREFERENCED_ParaMETER;UNREFERENCED_ParaMETER;//createtheapplicationinstanceAppDelegateapp;return

  10. MenuItemImage*图标菜单创建注意事项

    学习cocos2dx,看的是cocos2d-x3.x手游开发实例详解,这本书错误一大把,本着探索求知勇于发现错误改正错误的精神,我跟着书上的例子一起调试,当学习到场景切换这个小节的时候,出了个错误,卡了我好几个小时。

返回
顶部