java中的时间操作问题,难题求解

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了java中的时间操作问题,难题求解相关的知识,希望对你有一定的参考价值。

我数据库中有一批订单,都有时间,我想生产一份报表,就是JfreeChart,然后必须是把每个月都订单销售都给用柱状显示出来,

我现在的迷惑是: 如2009-1-22 到2010-2-4间隔每个月都按 2009-1 2009-2 2009-3 2009-4..... 2010-2 这样输出, 我得到了这批订单的List ,但是订单中不是每个月都有,
我求一个算法,就是得到2009-1-22 ,2010-2-4 这两个时间段, 一次打印出 2009-1 2009-2 2009-3 2009-4..... 2010-2 怎么搞?

我以前遇到过类似的问题。就是做值班表 自动排班的时候。
我不知道用SQL语句怎么做;而且又不想插入空数据,于是就用程序实现的。
其中有部分实现了时间段的构造,你可以看下。应该不难解决。
//半自动排班
@SuppressWarnings("unchecked")
public ActionForward automatic(ActionMapping mapping, ActionForm form,
HttpServletRequest request, HttpServletResponse response)
throws Exception
ScheduletableActionForm newform = (ScheduletableActionForm)form;
String id_str = request.getParameter("id");
if(id_str==null)
id_str = newform.getSchedule().getId_string();

long id = Long.parseLong(id_str);
Schedule schedule = ScheduleDAO.getScheduleById(id);
//排班明细结束时间由排班表决定
Date enddate = schedule.getEnddate();

//排班模板时间段由该排班表已有的明细决定
List datelist = SchedulelineDAO.getSchedulelineShowSizeScheduleId(id);
if(datelist==null||datelist.size()==0)
newform.setErrormsg("请至少先选择一个排班循环");


else//该datelist长度一定大于0,否则不允许进入自动排班
Date startpoint = (Date)datelist.get(0);
Date endpoint = (Date)datelist.get(datelist.size()-1);

//只排endpoint至enddate之间的明细
Date start_go = TimeFormat.getDateAfter(endpoint, 0, 0, 1, 0, 0, 0);
Date end_go = enddate;

//构造自动排班的日期列表
ArrayList<Date> datelist_go = new ArrayList<Date>();
Date tempdate = start_go;
while(!tempdate.after(end_go))
datelist_go.add(tempdate);
tempdate = TimeFormat.getDateAfter(tempdate, 0, 0, 1, 0, 0, 0);

int size = datelist_go.size();

//构造模板日期列表
ArrayList<Date> modellist = new ArrayList<Date>();
Date tempmodel = startpoint;
while(!tempmodel.after(endpoint))
modellist.add(tempmodel);
tempmodel = TimeFormat.getDateAfter(tempmodel, 0, 0, 1, 0, 0, 0);

int modelsize = modellist.size();

//班次列表
List shiftlist = ShiftDAO.getShiftlistByDeptid(schedule.getDeptid());

//按照自动排班的日期列表逐日进行
for(int i=0;i<size;i++)
int modelNo = getModelNo(i,modelsize);
String modeldate_s = TimeFormat.dateToString("yyyy-MM-dd",(Date)modellist.get(modelNo));
Date linedate = datelist_go.get(i);
for(int j=0;j<shiftlist.size();j++)//再按照班次进行
Scheduleline line = new Scheduleline();
Shift shift = (Shift)shiftlist.get(j);
long shiftid = shift.getId();
Scheduleline modelline = SchedulelineDAO.getSchedulelineTrTd(id, shiftid, modeldate_s);
if(modelline!=null)//如果不为空,复制过来进行插入
line.setId(IdCreater.getId("Scheduleline")); //id不复制
line.setDatea(linedate);
line.setMembers(modelline.getMembers());
line.setLeader(modelline.getLeader());
//line.setEndtime(modelline.getEndtime());
//line.setStarttime(modelline.getStarttime());
String datea_string = TimeFormat.dateToString("yyyy-MM-dd", linedate);
String starttime_str = datea_string+" "+TimeFormat.dateToString("HH:mm", shift.getStarttime());
Date nowdaytime = TimeFormat.stringToDate("yyyy-MM-dd", datea_string);
String endStr = shift.getEndstr();
if(endStr.equals("次日"))
nowdaytime = TimeFormat.getDateAfter(nowdaytime, 0, 0, 1, 0, 0, 0);

String endtime_str = TimeFormat.dateToString("yyyy-MM-dd", nowdaytime)+" "+TimeFormat.dateToString("HH:mm", shift.getEndtime());

line.setStarttime(TimeFormat.stringToDate("yyyy-MM-dd HH:mm", starttime_str));
line.setEndtime(TimeFormat.stringToDate("yyyy-MM-dd HH:mm", endtime_str));

line.setShiftname(modelline.getShiftname());
line.setScheduleid(modelline.getScheduleid());
line.setShiftid(modelline.getShiftid());
SchedulelineDAO.addScheduleline(line);



newform.setErrormsg("排班成功!");

newform.getSchedule().setId_string(id_str);
ActionForward forward = detail(mapping, newform, request, response);
return forward;


//计算第i行的排班表应该选择的模板的行号j
private static int getModelNo(int i,int modelsize)
return i%modelsize;

//****************************自动排班结束**************************//

补充一下:中间有个TimeFormat方法,贴出来
public class TimeFormat
//将传入的时间按照指定的格式转化成字符串返回
//注意传入的format_type必须为指定的时间格式,比如"yyyy-MM-dd HH:mm:ss"
//传入的date必须为Date型
public static String dateToString(String format_type,Date time_Date)
if(time_Date==null)return "";
SimpleDateFormat sdf = new SimpleDateFormat(format_type);
String format_time = sdf.format(time_Date);
return format_time;


