2012-09-30 4 views
0

음성 녹음 안드로이드 응용 프로그램을 개발 중입니다. 이를 위해 다음 코드를 사용하고 있습니다. 내가 녹음하고 연주 할 때 배경 잡음으로 목소리를 녹음하고 있습니다. 음성 녹음 만하려면 어떻게해야합니까? 누구든지 나를 도울 수 있습니까?안드로이드 음성 녹음 - 배경 잡음 문제가있는 음성

import java.io.File; 
import java.io.FileInputStream; 
import java.io.FileNotFoundException; 
import java.io.FileOutputStream; 
import java.io.IOException; 

import android.app.Activity; 
import android.media.AudioFormat; 
import android.media.AudioRecord; 
import android.media.MediaRecorder; 
import android.os.Bundle; 
import android.os.Environment; 
import android.view.View; 
import android.widget.Button; 

public class RecorderActivity extends Activity { 
     private static final int RECORDER_BPP = 16; 
     private static final String AUDIO_RECORDER_FILE_EXT_WAV = ".wav"; 
     private static final String AUDIO_RECORDER_FOLDER = "AudioRecorder"; 
     private static final String AUDIO_RECORDER_TEMP_FILE = "record_temp.raw"; 
     private static final int RECORDER_SAMPLERATE = 44100; 
     private static final int RECORDER_CHANNELS = AudioFormat.CHANNEL_IN_STEREO; 
     private static final int RECORDER_AUDIO_ENCODING = AudioFormat.ENCODING_PCM_16BIT; 

     private AudioRecord recorder = null; 
     private int bufferSize = 0; 
     private Thread recordingThread = null; 
     private boolean isRecording = false; 

