设为首页 收藏本站
查看: 920|回复: 0

[经验分享] 将程序由mysql迁移到Oracle实战(by quqi99)

[复制链接]

尚未签到

发表于 2016-8-6 08:59:13 | 显示全部楼层 |阅读模式
将程序由mysql迁移到Oracle实战(by quqi99)

作者:张华 发表于:2007-10-10 ( http://blog.csdn.net/quqi99 )

版权声明:可以任意转载,转载时请务必以超链接形式标明文章原始出处和作者信息及本版权声明。



1 数据迁移
首先是数据库的迁移,数据库迁移比较简单,直接用开源ETL工具Kettle搞定,非常顺利。
2 修改程序
接着要改程序,由于我不想用Hibernate,由于 mysql与Oracle的差异,改程序是必不可少的。
2.1 分页
在Mysql中分页比较简单,可以直接在SQL语句中用limit,如下:
limit "+(pg-1)*pgsz+","+pgsz; //pg为第几页,pgsz为每页大小
ORACLE处理翻页的SQL语句就比较繁琐了。每个结果集只有一个ROWNUM字段标明它的位置, 并且只能用ROWNUM<100, 不能用ROWNUM>80
由于我不想太多改已有的程序结构,用ROWNUM关键字会用到很多子查询,最后,我选择了直接用JDBCResultSet中的游标(absolute方法)去定位:
//pstmt = conn.prepareStatement(querySql);
  pstmt = conn.prepareStatement(querySql,ResultSet.TYPE_SCROLL_INSENSITIVE,ResultSet.CONCUR_READ_ONLY);
  rs = pstmt.executeQuery();
if(pg>1)
  rs.absolute((pg-1)*pgsz + 1); //分页,将记录指针指向相应位置
  int num = 0;
  while(rs.next()){
   num++;
if(num > pgsz)
break;
。。。。。。。。。其他
}
2.2 日期处理
MYSQL日期字段分DATETIME两种,ORACLE日期字段只有DATE,包含年月日时分秒信息,用当前数据库的系统时间为SYSDATE, 精确到秒,或者用字符串转换成日期型函数TO_DATE(‘2001-08-01’,’YYYY-MM-DD’)-- 24小时:分钟:的格式YYYY-MM-DD HH24:MI:SS TO_DATE()还有很多种日期格式, 可以参看ORACLE DOC.日期型字段转换成字符串函数TO_CHAR(‘2001-08-01’,’YYYY-MM-DD HH24:MI:SS’)
我使用了TO_CHAR函数,改动如下:
//querySql = "SELECT y.id,y.bumen,b.bumen,xingming,date_format(riqi,'%Y-%m-%d'),yijian,zhuangtai,huifu,manyi FROM yijian y,bumen b where y.bumen=b.id and "+qStr+" order by y.id desc " + lStr;
querySql = "SELECT y.id,y.bumen,b.bumen,xingming,TO_CHAR(riqi,'YYYY-MM-DD'),yijian,zhuangtai,huifu,manyi FROM yijian y,bumen b where y.bumen=b.id and "+qStr+" order by y.id desc ";

另外,在mysql数据库中在SQL语句中用sysdate()函数取系统当前日期,oracle数据库中在SQL语句中用SYSDATE函数取系统当前日期
2.3 CLOB大字段处理
Kettle工具能自动将mysql的TEXT字段转成ORACLE中的CLOB字段,所以在操作大字段的地方需要改动。
读取大字段的改动如下:
//读取clob字段
//CLOB yijianClob = ((OracleResultSet)rs).getCLOB(6);
//java.lang.ClassCastException: org.apache.tomcat.dbcp.dbcp.DelegatingResultSet错误
//应该将数据源配置中加上属性:accessToUnderlyingConnectionAllowed="true"
//如果是在Postgresql中,类似就应该是:PGConnection pgCon =(PGConnection)((DelegatingConnection)con).getInnermostDelegate();
CLOB yijianClob = (oracle.sql.CLOB)((org.apache.tomcat.dbcp.dbcp.DelegatingResultSet)rs).getClob(6);
String yijian="";
if(yijianClob!=null){
Reader is=yijianClob.getCharacterStream();
BufferedReader br=new BufferedReader(is);
String s=br.readLine();
while(s!=null){
  yijian+=s+"/n";
  s=br.readLine();
}
}
注意:应该将数据源配置中加上属性:accessToUnderlyingConnectionAllowed="true",并且使用org.apache.tomcat.dbcp.dbcp.DelegatingResultSet,不然会报错。
插入大字段如下:
//插入留言
conn = dataSource.getConnection();
//querySql = "insert into yijian (riqi,xingming,bumen,gongkai,lianxi,leixing,yijian,huifu,zhuangtai,manyi,pingjia,jiaoyan,district,tel)"
// + " values(sysdate(),'"+xingming+"',"+bmid+","+gongkai+",'"+lianxi+"',"+leixing+",'"+yijian+"','',0,0,'',"+j+",'"+district+"','"+tel+"')";
querySql = "insert into yijian (id,riqi,xingming,bumen,gongkai,lianxi,leixing,yijian,huifu,zhuangtai,manyi,pingjia,jiaoyan,district,tel)"
  + " values(?,SYSDATE,'"+xingming+"',"+bmid+","+gongkai+",'"+lianxi+"',"+leixing+",empty_clob(),empty_clob(),0,0,'',"+j+",'"+district+"','"+tel+"')";
try {
System.out.println("将要执行: "+querySql);
int id=getNextPublicId("yijian");
   pstmt = conn.prepareStatement(querySql);
   pstmt.setInt(1,id); //设置主键
  i = pstmt.executeUpdate();
  //插入大字段
  ResultSet rsClob = null;
  //用这种容易发生错误ORA-01006: 赋值变量不存在,改成下句带?的形式
  //ResultSet rsClob = pstmt.executeQuery("select yijian from yijian where id="+id+" for update");
  PreparedStatement pstmttemp = conn.prepareStatement("select yijian from yijian where id=? for update");
  pstmttemp.setInt(1,id);
  rsClob = pstmttemp.executeQuery();  
  if(rsClob.next()){
   //避免发生错误java.lang.ClassCastException: org.apache.tomcat.dbcp.dbcp.DelegatingResultSet,改用下句
   //CLOB yijianClob = ((OracleResultSet)rsClob).getCLOB(1);
   CLOB yijianClob = (oracle.sql.CLOB)((org.apache.tomcat.dbcp.dbcp.DelegatingResultSet)rsClob).getClob(1);
   //CLOB huifuClob = (oracle.sql.CLOB)((org.apache.tomcat.dbcp.dbcp.DelegatingResultSet)rsClob).getClob(2);
   yijianClob.putString(1,yijian);
   huifuClob.putString(1,"");
   PreparedStatement pstmtClob=conn.prepareStatement("update yijian set yijian=? where id='"+id+"' ");
   pstmtClob.setClob(1,yijianClob);
   //pstmtClob.setClob(2,huifuClob);
   pstmtClob.executeUpdate();
   pstmtClob.close();
  }
  rsClob.close();
  rsClob = null;
  pstmttemp.close();
   }
  catch (SQLException ex) {
   System.err.println("插入留言失败! "+ex.getMessage());
   throw ex;
  } finally {
try {
if (rs != null) {
rs.close();
rs = null;
}
} catch (SQLException ex1) {
}
  try {
if (pstmt != null) {
pstmt.close();
pstmt = null;
}
} catch (SQLException ex2) {
}
try {
if (conn != null) {
conn.close();
conn = null;
}
} catch (SQLException ex3) {
}
  }
2.4自动增长的数据类型处理
MYSQL有自动增长的数据类型,插入记录时不用操作此字段,会自动获得数据值。ORACLE没有自动增长的数据类型,需要建立一个自动增长的序列号,插入记录时要把序列号的下一个值赋于此字段。
CREATE SEQUENCE 序列号的名称 (最好是表名+序列号标记) INCREMENT BY 1 START WITH 1000 MAXVALUE 99999 CYCLE NOCACHE;
其中最大的值按字段的长度来定, 如果定义的自动增长的序列号 NUMBER(6) , 最大值为999999
INSERT
语句插入这个字段值为: 序列号的名称.NEXTVAL
我的改动如下,首先新建一个名为lyb_seq的序列,相应SQL语句如下:
CREATE SEQUENCE "LYB"."LYB_SEQ" INCREMENT BY 1 START WITH 1
MAXVALUE 1.0E28 MINVALUE 1 CYCLE
NOCACHE NOORDER
然后SQL语句改动如下:
// pstmt = conn.prepareStatement("insert into bumen (bumen) values ('其他')");
   pstmt = conn.prepareStatement("insert into bumen (id,bumen) values ('LYB_SEQ.NEXTVAL','其他')");

另外,在SQL语句中取序列方法为:select lyb_seq.nextval from DUAL;
2.5 表名
在Oracle中的表名不能叫user,否则会报错误:ORA-00903: 表名无效
2.6 空字符的处理
MYSQL的非空字段也有空的内容,ORACLE里定义了非空字段就不容许有空的内容。按MYSQLNOT NULL来定义ORACLE表结构, 导数据的时候会产生错误。因此导数据时要对空字符进行判断,如果为NULL或空字符,需要把它改成一个空格的字符串。
我的改动是,加上:
if("".equals(department))
department=" ";

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-253564-1-1.html 上篇帖子: oracle 11g RAC 的一些基本概念(一) 下篇帖子: Oracle的EM(OracleDBConsoleorcl服务)无法启动的问题解决
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表