//将传入的字符串转化成指定格式的日期返回
//注意传入的format_type必须为指定的时间格式,比如"yyyy-MM-dd HH:mm:ss"
//传入的time_str必须为符合format_type格式的字符串
public static Date stringToDate(String format_type,String time_Str)
SimpleDateFormat sdf = new SimpleDateFormat(format_type);
if(time_Str.length()<format_type.length())
if(time_Str.length()==4)
time_Str += "-00-00 00:00:00";

if(time_Str.length()==7)
time_Str += "-00 00:00:00";

if(time_Str.length()==10)
time_Str += " 00:00:00";

if(time_Str.length()==13)
time_Str += ":00:00";

if(time_Str.length()==16)
time_Str += ":00";


Date date = new Date();
try
date = sdf.parse(time_Str);
catch(Exception e)
e.printStackTrace();

return date;


//将传入的Date型数据转化成指定格式的Date型返回
//注意传入的format_type必须为指定的时间格式,比如"yyyy-MM-dd HH:mm:ss"
//传入的time_Date必须为Date型时间
public static Date dateToDate(String format_type,Date time_Date)
String format_str = dateToString(format_type,time_Date);
Date format_date = stringToDate(format_type,format_str);
return format_date;


public static String getCurrentDate(String format_type)
SimpleDateFormat sdf = new SimpleDateFormat(format_type);
String format_time = sdf.format(new Date());
return format_time;


//根据传入的日期取得星期,返回星期的字符串,形如"星期二"
public static String getWeekTimeByDate(Date date)
String week = "";
String[] dayNames = "星期日", "星期一", "星期二", "星期三", "星期四", "星期五","星期六";
Calendar calendar = Calendar.getInstance();
calendar.setTime(date);
int dayOfWeek = calendar.get(Calendar.DAY_OF_WEEK)-1;
if(dayOfWeek<0)
dayOfWeek=0;

week = dayNames[dayOfWeek];
return week;


//取得传入日期a年,b月,c天,d小时,e分钟,f秒以后的日期,若想得到以前的,参数为负值即可
public static Date getDateAfter(Date date,int years,int months,int days,int hours,int minutes,int seconds)
Calendar cal=Calendar.getInstance();
cal.setTime(date);
cal.add(Calendar.YEAR, years);
cal.add(Calendar.MONTH, months);
cal.add(Calendar.DATE, days);
cal.add(Calendar.HOUR, hours);
cal.add(Calendar.MINUTE, minutes);
cal.add(Calendar.SECOND, seconds);
date = cal.getTime();
return date;

//传入一个date得到该date的月
public static int getMonth(Date date)
Calendar c = Calendar.getInstance();
c.setTime(date);
return c.get(Calendar.MONTH)+1;

//传入一个date得到该date的月
public static int getDay(Date date)
Calendar c = Calendar.getInstance();
c.setTime(date);
return c.get(Calendar.DAY_OF_MONTH);

// 传入一个date得到该date的月
public static int getYear(Date date)
Calendar c = Calendar.getInstance();
c.setTime(date);
return c.get(Calendar.YEAR);


//取得A时间减去B时间后的毫秒数
public static long getPoorSeconds(Date a,Date b)
Calendar timea = Calendar.getInstance();
Calendar timeb = Calendar.getInstance();
timea.setTime(a);
timeb.setTime(b);
return timea.getTimeInMillis() - timeb.getTimeInMillis();

public static void main(String[] args)


参考技术A 假如你数据库中的日期字段是 dd 销售数量是 cc

select sum(cc) as count,to_char(dd,'yyyy-mm') as monthh
where dd>=to_date('2009-01-22','yyyy-mm-dd') and dd<=to_date('2010-02-02','yyyy-mm-dd') group by monthh

出来就是这样的格式

count monthh
24000 2009-01
434 2009-02
...
32 2009-12
..

这是在oracle中的写法,在其他库中可能没有to_char或者to_date的函数
你可以找一下其它库中类似oracle中to_char的函数...
参考技术B 主要两种做法吧,
1: 用SQL,具体来讲需要建辅助表,比如:一张表里面是所有的月份MONTH_LIST,从2009-01到2010-12,字段名:month.
然后用这张表和你的表(假设叫sale_list)left_join,假设你的表内日期字段时varchar sale_date 日期格式:'yyyy-MM-dd',销售数量是:number sale_vol ;
那么SQL是:
select m.month as month , nvl(sum (s.sale_vol), 0) as sale_vol
from month_list m
left join sale_list s on m.month = substr (s.sale_date, 1,7)
where
m.month >= '2009-01' and m.month <= '2010-02'
group by m.month order by m.month

得到的结果就是完整的。

第二种做法是用程序实现。假设JfreeChart的数据模型要求是:
interface ColumDiagramDataModel

getColumnCount () ;
getColumnName (int index) ;
getColumnValue (int index) ;

那么,可以简单的将数据加载到数组中,然后实现其接口。
在提取数据的时候,就可以将数据不全。

推荐用程序来实现。这样依赖性小,数据量也小,性能较好。
参考技术C 用flex很好实现 参考技术D 这个问题在于你统计报表用的是什么方法,就这样看可能简单的用count(*) group by column这种方法是不行了,你需要用一些方法把没有的月份做成0,我相信你应该讲到这里已经明白了。 第5个回答  2009-04-23 在对应月份放入0

以上是关于java中的时间操作问题,难题求解的主要内容,如果未能解决你的问题,请参考以下文章

A*算法求解八数码难题(python实现)

Java 内部类可见性难题

数据结构与算法之深入解析“口算难题”的求解思路与算法示例

世界难题!高分悬赏破译密码第一关!

java面试问独立解决过哪些难题

java超难题,高手帮我看一下下哪里有错误,重谢!