    @Override 
    public void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.main); 

     setButtonHandlers(); 
     enableButtons(false); 

     bufferSize = AudioRecord.getMinBufferSize(RECORDER_SAMPLERATE,RECORDER_CHANNELS,RECORDER_AUDIO_ENCODING); 
    } 

     private void setButtonHandlers() { 
       ((Button)findViewById(R.id.btnStart)).setOnClickListener(btnClick); 
     ((Button)findViewById(R.id.btnStop)).setOnClickListener(btnClick); 
     } 

     private void enableButton(int id,boolean isEnable){ 
       ((Button)findViewById(id)).setEnabled(isEnable); 
     } 

     private void enableButtons(boolean isRecording) { 
       enableButton(R.id.btnStart,!isRecording); 
       enableButton(R.id.btnStop,isRecording); 
     } 

     private String getFilename(){ 
       String filepath = Environment.getExternalStorageDirectory().getPath(); 
       File file = new File(filepath,AUDIO_RECORDER_FOLDER); 

       if(!file.exists()){ 
         file.mkdirs(); 
       } 

       return (file.getAbsolutePath() + "/" + System.currentTimeMillis() + AUDIO_RECORDER_FILE_EXT_WAV); 
     } 

     private String getTempFilename(){ 
       String filepath = Environment.getExternalStorageDirectory().getPath(); 
       File file = new File(filepath,AUDIO_RECORDER_FOLDER); 

       if(!file.exists()){ 
         file.mkdirs(); 
       } 

       File tempFile = new File(filepath,AUDIO_RECORDER_TEMP_FILE); 

       if(tempFile.exists()) 
         tempFile.delete(); 

       return (file.getAbsolutePath() + "/" + AUDIO_RECORDER_TEMP_FILE); 
     } 

     private void startRecording(){ 
       recorder = new AudioRecord(MediaRecorder.AudioSource.MIC, 
               RECORDER_SAMPLERATE, RECORDER_CHANNELS,RECORDER_AUDIO_ENCODING, bufferSize); 

       recorder.startRecording(); 

       isRecording = true; 

       recordingThread = new Thread(new Runnable() { 

         @Override 
         public void run() { 
           writeAudioDataToFile(); 
         } 
       },"AudioRecorder Thread"); 

       recordingThread.start(); 
     } 

     private void writeAudioDataToFile(){ 
       byte data[] = new byte[bufferSize]; 
       String filename = getTempFilename(); 
       FileOutputStream os = null; 

       try { 
         os = new FileOutputStream(filename); 
       } catch (FileNotFoundException e) { 
         // TODO Auto-generated catch block 
         e.printStackTrace(); 
       } 

       int read = 0; 

       if(null != os){ 
         while(isRecording){ 
           read = recorder.read(data, 0, bufferSize); 

           if(AudioRecord.ERROR_INVALID_OPERATION != read){ 
             try { 
               os.write(data); 
             } catch (IOException e) { 
               e.printStackTrace(); 
             } 
           } 
         } 

         try { 
           os.close(); 
         } catch (IOException e) { 
           e.printStackTrace(); 
         } 
       } 
     } 

     private void stopRecording(){ 
       if(null != recorder){ 
         isRecording = false; 

         recorder.stop(); 
         recorder.release(); 

         recorder = null; 
         recordingThread = null; 
       } 

       copyWaveFile(getTempFilename(),getFilename()); 
       deleteTempFile(); 
     } 

     private void deleteTempFile() { 
       File file = new File(getTempFilename()); 

       file.delete(); 
     } 

     private void copyWaveFile(String inFilename,String outFilename){ 
       FileInputStream in = null; 
       FileOutputStream out = null; 
       long totalAudioLen = 0; 
       long totalDataLen = totalAudioLen + 36; 
       long longSampleRate = RECORDER_SAMPLERATE; 
       int channels = 2; 
       long byteRate = RECORDER_BPP * RECORDER_SAMPLERATE * channels/8; 

       byte[] data = new byte[bufferSize]; 

       try { 
         in = new FileInputStream(inFilename); 
         out = new FileOutputStream(outFilename); 
         totalAudioLen = in.getChannel().size(); 
         totalDataLen = totalAudioLen + 36; 

         AppLog.logString("File size: " + totalDataLen); 

         WriteWaveFileHeader(out, totalAudioLen, totalDataLen, 
             longSampleRate, channels, byteRate); 

         while(in.read(data) != -1){ 
           out.write(data); 
         } 

         in.close(); 
         out.close(); 
       } catch (FileNotFoundException e) { 
         e.printStackTrace(); 
       } catch (IOException e) { 
         e.printStackTrace(); 
       } 
     } 

     private void WriteWaveFileHeader(
         FileOutputStream out, long totalAudioLen, 
         long totalDataLen, long longSampleRate, int channels, 
         long byteRate) throws IOException { 

       byte[] header = new byte[44]; 

       header[0] = 'R'; // RIFF/WAVE header 
       header[1] = 'I'; 
       header[2] = 'F'; 
       header[3] = 'F'; 
       header[4] = (byte) (totalDataLen & 0xff); 
       header[5] = (byte) ((totalDataLen >> 8) & 0xff); 
       header[6] = (byte) ((totalDataLen >> 16) & 0xff); 
       header[7] = (byte) ((totalDataLen >> 24) & 0xff); 
       header[8] = 'W'; 
       header[9] = 'A'; 
       header[10] = 'V'; 
       header[11] = 'E'; 
       header[12] = 'f'; // 'fmt ' chunk 
       header[13] = 'm'; 
       header[14] = 't'; 
       header[15] = ' '; 
       header[16] = 16; // 4 bytes: size of 'fmt ' chunk 
       header[17] = 0; 
       header[18] = 0; 
       header[19] = 0; 
       header[20] = 1; // format = 1 
       header[21] = 0; 
       header[22] = (byte) channels; 
       header[23] = 0; 
       header[24] = (byte) (longSampleRate & 0xff); 
       header[25] = (byte) ((longSampleRate >> 8) & 0xff); 
       header[26] = (byte) ((longSampleRate >> 16) & 0xff); 
       header[27] = (byte) ((longSampleRate >> 24) & 0xff); 
       header[28] = (byte) (byteRate & 0xff); 
       header[29] = (byte) ((byteRate >> 8) & 0xff); 
       header[30] = (byte) ((byteRate >> 16) & 0xff); 
       header[31] = (byte) ((byteRate >> 24) & 0xff); 
       header[32] = (byte) (2 * 16/8); // block align 
       header[33] = 0; 
       header[34] = RECORDER_BPP; // bits per sample 
       header[35] = 0; 
       header[36] = 'd'; 
       header[37] = 'a'; 
       header[38] = 't'; 
       header[39] = 'a'; 
       header[40] = (byte) (totalAudioLen & 0xff); 
       header[41] = (byte) ((totalAudioLen >> 8) & 0xff); 
       header[42] = (byte) ((totalAudioLen >> 16) & 0xff); 
       header[43] = (byte) ((totalAudioLen >> 24) & 0xff); 

       out.write(header, 0, 44); 
     } 

     private View.OnClickListener btnClick = new View.OnClickListener() { 
       @Override 
       public void onClick(View v) { 
         switch(v.getId()){ 
           case R.id.btnStart:{ 
             AppLog.logString("Start Recording"); 

             enableButtons(true); 
             startRecording(); 

             break; 
           } 
           case R.id.btnStop:{ 
             AppLog.logString("Start Recording"); 

             enableButtons(false); 
             stopRecording(); 

             break; 
           } 
         } 
       } 
     }; 
} 

