Какой самый простой способ генерировать звук синусоидальной волны на любой частоте в Java? Образец размером более 2 байтов поможет, но это не имеет большого значения.
Спасибо
Какой самый простой способ генерировать звук синусоидальной волны на любой частоте в Java? Образец размером более 2 байтов поможет, но это не имеет большого значения.
Спасибо
См. Beeper
для автономного примера.
Возможно, что-то проще?
Это 51 строка фрагмента (повторенная ниже - разнесенная для однострочных и строчных комментариев), как показано вверху связанного ответа, примерно такая же, как генерация тонального сигнала (ОК, вы можете вынуть 5 + линии для гармоники).
Люди, похоже, полагают, что это должен быть метод, встроенный в набор инструментальных средств для создания чистого тона. Это не так и требует небольшого вычисления, чтобы сделать это.
/** Generates a tone, and assigns it to the Clip. */
public void generateTone()
throws LineUnavailableException {
if ( clip!=null ) {
clip.stop();
clip.close();
} else {
clip = AudioSystem.getClip();
}
boolean addHarmonic = harmonic.isSelected();
int intSR = ((Integer)sampleRate.getSelectedItem()).intValue();
int intFPW = framesPerWavelength.getValue();
float sampleRate = (float)intSR;
// oddly, the sound does not loop well for less than
// around 5 or so, wavelengths
int wavelengths = 20;
byte[] buf = new byte[2*intFPW*wavelengths];
AudioFormat af = new AudioFormat(
sampleRate,
8, // sample size in bits
2, // channels
true, // signed
false // bigendian
);
int maxVol = 127;
for(int i=0; i<intFPW*wavelengths; i++){
double angle = ((float)(i*2)/((float)intFPW))*(Math.PI);
buf[i*2]=getByteValue(angle);
if(addHarmonic) {
buf[(i*2)+1]=getByteValue(2*angle);
} else {
buf[(i*2)+1] = buf[i*2];
}
}
try {
byte[] b = buf;
AudioInputStream ais = new AudioInputStream(
new ByteArrayInputStream(b),
af,
buf.length/2 );
clip.open( ais );
} catch(Exception e) {
e.printStackTrace();
}
}
Используйте Java Sound API и Math.sin
, чтобы создать фактические уровни волн.
http://www.developer.com/java/other/article.php/2226701 имеет отличный учебник по этому поводу, о котором я уже упоминал некоторое время назад. http://jsresources.org/examples/ была еще одной полезной ссылкой.
Если вам нужен простой код для начала работы, это должно помочь
import javax.sound.sampled.AudioFormat;
import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.SourceDataLine;
public class SinSynth {
//
protected static final int SAMPLE_RATE = 16 * 1024;
public static byte[] createSinWaveBuffer(double freq, int ms) {
int samples = (int)((ms * SAMPLE_RATE) / 1000);
byte[] output = new byte[samples];
//
double period = (double)SAMPLE_RATE / freq;
for (int i = 0; i < output.length; i++) {
double angle = 2.0 * Math.PI * i / period;
output[i] = (byte)(Math.sin(angle) * 127f); }
return output;
}
public static void main(String[] args) throws LineUnavailableException {
final AudioFormat af = new AudioFormat(SAMPLE_RATE, 8, 1, true, true);
SourceDataLine line = AudioSystem.getSourceDataLine(af);
line.open(af, SAMPLE_RATE);
line.start();
boolean forwardNotBack = true;
for(double freq = 400; freq <= 800;) {
byte [] toneBuffer = createSinWaveBuffer(freq, 50);
int count = line.write(toneBuffer, 0, toneBuffer.length);
if(forwardNotBack) {
freq += 20;
forwardNotBack = false; }
else {
freq -= 10;
forwardNotBack = true;
} }
line.drain();
line.close();
}
}
В первом совете я создайте класс Примечание, который возвращает частоты заметок и преобразует их в массив байтов.
Затем потоком это очень легко
protected static final int SAMPLE_RATE = 8 * 1024;
public static void main(String[] args) throws LineUnavailableException {
final AudioFormat af = new AudioFormat(SAMPLE_RATE, 8, 1, true, true);
SourceDataLine line = AudioSystem.getSourceDataLine(af);
line.open(af, SAMPLE_RATE);
line.start();
// fist argument is duration of playing note
byte[] noteDo = Note.DO.getTone(1, SAMPLE_RATE);
byte[] noteRe = Note.RE.getTone(0.5, SAMPLE_RATE);
byte[] noteMi = Note.MI.getTone(1.5, SAMPLE_RATE);
line.write(noteDo, 0, noteDo.length);
line.write(noteRe, 0, noteRe.length);
line.write(noteMi, 0, noteMi.length);
line.drain();
line.close();
}
public enum Note {
DO(0.0f), DO_DIEZ(1.0f),
RE(2.0f), RE_DIEZ(3.0f),
MI(4.0f),
FA(5.0f), FA_DIEZ(6.0f),
SOL(7.0f),SOL_DIEZ(8.0f),
LYA(9.0f),LYA_DIEZ(10.0f),
SI(11.0f);
private final double mPhase;
Note(double phase) {
mPhase = phase;
}
public double getNoteFrequencies() {
double index = getmPhase()/ 12.0d;
return 440 * Math.pow(2, index);
}
public static Note getNote(double phase) throws Exception {
Note findNote = null;
for (Note note : Note.values()){
if (note.getmPhase() == phase){
findNote = note;
}
}
if (findNote == null)
throw new Exception("Note not found: Ilegal phase " + phase);
else
return findNote;
}
public byte[] getTone(double duration, int rate){
double frequencies = getNoteFrequencies();
int maxLength = (int)(duration * rate);
byte generatedTone[] = new byte[2 * maxLength];
double[] sample = new double[maxLength];
int idx = 0;
for (int x = 0; x < maxLength; x++){
sample[x] = sine(x, frequencies / rate);
}
for (final double dVal : sample) {
final short val = (short) ((dVal * 100f));
// in 16 bit wav PCM, first byte is the low order byte
generatedTone[idx++] = (byte) (val & 0x00ff);
generatedTone[idx++] = (byte) ((val & 0xff00) >>> 8);
}
return generatedTone;
}
private double sine(int x, double frequencies){
return Math.sin( 2*Math.PI * x * frequencies);
}
public double getmPhase() {
return mPhase;
}
}
если вы ищете просто класс для вызова звукового сигнала, попробуйте следующее: (некоторый код заимствован из Thumbz)
package ditdah;
import javax.sound.sampled.AudioFormat;
import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.SourceDataLine;
public class Beep {
protected static final int SAMPLE_RATE = 16 * 1024;
public void play(double freq, int length) {
final AudioFormat af = new AudioFormat(SAMPLE_RATE, 8, 1, true, true);
try {
SourceDataLine line = AudioSystem.getSourceDataLine(af);
line.open(af, SAMPLE_RATE);
line.start();
byte[] toneBuffer = this.createSinWaveBuffer(freq, length);
say.it(toneBuffer.toString() + " " + toneBuffer.length);
int count = line.write(toneBuffer, 0, toneBuffer.length);
line.drain();
line.close();
} catch (LineUnavailableException e) {
say.it(e.getLocalizedMessage());
}
}
public byte[] createSinWaveBuffer(double freq, int ms) {
int samples = (int) ((ms * SAMPLE_RATE) / 1000);
byte[] output = new byte[samples];
//
double period = (double) SAMPLE_RATE / freq;
for (int i = 0; i < output.length; i++) {
double angle = 2.0 * Math.PI * i / period;
output[i] = (byte) (Math.sin(angle) * 127f);
}
return output;
}
}
Метод createSinWaveBuffer() в этих ответах не дает хороших данных формы сигнала для непрерывного воспроизведения. Последний байт должен быть близок к нулю, чтобы получить полную форму волны. Лучший пример -
protected static final float SAMPLE_RATE = 16 * 1024;
public static byte[] createSinWaveBuffer(double freq) {
double waveLen = 1.0/freq;
int samples = (int) Math.round(waveLen * 5 * SAMPLE_RATE);
byte[] output = new byte[samples];
double period = SAMPLE_RATE / freq;
for (int i = 0; i < output.length; i++) {
double angle = 2.0 * Math.PI * i / period;
output[i] = (byte)(Math.sin(angle) * 127f); }
return output;
}
Я хотел бы отметить, что существует очень эффективный алгоритм генерации синусоидальных волн.
DSP Trick: Генератор синусоидального тона http://www.dspguru.com/dsp/tricks/sine_tone_generator