- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
在阅读本文之前,请了解我知道有更好的方法来编写此代码。
这段代码是很久以前写的,我对Java的了解比以前多了。 尽管如此,我最近查看了代码并希望找出导致问题的原因,以便将来知道。
更新:我已经下载了一个 JProfiler 进行实验。事实证明,当我绘制字符串时,第 158 行的内存分配出现了巨大的峰值。但是,我不知道它到底意味着什么,也不知道为什么它会导致巨大的滞后峰值。
public void draw(Graphics g) {
if (isVisible) {
for (int i = 0; i < debugList.length - 1; i += 2) {
g.setColor(color);
g.setFont(font);
try {
g.drawString(debugList[i] + ":", xPos, yPos + (i * 15)); // Line 158
g.drawString(debugList[i + 1], xPos, yPos + 10 + (i * 15));
}
catch (NullPointerException e) {
if (!errorDisplayed) {
System.out
.println("There was a problem while displaying the debug variables. Check to make sure you added all of the variables you declared in the debug constructor");
errorDisplayed = true;
}
}
}
}
}
不久前,我正在制作一个小型 2D 平台游戏,并且必须修复错误。我认为修复错误的一个很酷的方法是使用一个调试类来在屏幕上显示变量。事实证明,效果很好!它在屏幕上显示所有变量,没有任何问题!
但是,当我在按下某个键后调用 changeVisible()
(位于代码的最底部) 方法时,它会将整个线程卡住大约 3 秒。
这并不是那么重要,因为它已经与我制作的其他调试类一起过时了,但我仍然想知道是什么导致了 future 项目的峰值。
import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
public class Debug {
private String[] debugList;
private boolean errorDisplayed = false, isVisible = true;
private Color color;
private int xPos = 0, yPos = 0, currentIndex = 0;
private Font font;
/**
* This constructor will create an array that can fit (numberOfVars)
* variables
*
* @param numberOfVars
* - The amount of variables you wish to display
* @param setVisible
* - Will toggle visibility on instantiation (true):on
* (false):off
*/
public Debug(int numberOfVars, boolean setVisible) {
debugList = new String[numberOfVars * 2];
if (setVisible)
isVisible = true;
else
isVisible = false;
color = new Color(0, 0, 0);
xPos = 20;
yPos = 30;
font = new Font("SanSerif",Font.BOLD,12);
}
/**
* This constructor will create an array that can fit (numberOfVars)
* variables
*
* @param numberOfVars
* - The amount of variables you wish to display
* @param setVisible
* - Will toggle visibility on instantiation (true):on
* (false):off
* @param inputXPos
* - The location of the text on the x axis
* @param inputYPos
* - The location of the text on the y axis
*/
public Debug(int numberOfVars, boolean setVisible, int inputXPos,
int inputYPos) {
debugList = new String[numberOfVars * 2];
if (setVisible)
isVisible = true;
else
isVisible = false;
color = new Color(0, 0, 0);
xPos = inputXPos;
yPos = inputYPos;
}
/**
* This method will initialize the variable you wish to display
*
* @param objectName
* - the display name of the variable
* @param arg
* - the value of the variable you wish to display
*/
public void addDebug(String objectName, Object arg) {
if (currentIndex + 2 <= debugList.length) {
debugList[currentIndex] = objectName;
debugList[currentIndex + 1] = arg.toString();
currentIndex += 2;
}
}
/**
* This method will change the text color
*
* @param newColor
* - the color of the debug text
*/
public void changeColor(Color newColor) {
color = newColor;
}
/**
* This method will change the text font
* @param font
* - the font of the debug text
*/
public void changeFont(Font font) {
this.font = font;
}
/**
* This method will update the variable value
*
* @param objectName
* - the name of the variable you wish to display (must be the
* same name used in addDebug())
* @param arg
* - the value of the variable you wish to display
*/
public void update(String objectName, Object arg) {
for (int i = 0; i < debugList.length; i += 2) {
if (debugList[i] != null && debugList[i].equals(objectName)) {
if (i < debugList.length + 1)
debugList[i + 1] = arg.toString();
}
}
}
/**
* This method will draw the debug information
*
* @param g
* - The graphics you wish to paint on
* @param lineSpacing
* - The display space between the variables
*/
public void draw(Graphics g, int lineSpacing) {
if (isVisible) {
if (currentIndex == debugList.length) {
for (int i = 0; i < debugList.length - 1; i += 2) {
g.setColor(color);
g.setFont(font);
try {
g.drawString(debugList[i] + ":", xPos, yPos
+ (i * lineSpacing));
g.drawString(debugList[i + 1], xPos, yPos + 10
+ (i * lineSpacing));
}
catch (NullPointerException e) {
if (!errorDisplayed) {
System.out
.println("There was a problem while displaying the debug variables. Check to make sure you added all of the variables you declared in the debug constructor");
errorDisplayed = true;
}
}
}
}
}
}
/**
* This method will draw the debug information
*
* @param g
* - The graphics you wish to paint on
*/
public void draw(Graphics g) {
if (isVisible) {
for (int i = 0; i < debugList.length - 1; i += 2) {
g.setColor(color);
g.setFont(font);
try {
g.drawString(debugList[i] + ":", xPos, yPos + (i * 15));
g.drawString(debugList[i + 1], xPos, yPos + 10 + (i * 15));
}
catch (NullPointerException e) {
if (!errorDisplayed) {
System.out
.println("There was a problem while displaying the debug variables. Check to make sure you added all of the variables you declared in the debug constructor");
errorDisplayed = true;
}
}
}
}
}
/**
* Will toggle the visibility of the debug display
*/
public void changeVisible() {
isVisible = !isVisible;
}
}
主类(用 Debug 注释的行很重要):
package com.bustedearlobes.platformergame;
import java.applet.Applet;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Image;
import java.awt.Point;
import java.util.*;
import javax.swing.JFrame;
public class Game extends Applet implements Runnable {
private static final long serialVersionUID = 1L;
public static int pixelSize = 3;
public static double sx = 0, sy = 0, dir = 0;
public static Dimension windowSize = new Dimension(700,560), pixels = new Dimension(windowSize.width/pixelSize,windowSize.height/pixelSize);
public static Point mousePos = new Point(0,0);
public static boolean isRunning = false,
isMoving = false,
isJumping = false,
isMouseLeft = false,
isMouseRight = false;
public static String name = "2D Game";
public static Level level;
public static Character character;
public static Inventory inventory;
public static Sky sky;
public static ArrayList<Mob> mobs = new ArrayList<Mob>();
public static ArrayList<BrokenBlocks> brokenBlocks = new ArrayList<BrokenBlocks>();
public static Spawner spawner;
public static Sound sound;
public static GameFile gameFile;
public static JFrame frame;
public static Tile tile;
public final static double GRAVITYCONSTANT = 1;
private Image screen;
public static Debug debug = new Debug(5,false); //Debug
public Game() {
setPreferredSize(windowSize);
addKeyListener(new Listening()); //Debug this is where the key listener is that calls the changeVisible() method
addMouseListener(new Listening());
addMouseMotionListener(new Listening());
addMouseWheelListener(new Listening());
}
public void start() { // Defining all the objects required
requestFocus();
tile = new Tile(); // Loading Images...
gameFile = new GameFile();
level = new Level();
character = new Character(Tile.tileSize, Tile.tileSize * 2);
isRunning = true;
sound = new Sound();
inventory = new Inventory();
sky = new Sky();
spawner = new Spawner(10);
debug.changeFont(new Font("San Serif",Font.PLAIN,10)); // Debug
debug.addDebug("Player", character); // Debug
debug.addDebug("Side X", sx); // Debug
debug.addDebug("Side Y", sy); // Debug
debug.addDebug("Sky", sky); // Debug
debug.addDebug("Level.animation", level.animation); // Debug
new Thread(this).start();
}
public void stop() {
isRunning = false;
}
public void tick() {
character.tick();
level.tick(pixels.width / Tile.tileSize + 2,pixels.height / Tile.tileSize + 2);
sky.tick();
inventory.tick();
for (int i = 0; i < mobs.toArray().length;i++)
mobs.get(i).tick();
for (int i = 0; i < brokenBlocks.toArray().length;i++)
brokenBlocks.get(i).tick();
debug.update("Player", character); // Debug
debug.update("Side X", sx); // Debug
debug.update("Side Y", sy); // Debug
debug.update("Sky", sky); // Debug
debug.update("Level.animation", level.animation); // Debug
}
public void render() {
Graphics g = screen.getGraphics();
g.setColor(new Color(100,100,255));
g.fillRect(0,0,pixels.width, pixels.height);
sky.render(g);
level.render(g,pixels.width / Tile.tileSize + 2,pixels.height / Tile.tileSize + 2);
for (int i = 0; i < brokenBlocks.toArray().length;i++)
brokenBlocks.get(i).render(g);
for (int i = 0; i < mobs.toArray().length;i++)
mobs.get(i).render(g);
character.render(g);
inventory.render(g);
debug.draw(g); // Debug
g = getGraphics();
g.drawImage(screen,0,0,windowSize.width,windowSize.height,0,0,pixels.width,pixels.height,null);
g.dispose();
}
public static void main(String[] args) {
Game game = new Game();
frame = new JFrame();
frame.add(game);
frame.setTitle(name);
frame.setResizable(false);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setVisible(true);
game.start();
}
public void run() {
screen = createVolatileImage(pixels.width, pixels.height);
while(isRunning) {
tick();
render();
try {
Thread.sleep(5);
} catch (Exception e) {
}
}
}
}
最佳答案
我无法立即看出问题所在,除非 debugList
可能是一个列表而不是 HashMap。 (不确定这会产生多大的影响。)
但是,我可以告诉你如何找到自己。查看一个名为 JProfiler 的工具。它允许您查看代码中的逐行执行时间,这将让您了解瓶颈在哪里。说明here .
关于java - 这段代码如何导致几秒钟的暂停?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17243245/
我在 Window 2008 x64 上的 IIS 7 下托管了一个网站。 IIS 以 64 位模式运行,该站点有自己的 64 位应用程序池等。该网站大部分时间似乎运行良好,然后每小时突然卡住用户请求
我有一个 imageView 并希望它像这样工作: ImageViewer可见 5秒暂停 ImageView 不可见 5秒暂停 ImageViewer可见 等等... 我该怎么做?我试过 sleep
我在我的 JavaScript 函数中使用了下面的代码。我想在 10 秒后调用这个函数。然而该函数立即被触发!?!不确定发生了什么。 function testing() { //oth
我想暂停计时器,点击按钮后我想继续计时器计数...我搜索但找不到与此相关的功能.. 怎么办? 最佳答案 您将需要一个变量来跟踪自 Chronometer 启动以来耗时: long timeWhenSt
我目前有一个程序可以从麦克风收集声音信号并在 python 上实时显示波形。对于 matplotlib funcanimation,我正在尝试通过这种方式向我的程序添加一些暂停和启动按钮或功能。但它没
我有一个由套接字提供的热Observable。我可以使用pausable暂停套接字供稿。但是一旦“取消暂停”可观察对象,就需要显示套接字在暂停订阅时可能发送的最后一个值。我不想跟踪套接字手动发送的最后
我知道这是可能的,但我还没有找到方法,所以我在问。 在播放 3rd 方音乐(例如 Spotify)时开始在我的应用程序中播放 mp3 声音时。 Spotify 暂停,您必须恢复 Spotify,让它再
我正在尝试使用 iPhone 的前置摄像头录制有声视频。因为我还需要支持暂停/恢复功能,所以我需要使用 AVAssetWriter .我在网上找到了一个用 Objective-C 编写的示例,它几乎实
我知道互斥锁可以作为一种实现,但是我想知道是否有一种方法可以像视频播放一样暂停/恢复另一个线程。当其他正在运行的线程很复杂时,此方法更易于编程。 最佳答案 SIGTSTP是用于暂停进程的信号,如果您有
到目前为止,我已经看到了以下停止动画的技术,但我在这里寻找的是旋转 View 停止在当前的角度,而不是返回到 0。 struct DemoView: View { @State private
我一般在问有关多线程的问题。例如我锁定了一个互斥锁并恢复任务,然后我想挂起它,我的问题是,我应该在挂起之前解锁互斥锁吗?这样当我再次使用互斥锁恢复它时,它会成功恢复吗? 我刚刚开始使用多线程的东西,我
我有2个缩略图链接,单击它们时,它们会以灯箱样式打开视频。我的目标是让它们在打开时播放,在关闭时暂停(单击背景区域时关闭)。 我的HTML代码在这里: M
到目前为止,我没有将我发现的几种不同方法拼凑在一起: http://192.185.121.49/~steveobr/ 我需要所有的语音演示像第一个“商业”一样工作 正如您在实时示例中看到的那样,每个
所以我正在制作某种游戏,玩家可以在其中获得一些能力。玩家回合结束后,服务器应有 5 秒的超时时间,其中不执行任何代码,然后在该时间后结束回合。但是,如果客户端单击其中一项电源,服务器应停止 5 秒超时
我尝试将自己的方法添加到 Tween 类中以暂停/恢复所有补间。这就是我所拥有的: createjs.Tween.pauseAllTweens = function() { for ( var
我对 Azure 搜索标准级别的成本有疑问。是否可以将 Azure 搜索级别从标准更改为基本?是否可以暂时暂停 Azure 搜索标准?我在门户中没有看到此控件。我是否需要将 Azure 搜索实现重新创
如何用相同的代码制作play/Pause按钮。 - (IBAction)min:(id)sender { NSString *path = [[NSBundle mainBundle] pathF
我知道这很可能超出了沙箱范围,但我还是想问一下: 我想在我的应用程序中放置一个“暂停/播放”按钮,以暂停或播放任何背景音频。基本上,我希望实现在多任务栏中找到的播放/暂停按钮。 一个简单的例子是有人用
我正在制作一款编程游戏,玩家可以在其中对盟友的行为进行编程。玩家为给定的盟友编写decide()函数的主体,可以用任何java代码填写,但必须返回一个 Action 。我想为每个盟友提供一组有限的每个
我有功能 1 用于播放音乐,第二个用于设置实际音乐的暂停,我的暂停功能不起作用。我该如何设置暂停? function play(id){ var audio = new Audio('
我是一名优秀的程序员,十分优秀!