답변

4

에뮬레이터 또는 실제 장치 (있는 경우)에서 테스트 중이십니까? 음향 튜닝 (게인 컨트롤, 노이즈 감소 등을 포함)은 특정 플랫폼 및 제품에 따라 다르며 변경할 수있는 것이 아닙니다.

젤리에는 응용 프로그램에서 녹음에 특정 음향 필터를 적용 할 수있는 API가 포함되어 있으며 noise suppressor 중 하나입니다. 그러나 해당 API를 사용하면 Jellybean 이상을 실행하는 기기에서만 앱이 올바르게 작동하도록 제한됩니다 (모든 기기가 실제로이 기능을 구현하지는 못합니다).

또 다른 가능성은 앱에 노이즈 서프 레서를 포함시키는 것입니다. 나는 생각한다. Speex는 잡음 억제 기능을 포함하지만 낮은 비트 전송률의 음성 인코딩을 목표로합니다. 당신이 NoiseSuppressor하려면

+0

감사합니다. 내 android2.4.3 장치로 테스트 중입니다. 나는 인터넷에서 수색했다. 하지만 나는 샘플을 얻지 못했습니다. NoiseSuppressor 또는 Speex를 사용하여 샘플을 몇 개 줄 수 있습니까? – Manoj

+1

NoiseSuppressor를 사용하는 것은'NoiseSuppressor.create (audioRecord.getAudioSessionId());'처럼 간단해야합니다. 그러나 2.4.3에서는이 API를 사용할 수 없습니다. 4.1 이상에서만 가능합니다. 나는 Speex에 대한 모범이 없지만, 안드로이드 애플리케이션에서 사용하는 사람들이 있다는 것을 알고있다. 적어도 그렇게 할 수있다. – Michael

0

당신은이를 사용할 수 있습니다

// 다음은 젤리 빈 이상에 적용됩니다.

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) { 
     Log.i("Trying to clean up audio because running on SDK " + Build.VERSION.SDK_INT); 

    if (noise && NoiseSuppressor.create(getAudioSessionId()) == null) { 
     Log.i("NoiseSuppressor not present :("); 
    } else { 
     Log.i("NoiseSuppressor enabled!"); 
    } 

    if (gain && AutomaticGainControl.create(getAudioSessionId()) == null) { 
     Log.i("AutomaticGainControl not present :("); 
    } else { 
     Log.i("AutomaticGainControl enabled!"); 
    } 

    if (echo && AcousticEchoCanceler.create(getAudioSessionId()) == null) { 
     Log.i("AcousticEchoCanceler not present :("); 
    } else { 
     Log.i("AcousticEchoCanceler enabled!"); 
    } 
} 

이 코드는 배경 잡음, 에코 및 이득을 제거 할 수 있습니다.