-1

实际上我想禁用一个按钮 30 分钟。

下面的代码运行良好,但如果我们终止应用程序并在 30 分钟内打开它,则该按钮现在已启用。

public static void timmer(final ImageButton btn){
    new Handler().postDelayed(new Runnable() {
        @Override
        public void run() {
            isTimerRunning = false;
            btn.setEnabled(true);
        }

    }, 1800000);
}

现在我在单击按钮时获取系统时间,然后将其添加 30 分钟并将其保存到DataManager(用户对象的单个类)。再次单击按钮,我将保存时间与当前系统时间进行比较。

 @OnClick(R.id.generalReportBtn)
    public void setGeneralReport(ImageButton button) {
    Log.e("Data Manager Time: ",DataManager.getInstance().getCurrentUser().getLastTime().MINUTE+"");
    Log.e("Current Time: ", java.util.Calendar.getInstance().MINUTE+"");
    if(DataManager.getInstance().getCurrentUser().getLastTime().after(java.util.Calendar.getInstance())){
      DataManager.getInstance().getCurrentUser().lastTime.set(java.util.Calendar.getInstance().YEAR,java.util.Calendar.getInstance().MONTH,java.util.Calendar.getInstance().DATE,java.util.Calendar.getInstance().HOUR_OF_DAY,java.util.Calendar.getInstance().MINUTE,java.util.Calendar.getInstance().SECOND);
    DataManager.getInstance().getCurrentUser().lastTime.add(DataManager.getInstance().getCurrentUser().lastTime.MINUTE,30);

    }else if(Utilities.isFirstTimePressed){
        Utilities.isFirstTimePressed = false;
        DataManager.getInstance().getCurrentUser().lastTime.set(java.util.Calendar.getInstance().YEAR,java.util.Calendar.getInstance().MONTH,java.util.Calendar.getInstance().DATE,java.util.Calendar.getInstance().HOUR_OF_DAY,java.util.Calendar.getInstance().MINUTE,java.util.Calendar.getInstance().SECOND);
    DataManager.getInstance().getCurrentUser().lastTime.add(DataManager.getInstance().getCurrentUser().lastTime.MINUTE,30);

    }else {
        Toast.makeText(mainclass,"Button is disabled for 30 minutes",Toast.LENGTH_SHORT).show();
    }
}

但是这段代码对我也不起作用。

4

3 回答 3

2

首先设置一个变量。然后您的代码将更易于阅读:

java.util.Calendar cal = java.util.Calendar.getInstance();
User user = DataManager.getInstance().getCurrentUser();

然后使用这些变量而不是多次调用 Calendar.getInstance() 。

您正在混淆常量和值。YEAR 是一个常量,当您想要年份值时使用该常量。

而不是这个(它总是返回相同的值,因为它是一个常量):

java.util.Calendar.getInstance().YEAR

用这个:

cal.get(Calendar.YEAR)

为你所有的年、月等做这件事,一切都会奏效。

于 2017-10-05T17:47:22.917 回答
2

Calendar.getInstance().YEAR表示年份字段的常量(其值等于1)。其实和直接调用是一样的Calendar.YEAR

