- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试使用 C++ 和 OpenGL/GLUT 制作一个 Snake 克隆。然而,我一直在编程允许输入 Action 之间的短时间间隔时遇到问题。我已经尝试了一些计时方法,最后我为它创建了一个类(如下所示)。这似乎是对输入延迟进行编程的最佳方式(而不是 glutTimerFunc() 或 sleep()),因为计时器独立于游戏循环运行,而不是暂停整个程序。这很重要,因为我希望播放器能够随时暂停。不幸的是,我现在也遇到了这种方法的问题。我的计时器类似乎忽略了我给它的双倍时间限制(简单表示为双倍“限制”)。
为了测试该类,我设置了一个简单的循环控制台程序,该程序在计时器达到时间限制时显示来自用户的定向输入。它应该每 0.33 秒显示一次输入。相反,它以大约 0.8 秒的固定间隔显示输入,而不管时间限制的值是多少。 为什么它不在给定的时间间隔显示输入,为什么它有自己的时间限制?
这也恰好是我第一个没有教程的主要 C++/OpenGL 项目,因此欢迎对我的代码/方法提出任何意见或建议!
#include <iostream>
#include "timer.h"
// Include all files necessary for OpenGL/GLUT here.
using namespace std;
Timer timer;
// Insert necessary OpenGL/GLUT code for display/looping here.
void update(int value)
{
if (timer.checkTime())
{
if (GetAsyncKeyState(VK_LEFT))
cout << "You pressed LEFT!" << endl;
else if (GetAsyncKeyState(VK_RIGHT))
cout << "You pressed RIGHT!" << endl;
else if (GetAsyncKeyState(VK_UP))
cout << "You pressed UP!" << endl;
else if (GetAsyncKeyState(VK_DOWN))
cout << "You pressed DOWN!" << endl;
}
glutTimerFunc(1000/60, update, 0);
glutPostRedisplay();
}
定时器.h
#pragma once
#include <time.h>
class Timer
{
public:
Timer();
bool checkTime(double limit = 0.33);
private:
double getElapsed();
time_t start;
time_t now;
double elapsed;
bool running;
};
定时器.cpp
#include "timer.h"
Timer::Timer()
{
running = false;
}
bool Timer::checkTime(double limit)
{
elapsed = getElapsed();
if (elapsed < limit)
{
return false;
}
else if (elapsed >= limit)
{
running = false;
return true;
}
}
double Timer::getElapsed()
{
if (! running)
{
time(&start);
running = true;
return 0.00;
}
else
{
time(&now);
return difftime(now, start);
}
}
最佳答案
glutTimer 每 1000/60 毫秒触发一次,您检查 Timer::checkTime,它调用 getElapsed。您的所有时间函数都以 double 定义,但您使用的是 time_t,它的分辨率仅为 1 秒。
因此,你会得到看起来像这样的东西(模拟数字)
start time: 1234.5 seconds
glutTimer: 1234.516 seconds (getElapsed = 0) // checkTime returns false
glutTimer: 1234.532 seconds (getElapsed = 0)
...
glutTimer: 1234.596 seconds (getElapsed = 0)
glutTimer: 1235.012 seconds (getElapsed = 1) // condition finally returns true
...
因此,实际延迟取决于您何时设置开始时间相对于从 time() 使用的纪元开始的一秒的实际开始时间;
我怀疑如果您对其进行统计测量,平均时间接近 0.5 秒。
关于“决议”的回答问题:
返回当前时间的不同函数会返回不同级别的准确度。例如,现在,我屏幕右下角的时钟显示为“12:14 PM”。是正好 12:14 没有秒,还是 12:14 加 59 秒?我不能说,因为时钟显示的“分辨率”是一分钟。同样,我可能会说现在是 12 点一刻,如果我以“一刻钟”的分辨率报告时间,那么实际上是整点过 14 分钟。作为人类,我们一直都在不假思索地这样做。在软件中,您必须注意调用任何函数的这些细节。
如果您使用的是 Windows,则可以通过 QueryPerformanceCounter API 获得高分辨率计时器。在大多数平台上,性能计数器是基于硬件的,分辨率在微秒范围内。
LARGE_INTEGER StartingTime, EndingTime, ElapsedMicroseconds;
LARGE_INTEGER Frequency;
QueryPerformanceFrequency(&Frequency); // get number of ticks per second
QueryPerformanceCounter(&StartingTime); // get starting # of ticks
// Activity to be timed
QueryPerformanceCounter(&EndingTime); // get ending # of ticks
ElapsedMicroseconds.QuadPart = EndingTime.QuadPart - StartingTime.QuadPart;
//
// We now have the elapsed number of ticks, along with the
// number of ticks-per-second. We use these values
// to convert to the number of elapsed microseconds.
// To guard against loss-of-precision, we convert
// to microseconds *before* dividing by ticks-per-second.
//
ElapsedMicroseconds.QuadPart *= 1000000;
ElapsedMicroseconds.QuadPart /= Frequency.QuadPart;
Linux 上可能有类似的工具,但我不熟悉。
试试这个:
void update(int value)
{
if (timer.hasTicked())
{
if (GetAsyncKeyState(VK_LEFT))
cout << "You pressed LEFT!" << endl;
else if (GetAsyncKeyState(VK_RIGHT))
cout << "You pressed RIGHT!" << endl;
else if (GetAsyncKeyState(VK_UP))
cout << "You pressed UP!" << endl;
else if (GetAsyncKeyState(VK_DOWN))
cout << "You pressed DOWN!" << endl;
}
else if (!timer.isRunning())
{
timer.start();
}
glutTimerFunc(1000/60, update, 0);
glutPostRedisplay();
}
定时器.h
// this class provides a timer that can be polled and will allow the user to tell if a period has elapsed.
// note that this timer does NOT throw any events for timeout.
class PollTimer
{
public:
PollTimer();
// assuming time limit is a number of msec that and fits within a normal integer rather than the 64 bit
// variant (that would be a LONG LONG delay).
void setTimeout(int msDelay);
// Timers generally expose start/stop and it’s not generally a good idea to make a single function
// that overloads complex combinations of behaviors as here both the start & elapsed operations.
// admit this is a simple case, but generally it’s a bad design pattern that leads to “evil”.
void start();
void stop();
bool isRunning();
// Has the timer expired since the last poll
bool hasTicked();
private:
LARGE_INTEGER startTime;
LARGE_INTEGER frequency; // per second
int delay; // in milliseconds
bool running;
};
定时器.cpp
#include "timer.h"
PollTimer::PollTimer()
{
// omitting error check for hardware that doesn’t support this.
QueryPerformanceFrequency(& frequency); // get number of ticks per second
running = false;
}
void PollTimer::setTimeout(int msDelay)
{
delay = msDelay;
}
void PollTimer::start()
{
QueryPerformanceCounter(&startTime);
running = true;
}
void PollTimer::stop()
{
running = false;
}
bool PollTimer::isRunning()
{
return running;
}
bool PollTimer::hasTicked()
{
if (!running)
return false;
LARGE_INTEGER now;
QueryPerformanceCounter(&now);
LARGE_INTEGER ElapsedMilliseconds;
ElapsedMilliseconds.QuadPart = now.QuadPart - startTime.QuadPart;
ElapsedMilliseconds.QuadPart *= 1000000;
ElapsedMilliseconds.QuadPart /= frequency.QuadPart; // now microseconds
ElapsedMilliseconds.QuadPart /= 1000; // milliseconds
bool fExpired = ( ElapsedMilliseconds.HighPart > 0 || ElapsedMilliseconds.LowPart >= delay ) ;
if (fExpired)
{
// reset start time
start(); // don’t copy/paste code you can call.
}
return fExpired;
}
关于C++ 蛇克隆 : timer function ignores given stop time and stops at it's own fixed time,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24846663/
根据我的理解,INSERT IGNORE 插入一个新条目,如果它不存在,如果存在,则忽略它。所以我已经尝试这样做了一段时间,但似乎没有用。这是我的尝试: insert insert ignore in
出于某种奇怪的原因,StyleCop 不尊重我的文档规则设置。考虑以下代码: internal class SomeClass { public SomeClass() { }
我有一个带有字符串参数和 bool 返回值的方法。我想确保它总是返回 true 我试过了 myClass.Foo(A.Ignored) .WithReturnType() .Returns(tru
这个问题在这里已经有了答案: Filter invoke twice when register as Spring bean (2 个答案) 关闭 1 年前。 我们正面临 SpringSecuri
将 iconv 与 //TRANSLIT//IGNORE 一起使用从 utf8 转换为 ascii 工作正常;它根据当前语言环境(在我的情况下为 de_DE)将不可转换的字符替换为正确的音译: > e
我有这样的查询: $query="INSERT IGNORE INTO mytable (key, word1, word2) VALUES (1, 'abc', 'def')"; 如果因为主键已经存
我有一个 MySQL 表,如下所示: `id` int(10) unsigned NOT NULL AUTO_INCREMENT, `company_id` int(8) unsigned NOT N
git blame --ignore-revs-file显然是现代 Git 中存在的一个选项。 只有一个问题。它不起作用。 或者至少,它对我不起作用: 您可以将其添加到 shell 脚本中: mkdi
使用docker stack deploy,我可以看到如下信息: Ignoring unsupported options: restart 是否意味着重启政策没有到位? 是否必须在撰写文件之外指定它
我正在尝试检查被 git 忽略的文件,我发现 2 个命令显示相同的结果。 哪个是 git check-ignore * git ls-files --others --ignored --exclud
我想为我的 bash 设置和插件等创建一个 git 仓库。我忽略了所有内容(第 0 行),然后在 repo 中手动添加了我想要的文件/文件夹。 (我必须这样做,因为 repo 在我的 ~ 文件夹中。)
我们有一个集中式开发服务器,每个人都在本地结账处工作。我们如何仅忽略本地存储库中的特定目录,而不忽略集中存储库中的特定目录? 一些背景资料: 该项目是一个Drupal站点,该站点的文件目录一直在变化。
当我向 svn:ignore 条目添加一些东西时,它是存储在存储库中还是存储在我的本地副本中? (在 Tortoise 中,为什么添加到 svn:ignore 的项目会出现在我的更改列表中?) 最佳答
我想确保当我从 svn 结帐时,某些 ide 项目元数据文件没有更新。不幸的是,在创建项目时它已被 checkin 项目,但我有我自己的这些文件版本。 svn:ignore 是否也会在更新期间忽略文件
我有一个目录,其中包含具有各种扩展名的文件。我想在 svn:ignore 中执行与以下 .gitignore 代码等效的操作: *.* # ignore everything !.htaccess #
我在实时服务器上有一个大型项目,我想将其提交给 svn 存储库。我已使用 svn add 将所有文件和目录添加到存储库中. 问题是我想提交一个文件夹结构,但没有它的内容。文件夹名称是: /home/m
当我使用 svn:ignore 从 SVN 中排除文件时然后另一个人尝试提交排除的文件, svn 阻止他?或者他必须在他的电脑上执行相同的命令? 最佳答案 想象一下,如果您有一堆 Java 文件( *
如果已经有一个带有_id的文档,是否可以不重新索引文档?也许像MySQL中的INSERT IGNORE查询一样? 我使用批量API,并且重新索引需要很长时间,因此我只想索引丢失的文档。 可能吗? 最佳
我的房间有问题。 我正在使用带有 Gson 转换器的改造作为其余 api,我想与房间分享 pojos。一般来说它可以工作,但在某些情况下我需要忽略一些字段,因为我有对象列表。我尝试使用 @Ignore
我正在使用 vba 代码从数组中查找值。有些结果是“#N/A”,并且单元格的左上角会出现一个绿色三角形。我想自动删除绿色三角形(忽略错误),但“#N/A”结果应该保留,只需删除绿色三角形即可。 有人知
我是一名优秀的程序员,十分优秀!