一、作弊方法描述
一些游戏会在某些情况下让玩家等待一段时间,例如candy crush中游戏失败会消耗一点体力,体力不满时,会每隔30分钟涨一点。体力耗尽时就无法再进行游戏,此时玩家可以修改系统时间,调到未来的时间,再回到游戏中,体力就会涨满了。游戏的这种控制就失效了。 原理:当消耗一点体力时,游戏会记录一个时刻t1,等到当前时刻为t1+30分钟时,就会加上一点体力。如果把系统时间调后半个小时,那就不需要等待直接回复体力了。
二、解决思路
1、通过开机运行时间计算开机时刻
ios和android都提供了一个方法:获取自开机到现在运行时间。就是利用这个方法来做时间校验。通过获取当前时间(有可能被修改)和开机运行时间,可以计算出开机时刻(前者减后者),如果一直保持开机状态,那么这个开机时刻应该是不变的。如果某次开机时刻算计算出来与上一次不同,那么就可以判断计时器出错。
2、记录开机时刻
第一次运行游戏时会把开机时刻记录下来,用于之后计算的对比。
3、联网时间修正
当用户时间异常,比如往后调了1个小时,计算出开机时刻与上次不同,此时需要联网修正时间。从网络上取得正确的当前时间,对比用户时间(有可能被修改),计算出时间差1个小时,记录这个时间差。之后每次获取用户当前时间的时候都要减去这个时间差,结果可以能当成正确的系统时间,再根据1的方法算出开机时刻就比较准确了。时间修正后会重设上次开机时刻做为新的参考值。
三、示例
进入游戏时,取得当前系统时间8点,获得开机运行3个小时,计算出开机时刻是5点。 游戏中在9点钟触发了一个30分钟的计时器,应该在9点半的时候完成。此时用户切到设置中修改了系统时间到10点。 再回到游戏中,会进行计时器检查,系统时间是10点,开机运行4小时,计算出开机时刻是6点,与之前记录的5点不一样,判断出计时器出错。如果用户不进行联网校正,那就无法继续操作。 用户联网后,取出网络真实时间9点,计算出用户时间差1小时(10-9),重置上次开机时刻,还是5点钟没变(如果是关机再开机这个值会变)。 校正完后让计时器继续,取用户时间会减去时间差,即可以得到真实时间,那么计时器就可以正常运行了。 如果用户保持联网状态,那么计时器出错就会瞬间被校正。
四、缺陷
对于不作弊的用户,如果机关再开机,并且没有联网,会判断计时器出错无法继续操作。需要联网校正。
获取开机时间
NSProcessInfo *info = [NSProcessInfo processInfo];
NSLog(@"%f", info.systemUptime);
NSDate *now = [NSDate date];
NSTimeInterval interval = [now timeIntervalSince1970];
NSLog(@"start time: %@", [AppDelegate getDateStrFromTimeStep:interval - info.systemUptime]);
+(NSString *)getDateStrFromTimeStep:(long long)timestep{
NSDate *timestepDate = [NSDate dateWithTimeIntervalSince1970:timestep];
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
NSTimeZone* timeZone = [NSTimeZone systemTimeZone];
[formatter setTimeZone:timeZone];
[formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
return [formatter stringFromDate:timestepDate];
}