我正在尝试从Servlet将日期插入Oracle数据库的TIMESTAMP字段中。 但是当我执行插入操作时,出现错误消息:“ ORA-01843:月份无效”
我在表格中看到的日期类型是:05-FEB-19 09.36.10.000000000 AM 当表创建为TIMESTAMP类型时我声明了它。
在JSP文件中,我有以下2种输入类型:
<label>Data
<input type="date" name="date" value="13-AUG-2019" required>
</label>
<label>Time
<input type="time" name="time" value="11:00:00 AM" required>
</label>
在Servlet Java中,我有这个:
String data= request.getParameter("date");
String time= request.getParameter("time");
String paramData = data.replaceAll("\n", "");
String paramTime = time.replaceAll("\n", "");
String dataTime = paramData + " " + paramTime;
try{
Class.forName("oracle.jdbc.driver.OracleDriver");
Connection connCf = DriverManager.getConnection(url, user, pass);
PreparedStatement pstmtCf=connCf.prepareStatement("insert into screening_tab (film_oid, room_oid, data_hour) values((select ref(fi) from film_tab fi where fi.title=?), (select ref(ro) from room_tab ro where ro.code=?), ?)");
pstmtCf.setString(1, paramFilm);
pstmtCf.setString(2, paramRoom);
String paramDataTime= paramData + " " + paramTime;
pstmtCf.setString(3, paramDataTime);
Boolean result= pstmtCf.execute();
我该如何解决这个问题?
我已经在Oracle 10中尝试过该解决方案,该解决方案给出了来自servlet的字符串返回时间戳,但是问题是相同的:
create or replace procedure Sdata (datatime VarCHAR2) is
date_ho TIMESTAMP;
begin
SELECT TO_TIMESTAMP (datatime, 'DD-MON-RR HH.MI.SSXFF AM') into date_ho FROM DUAL;
dbms_output.enable;
dbms_output.put_line(date_ho);
end sdata;
在各种尝试和试验之间,我无法解释如何进行操作,如果我在Oracle SQL Developer中执行类似的操作,则一切正常!:
insert into screening_tab (film_oid, room_oid, data_hour) values((select ref(fi) from film_tab fi where fi.title='FilmNumero601'), (select ref(ro) from room_tab ro where ro.code=301), '14-DEC-19 07.00.00.00 PM' );
当我从servlet运行完全相同的查询时,我总是在无效的月份日期得到错误
我也尝试过这种解决方案:我创建了一个新的Servlet和JSP文件,其中的数据输入为:
<h4>Select Data: </h4><br></br>
<input type="datetime-local" name="date" arequired placeholder="Date">
<br></br>
<input type="reset" value="Resetta la form"></input>
<input type="submit" value="Invia"></input>
在相应的servlet中,我有这个:
PreparedStatement pstmtCf=connCf.prepareStatement("insert into screening_tab (film_oid, room_oid, data_hour) values((select ref(fi) from film_tab fi where fi.title=?), (select ref(ro) from room_tab ro where ro.code=?), ?)");
pstmtCf.setString(1, film);
pstmtCf.setString(2, room);
String date= getCorrectFormat(request.getParameter("date")+":00");
pstmtCf.setString(3, date );
pstmtCf.execute();
pstmtCf.close();
connCf.close();
getCorrectFormat()在哪里:
public static String getCorrectFormat(String date){
String out="";
out += (date.substring(8,10))+ "-";
switch(date.substring(5,7)){
case "01":
out += "JAN-";
break;
case "02":
out += "FEB-";
break;
case "03":
out += "MAR-";
break;
case "04":
out += "APR-";
break;
case "05":
out += "MAY-";
break;
case "06":
out += "JUN-";
break;
case "07":
out += "JUL-";
break;
case "08":
out += "AUG-";
break;
case "09":
out += "SEP-";
break;
case "10":
out += "OCT-";
break;
case "11":
out += "NOV-";
break;
case "12":
out += "DEC-";
break;
}
out += (date.substring(0,4))+ " ";
int hour =Integer.parseInt(date.substring(11,13))%12;
if(hour == 0)
out += String.valueOf(12);
else
out += String.valueOf(hour);
out += (date.substring(13));
if(Integer.parseInt(date.substring(11,13))<12)
out += " AM";
else
out += " PM";
return out;
}
错误始终相同:“ ORA-01843:无效月份”
答案 0 :(得分:2)
从JDBC 4.2开始,我们可以通过PreparedStatement::setObject
和ResultSet::getObject
方法与数据库直接交换 java.time 对象。在准备好的语句中,使用Oracle Tutorial中的?
占位符。
LocalDate ld = LocalDate.of( 2019 , 8 , 13 ) ;
myPreparedStatement.setObject( … , ld ) ;
然后……
LocalTime lt = LocalTime.of( 11 , 0 ) ;
myPreparedStatement.setObject( … , lt ) ;
检索。
LocalDate ld = myResultSet.getObject( … , LocalDate.class ) ;
LocalDate lt = myResultSet.getObject( … , LocalTime.class ) ;
java.time框架已内置在Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.Date
,Calendar
和SimpleDateFormat
。
目前位于Joda-Time的maintenance mode项目建议迁移到java.time类。
要了解更多信息,请参见Oracle Tutorial。并在Stack Overflow中搜索许多示例和说明。规格为JSR 310。
您可以直接与数据库交换 java.time 对象。使用符合JDBC driver或更高版本的JDBC 4.2。不需要字符串,不需要java.sql.*
类。
在哪里获取java.time类?
ThreeTen-Extra项目使用其他类扩展了java.time。该项目为将来可能在java.time中添加内容提供了一个试验场。您可能会在这里找到一些有用的类,例如Interval
,YearWeek
,YearQuarter
和more。
答案 1 :(得分:1)
您传入的是String / varchar2,并且由于日期/时间格式与Oracle默认日期/时间格式不匹配,因此隐式转换为时间戳失败。
最简单的选择是传递格式字符串,以便Oracle知道如何进行转换。不用传递时间戳?
,而是可以to_timestamp(?, 'DD-MON-YYYY HH:MI:SS AM')
如果您已经有了Java时间戳记,我想您想使用setTimestamp()
-see this question。