Почему я получаю лаги от звука? - PullRequest
3 голосов
/ 04 сентября 2011

Я занимаюсь разработкой игры на Java и попытался написать для нее простую звуковую систему, состоящую из двух файлов классов: один обрабатывает их, а другой - для каждого загруженного звукового файла.

Моя проблема в том, что время от времени я получаю огромный лаг-шип, который означает полную остановку на секунду или две, действительно прерывая игровой процесс. Я подозреваю, что контроль над мусором, но я не совсем уверен. Единственное, что я точно знаю, это то, что источник проблемы лежит в звуках.

Вот мой SoundHandler.java:

(Загрузка звуков не важна, насколько я знаю, это не должно быть связано с проблемой)

package Classes;

import java.io.*;

import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.Clip;
import javax.sound.sampled.LineEvent;
import javax.sound.sampled.LineListener;
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.FloatControl;

public class SoundHandler {

Sound Sounds[] = new Sound[1];

public SoundHandler()
{
    if(!LoadSounds("Sounds.cfg"))
        System.out.print("Failiure upon loading sounds!");
}

public boolean LoadSounds(String FileName)
{
    String line = "";
    String SoundName = "";
    String SoundFile = "";
    String[] token3 = new String[10];
    boolean EndOfFile = false;
    int LineCount = 0;
    BufferedReader characterfile = null;

    try
    {
        characterfile = new BufferedReader(new FileReader(GameMain.GameFolder+"Data/"+FileName));
    }
        catch(FileNotFoundException fileex)
        {
            System.out.println(FileName+": file not found.");
            return false;
        }
    while(!EndOfFile && line != null)
    {
        try
        {
            line = characterfile.readLine();
            if(line != null)
            {
                if(line.indexOf("//") == -1 && !line.equals(""))
                    LineCount++;
            }
        }
            catch(IOException ioexception)
            {
                if(LineCount == 0)
                {
                    System.out.println(FileName+": error loading file.");
                    return false;
                }
                EndOfFile = true;
            }
    }
    try { characterfile.close(); } catch(IOException ioexception) { characterfile = null; }

    Sounds = new Sound[LineCount];
    EndOfFile = false;
    LineCount = 0;

    try
    {
        characterfile = new BufferedReader(new FileReader(GameMain.GameFolder+"Data/"+FileName));
    }
        catch(FileNotFoundException fileex)
        {
            System.out.println(FileName+": file not found.");
            return false;
        }

    try
    {
        line = characterfile.readLine();
        if(line != null)
            if(line.indexOf("//") == -1 && !line.equals(""))
                LineCount++;
    }
        catch(IOException ioexception) { }

    while(EndOfFile == false && line != null) {
        if(line.indexOf("//") == -1 && !line.equals(""))
        {
            line = line.trim();
            line = line.replaceAll("\t\t", "\t");
            line = line.replaceAll("\t\t", "\t");
            line = line.replaceAll("\t\t", "\t");
            line = line.replaceAll("\t\t", "\t");

            int Spot = line.indexOf("\t");
            SoundName = line.substring(0,Spot);
            SoundFile = line.substring(Spot+1);
            Sounds[LineCount-1] = new Sound(SoundName,SoundFile);
        }

            try {
                    line = characterfile.readLine();
                    if(line != null)
                        if(line.indexOf("//") == -1 && !line.equals(""))
                            LineCount++;
            } catch(IOException ioexception1) { EndOfFile = true; }
    }
    try { characterfile.close(); } catch(IOException ioexception) { }
    return true;
}

public File GetSoundFile(String Name)
{
    File result = null;
    for(int i = 0; i < Sounds.length; i++)
        if(Sounds[i].Name.equals(Name))
            result = Sounds[i].File;
    return result;
}

public Sound GetSound(String Name)
{
    Sound result = null;
    for(int i = 0; i < Sounds.length; i++)
        if(Sounds[i].Name.equals(Name))
            result = Sounds[i];
    return result;
}

public int GetSoundID(String Name)
{
    int result = 0;
    for(int i = 0; i < Sounds.length; i++)
        if(Sounds[i].Name.equals(Name))
            result = i;
    return result;
}

Double MasterVolume = 0.75;

public float CalcVolume(double Vol)
{
    float result = 0f;
    result = -40.0f + (float)(MasterVolume*Vol*40);
    if(result < -40.0f)
        result = -40.0f;
    if(result > 0.0f)
        result = 0.0f;
    return result;
}

public boolean PlaySound(String SoundName, double Vol)
{
    int ID = GetSoundID(SoundName);

    try
    {
        Clip CurrentClip;
        Sounds[ID].Reset(false);
        CurrentClip = (Clip) AudioSystem.getLine(Sounds[ID].info);
        CurrentClip.addLineListener(new LineListener() {
            public void update(LineEvent event) {
                if (event.getType() == LineEvent.Type.STOP) {
                    event.getLine().close();
                }
            }
        });
        CurrentClip.open(Sounds[ID].sound);

        FloatControl Volume;
        Volume = (FloatControl) CurrentClip.getControl(FloatControl.Type.MASTER_GAIN);
        Volume.setValue(CalcVolume(Vol));
        CurrentClip.start();
    }
    catch(LineUnavailableException e) { e.printStackTrace(); }
    catch(IOException e) { e.printStackTrace(); }

    return true;
}
}

Вот мой Sound.java:

package Classes;

import java.io.File;
import java.io.IOException;

import javax.sound.sampled.AudioInputStream;
import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.UnsupportedAudioFileException;
import javax.sound.sampled.Clip;
import javax.sound.sampled.DataLine;

public class Sound {

public String Name = "";
public File File = null;

AudioInputStream sound;
DataLine.Info info;
Clip clip;

public Sound(String iName, String FileName)
{
    Name = iName;
    File = new File(GameMain.GameFolder+"Sound/"+FileName+".wav");
    Reset(true);
}

public void Reset(boolean init)
{
    try
    {
        if(!init)
            sound.close();
        sound = AudioSystem.getAudioInputStream(File);
        info = new DataLine.Info(Clip.class, sound.getFormat());
    }
    catch(IOException e) { e.printStackTrace(); }
    catch(UnsupportedAudioFileException e) { e.printStackTrace(); }
}
}

Я попытался найти общий способ реализации звуков в игре, большинство из которых я встречал, в которых не было одновременно запущено несколько экземпляров одного и того же звукового файла, это то, что я до сих пор видел. Я уверен, что есть несколько крошечных недостатков, особенно в загрузочном коде, но должна быть какая-то утечка, которую я пропускаю. Любые советы по использованию GC также приветствуются. Заранее спасибо.

1 Ответ

0 голосов
/ 06 сентября 2012

Может быть проще исследовать параллельные потоки на сайте Java Oracle ... Я могу получить ссылку. Есть способы синхронизировать потоки вместе, и у них есть примеры.

Документация по параллелизму Oracle ! - >> Синхронизация объектов

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...