为什么我会听到声音中的滞后现象?

发布于 2024-12-02 23:30:57 字数 6284 浏览 0 评论 0原文

我正在用 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 使用的提示。 提前致谢。

I'm developing a game in Java, and have made an attempt at coding a simple sound system for it made of two class files: one that handles them, and one for each sound file loaded.

My problem is that every once in a while I get a huge lagspike which means a complete stop for like a second or two, really interrupting gameplay. I'm suspecting Garbage control but I'm not entirely sure. The only thing I know for a fact is that the source of the problem lies in the sounds.

Here's my SoundHandler.java:

(The loading of sounds is unimportant, as far as I know it should be unrelated to the problem)

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;
}
}

Here's my 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(); }
}
}

I've tried searching around for a general way to implement sounds in a game, most I've encountered didn't feature multiple instances of the same sound file running at once, this is what I have so far. I'm sure there are a few tiny inefficiencies especially in the loading code, but there's got to be some leak I'm missing. Any tips for custom GC usage are also welcome.
Thanks in advance.

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

可遇━不可求 2024-12-09 23:30:57

在 Java Oracle 站点上研究并发线程可能会更容易......我可以获得一个链接。有多种方法可以将线程同步在一起,并且它们确实有示例。

Oracle 并发文档! ->> 对象同步

It might be easier to research concurrent threading on the Java Oracle site... I can get a link. There are ways to sync threads together and they do have examples.

Oracle Concurrency Documenation! ->> Object Syncing

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文