但这不是对象的年份值Calendar。当您这样做时,set(java.util.Calendar.getInstance().YEAR您将年份设置为1(这可能不是您想要的)。这同样适用于MONTH以及MINUTE您正在使用的所有其他常量。

另一个细节是您不需要一直创建另一个日历(通过调用Calendar.getInstance())。这在您的代码中是多余的并且不是必需的。

无论如何,第一件事。在前 2 行中,您打印的是MINUTE常量,而不是日期的分钟值。如果你想打印完整的日期(这比分钟更有用,IMO),只需调用该getTime()方法。此外,在使用它们之前将值设置为变量,它使代码更清晰:

import java.util.Calendar;

Calendar lastTime = DataManager.getInstance().getCurrentUser().getLastTime();
Calendar now = Calendar.getInstance(); // current date/time
Log.e("Data Manager Time: ", lastTime.getTime() + "");
Log.e("Current Time: ", now.getTime() + "");

只是提醒一下,这将打印设备的 JVM 默认时区中的日期。如果您想更改格式,只需检查此链接

如果您仍想只打印分钟,请将调用更改getTime()get(Calendar.MINUTE)

现在你的其余逻辑(如果我理解正确的话)将是这样的:

if (lastTime.after(now)) {
    lastTime = (Calendar) now.clone(); // lastTime becomes a copy of "now"
    lastTime.add(Calendar.MINUTE, 30); // add 30 minutes
} else if (Utilities.isFirstTimePressed) {
    Utilities.isFirstTimePressed = false;
    lastTime.add(Calendar.MINUTE, 30); // add 30 minutes
} else {
    Toast.makeText(mainclass, "Button is disabled for 30 minutes", Toast.LENGTH_SHORT).show();
}

Java 新的日期/时间 API

旧的类(DateCalendarSimpleDateFormat很多问题设计问题,它们正在被新的 API 取代。

在 Android 中,您可以使用ThreeTen Backport,它是 Java 8 新日期/时间类的一个很好的后向端口。要使其工作,您还需要ThreeTenABP(更多关于如何使用它的信息)。

检查 30 分钟是否已过的一种方法是使用org.threeten.bp.Instant课程。然后你用isAfter方法来比较,aorg.threeten.bp.temporal.ChronoUnit加30分钟。

我还使用 aorg.threeten.bp.ZoneId将日期转换为 JVM 默认时区(只是为了打印它)。在此链接中,还有一些答案涵盖了新 API,使用 a DateTimeFormatter,因此如果您想更改格式,可以参考它们。

我还使用org.threeten.bp.DateTimeUtils该类从和转换为java.util.Calendar. 代码将是这样的:

// convert java.util.Calendar to org.threeten.bp.Instant
Instant lastTime = DateTimeUtils.toInstant(DataManager.getInstance().getCurrentUser().getLastTime());
Instant now = Instant.now();
Log.e("Data Manager Time: ", lastTime.atZone(ZoneId.systemDefault()) + "");
Log.e("Current Time: ", now.atZone(ZoneId.systemDefault())  + "");
if (lastTime.isAfter(now)) {
    // add 30 minutes to current date/time
    lastTime = now.plus(30, ChronoUnit.MINUTES);
} else if (Utilities.isFirstTimePressed) {
    Utilities.isFirstTimePressed = false;
    lastTime = lastTime.plus(30, ChronoUnit.MINUTES);
} else {
    Toast.makeText(mainclass, "Button is disabled for 30 minutes", Toast.LENGTH_SHORT).show();
}

// convert back to java.util.Calendar (set it anywhere you need)
DateTimeUtils.toGregorianCalendar(lastTime.atZone(ZoneId.systemDefault()));

我正在使用ZoneId.systemDefault(),它采用 JVM 默认时区(创建时使用的时区相同Calendar.getInstance(),因此通常不会有太大问题,除非用户更改它)。

问题是可以更改默认时区,即使在运行时也是如此,因此如果可能的话,最好指定一个。API 使用IANA 时区名称(始终采用格式Region/City,例如America/Sao_PauloEurope/Berlin)。避免使用 3 个字母的缩写(如CSTor PST),因为它们模棱两可且不标准

您可以通过调用获取可用时区列表(并选择最适合您系统的时区)ZoneId.getAvailableZoneIds()

于 2017-10-05T17:48:28.283 回答
0

简单的方法是获取系统时间戳,例如:

long systemTime = System.currentTimeMillis()

再加上 30 分钟。

systemTime += 180000 //(180000 miliseconds is equivalent to 30 minutes)

将其存储到共享首选项。现在从共享偏好中获取值并与当前时间进行比较。

于 2017-10-06T04:24:22.617 回答