Swift-语音变声播放

QQ里面的变声功能是不是很搞笑?想要实现这样的功能?其实很简单,只需要使用音频引擎(AVAudioEngine)来改变节点的速率(AVAudioUnitTimePitch)、调整回声(AVAudioUnitDistortion)和混响(AVAudioUnitReverb)的值就可以达到你想要的很多效果。

[TOC]

音频录制

初始化录音器

//初始化录音器
func setUpRecord() -> Void {
   //创建录音文件保存路径
   let url = self.getSavePath()
   //创建录音格式设置
   let settingDic = self.getAudioSetting()
   //创建录音机
   do{
       try audioRecorder = AVAudioRecorder(url: url, settings: settingDic)
       audioRecorder.delegate = self
       audioRecorder.isMeteringEnabled = true
       audioRecorder.prepareToRecord()
       print("成功初始化")
   }
   catch{
       print("初始化失败")
   }
}

获取录音权限

//获取录音权限. 返回YES为无拒绝,NO为拒绝录音.
func canRecord() -> Bool {
   var canR = false
   if (UIDevice.current.systemVersion as NSString).floatValue >= 7.0 {
       let audioSession = AVAudioSession.sharedInstance()
       if (audioSession.responds(to: #selector(AVAudioSession.requestRecordPermission(_:)))) {
           AVAudioSession.sharedInstance().requestRecordPermission({(granted: Bool)-> Void in
               
               canR = granted
               if granted {
                   print("granted")
               } else{
                   print("not granted")
               }
           })
           
       }
   }else{
       canR = true
   }
   
   return canR
}

长按开始录音

//长按响应函数
func longGesAction(longGes:UILongPressGestureRecognizer) -> Void {
   switch longGes.state {
   case .began:
       NSLog("录音开始")
       //判断下是否授权使用麦克风
       if self.canRecord() {
           audioRecorder.record()
       }
       .....

音频播放

创建AudioFile

AVAudioFile的初始化函数 init(forReading fileURL: URL) throws 后面跟着 throws 关键词,所以在调用是需要使用 Do Try Catch 来处理错误。

do {
  try audioFile = AVAudioFile(forReading: url)
} catch {
  showAlert(Alerts.AudioFileError, message: String(describing: error))
}

节点速率、节距、回声、混响的调整

在改变节点数据的时候,你需要先申请一个音频引擎,然后将修改的东西使用 attach() 函数加入到引擎中。

速率、节距

let changeRatePitchNode = AVAudioUnitTimePitch()
//速率
changeRatePitchNode.rate = rate!
//节距
changeRatePitchNode.pitch = pitch!

回声

let echoNode = AVAudioUnitDistortion()
echoNode.loadFactoryPreset(.multiEcho1)

混响

let revrebNode = AVAudioUnitReverb()
revrebNode.loadFactoryPreset(.cathedral)
revrebNode.wetDryMix = 50

做完这些过后,你还需要将这些节点用音频引擎的 connect(_ node1: AVAudioNode, to node2: AVAudioNode, format: AVAudioFormat?) 方法拼接起来。

拼接所有节点

在这里我写了一个多参数的函数来拼接

func connectAudioNodes(_ nodes: AVAudioNode...) {
   for x in 0..<nodes.count-1 {
       audioEngine.connect(nodes[x], to: nodes[x+1], format: audioFile.processingFormat)
   }
}

调用:
connectAudioNodes(audioPlayerNode,changeRatePitchNode,audioEngine.outputNode) //记得这个 audioEngine.outputNode 是必须要的,相当于上下文的结尾

启动播放引擎

在这里我加了一个计时器来停止播放

//启动引擎 schedule to play and start the engine!
audioPlayerNode.stop()
audioPlayerNode.scheduleFile(audioFile, at: nil) { 
  var delayInSeconds :Double = 0.0 //延迟秒数
  if let lastRenderTime = self.audioPlayerNode.lastRenderTime, let playerTime = self.audioPlayerNode.playerTime(forNodeTime: lastRenderTime) {
      if rate != nil {
          delayInSeconds = Double(self.audioFile.length - playerTime.sampleTime)/Double(self.audioFile.processingFormat.sampleRate)/Double(rate!)
      }else {
          delayInSeconds = Double(self.audioFile.length - playerTime.sampleTime)/Double(self.audioFile.processingFormat.sampleRate)
      }
  }
  //组装一个定时器在播放完成时调用stopAudio schedule a stop timer for when audio finishes playing
  self.stopTimer = Timer(timeInterval: delayInSeconds, target: self, selector: #selector(VideoPlayHelper.stopAudio), userInfo: nil, repeats: false)
  RunLoop.main.add(self.stopTimer, forMode: RunLoopMode.defaultRunLoopMode)
}
do {
  try audioEngine.start()
} catch {
  showAlert(Alerts.AudioEngineError, message: String(describing: error))
  return
}
audioPlayerNode.play()

当然在停止的时候记得将 播放引擎、定时器、播放器都停止掉哟。

func stopAudio() {
   if audioPlayerNode != nil {
       audioPlayerNode.stop()
   }
   
   if stopTimer != nil {
       stopTimer.invalidate()
   }
   
   if audioEngine != nil {
       audioEngine.stop()
       audioEngine.reset()
   }
}

Demo地址

GitHub地址 你如果觉得喜欢帮忙给个Star!谢谢!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容