在android / java中,如何找到两个日期之间的百分比? 这个问题基于this question。
我想做这样的事情:
Calendar c = Calendar.getInstance();
yeartoday = c.get(Calendar.YEAR);
monthtoday = c.get(Calendar.MONTH);
daytoday = c.get(Calendar.DAY_OF_MONTH);
...
Date datestart = getDate(styear,(stmonth-1),stday);
Date dateend = getDate(year,(month-1),day);
Date datetoday = getDate(yeartoday,monthtoday,daytoday);
double percent = (((datetoday - datestart) / (dateend - datestart)) * 100);
...
public Date getDate(int year,int month,int day){
Calendar calendar = Calendar.getInstance();
calendar.set(Calendar.YEAR, year);
calendar.set(Calendar.MONTH, month);
calendar.set(Calendar.DAY_OF_MONTH, day);
Date date = calendar.getTime();
return date;
}
答案 0 :(得分:3)
( ChronoUnit.DAYS.between( start , today ) * 100 )
/
ChronoUnit.DAYS.between( start , stop )
您使用的是可怕的日期时间类,而这些类在几年前已被JSR 310中定义的 java.time 类取代。
LocalDate
LocalDate
类表示没有日期,没有time zone或offset-from-UTC的仅日期值。
时区对于确定日期至关重要。在任何给定时刻,日期都会在全球范围内变化。例如,Paris France午夜之后的几分钟是新的一天,而Montréal Québec仍然是“昨天”。
如果未指定时区,则JVM隐式应用其当前的默认时区。该默认值可能在运行时(!)期间change at any moment,因此您的结果可能会有所不同。最好将您的期望/期望时区明确指定为参数。如果紧急,请与您的用户确认区域。
以Continent/Region
的格式指定proper time zone name,例如America/Montreal
,Africa/Casablanca
或Pacific/Auckland
。切勿使用2-4个字母的缩写,例如EST
或IST
,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。
ZoneId z = ZoneId.of( "America/Montreal" ) ;
LocalDate today = LocalDate.now( z ) ;
如果要使用JVM的当前默认时区,请提出要求并作为参数传递。如果省略,代码将变得难以理解,因为我们不确定您是否打算使用默认值,还是像许多程序员一样不知道该问题。
ZoneId z = ZoneId.systemDefault() ; // Get JVM’s current default time zone.
或指定日期。您可以用数字设置月份,一月至十二月的理智编号为1-12。
LocalDate ld = LocalDate.of( 1986 , 2 , 23 ) ; // Years use sane direct numbering (1986 means year 1986). Months use sane numbering, 1-12 for January-December.
或者更好的是,使用预定义的Month
枚举对象,每年的每个月使用一个。提示:在整个代码库中使用这些Month
对象,而不是仅使用整数,可以使您的代码更具自记录性,确保有效值并提供type-safety。 Year
和YearMonth
的同上。
LocalDate ld = LocalDate.of( 1986 , Month.FEBRUARY , 23 ) ;
您似乎想要一个基于天数的百分比。要获取经过天数,请使用between
枚举类上的ChronoUnit
方法。
long days = ChronoUnit.DAYS.between( earlier , later ) ;
使用这些整数对,您可以计算百分比。
ZoneId z = ZoneId.of( "America/Montreal" ) ;
LocalDate today = LocalDate.now( z ) ;
LocalDate start = today.minusDays( 5 ) ;
LocalDate stop = today.plusDays( 15 ) ;
long totalDays = ChronoUnit.DAYS.between( start , stop ) ;
long elapsedDays = ChronoUnit.DAYS.between( start , today ) ;
long percentComplete = ( elapsedDays * 100 ) / totalDays ;
转储到控制台。
System.out.println( "start.toString(): " + start ) ;
System.out.println( "today.toString(): " + today ) ;
System.out.println( "stop.toString(): " + stop ) ;
System.out.println( "totalDays: " + totalDays ) ;
System.out.println( "elapsedDays: " + elapsedDays ) ;
System.out.println( percentComplete + "%" ) ;
请参阅此code run live at IdeOne.com。
start.toString():2019-06-04
today.toString():2019-06-09
stop.toString():2019-06-24
总天数:20
经过的天数:5
25%
当然,对于实际工作,您需要做更多的事情。
您应确认起点在终点之前。
boolean startIsBeforeStop = start.isBefore( stop ) ;
您需要验证今天是否在两个日期之间。
boolean todayIsWithinRange = ( ! today.isBefore( start ) ) && today.isBefore( stop ) ;
如果您的工作经常涉及日期范围,请将ThreeTen-Extra库添加到您的项目中。这为您提供了方便的LocalDateRange
类。
LocalDateRange range = LocalDateRange.of( start , stop ) ;
boolean startIsBeforeStop = ( range.lengthInDays() > 0 ) ; // A negative number beens the start is *not* before the stop.
boolean todayIsWithinRange = range.contains( today ) ;
java.time框架已内置在Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.Date
,Calendar
和SimpleDateFormat
。
要了解更多信息,请参见Oracle Tutorial。并在Stack Overflow中搜索许多示例和说明。规格为JSR 310。
目前位于Joda-Time的maintenance mode项目建议迁移到java.time类。
您可以直接与数据库交换 java.time 对象。使用符合JDBC driver或更高版本的JDBC 4.2。不需要字符串,不需要java.sql.*
类。
在哪里获取java.time类?
答案 1 :(得分:0)
这就是我所做的:
public void addExpenditure(Stack<Double> costStack){
try(BufferedWriter bw = new BufferedWriter(new FileWriter(f.getPath(), true))) {
while (costStack.size() != 0)
bw.write(costStack.pop().toString() + " ,");
}catch(Exception e){
e.printStackTrace();
}
}
public double availableFundsLeft(){
try{return afterTaxIncome - savingsGoalPerMonth -
getTotalExpenditureThisMonth();}
catch(Exception e){e.printStackTrace();}
return -1;
}