将程序由mysql迁移到Oracle实战(by quqi99)

原创 2007年10月10日 16:57:00
 
                                       将程序由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 关键字会用到很多子查询,最后,我选择了直接用 JDBC ResultSet 中的游标 (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 日期字段分 DATE TIME 两种, 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 里定义了非空字段就不容许有空的内容。按 MYSQL NOT NULL 来定义 ORACLE 表结构 , 导数据的时候会产生错误。因此导数据时要对空字符进行判断,如果为 NULL 或空字符,需要把它改成一个空格的字符串。
我的改动是,加上:
   if ( "" .equals(department))
       department= " " ;
 
版权声明:本文为博主原创文章,如需转载,请注明出处!

mysql数据库迁移至Oracle数据库

1.使用工具:(1) Navicat Premium (2) PL/SQL Developer 11.0 (3) Oracle SQL Developer 4.0.0.12.84(点击可进入下载页...
  • javaee_sunny
  • javaee_sunny
  • 2016年10月20日 17:14
  • 790

从Oracle迁移到Mysql之前必须知道的50件事

1. 对子查询的优化表现不佳.2. 对复杂查询的处理较弱3. 查询优化器不够成熟4. 性能优化工具与度量信息不足5. 审计功能相对较弱6. 安全功能不成熟,甚至可以说很粗糙.没有用户组与角色的概念,没...
  • renzhenhuai
  • renzhenhuai
  • 2015年01月04日 16:17
  • 1053

从Oracle迁移到MySQL的各种坑及自救方案

当企业内部使用的数据库种类繁杂时,或者有需求更换数据库种类时,都可能会做很多数据迁移的工作。有些迁移很简单,有些迁移可能就会很复杂,大家有没有考虑过为了顺利完成复杂的数据库迁移任务,都需要考虑并解决哪...
  • zhang123456456
  • zhang123456456
  • 2017年06月07日 20:58
  • 305

MYSQL数据库迁移到ORACLE数据库

一、环境和需求 1、环境  Mysql数据库服务器: OS version:linux 5.3 for 64 bit Mysql Server version: 5.0.45 ...
  • wwwzys
  • wwwzys
  • 2013年04月17日 13:24
  • 42902

Oracle中存储过程迁移到Mysql的问题集锦

1.关于字符串拼接问题? 例如Oracle中某段字符创拼接如下:  v_str := substr(v_resource_ids,                     1,        ...
  • QCIWYY
  • QCIWYY
  • 2016年02月25日 14:33
  • 1544

数据库迁移之从oracle 到 MySQL

数据库迁移之从oracle 到 MySQL
  • swebin
  • swebin
  • 2017年05月28日 15:46
  • 476

数据库迁移-从Oracle到MySQL

从Oracle数据库向MySQL数据库迁移中的SQL语言差异及数据库兼容性问题整理
  • pierre_
  • pierre_
  • 2017年05月24日 21:27
  • 625

sqlserver数据库迁移到mysql的方法和步骤

一.迁移方法 工具:用mysql yog,下载地址1:http://wh.uzzf.com//jxl/SQLyog_Enterprise_chs.zip 下载地址2:http://yunpan.c...
  • LT_1029
  • LT_1029
  • 2014年08月26日 17:12
  • 3801

kettle实现数据库迁移之oracle到mysql

本文介绍,如何通过kettle从oracle库迁移表到mysql库。但由于我第一次接触kettle,只知道单表复制,不知如何做批量复制,现在下面只介绍单表复制的做法,大家如果自己做过批量复制,希望可以...
  • dd276
  • dd276
  • 2016年08月01日 20:41
  • 3449

从Sql Server迁移数据到Oracle

1.迁移前准备 1.1获取工具,工具就是Oracle SQL Developer,地址:http://www.oracle.com/technetwork/developer-tools/sql-de...
  • minsenwu
  • minsenwu
  • 2015年05月08日 08:47
  • 6419
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:将程序由mysql迁移到Oracle实战(by quqi99)
举报原因:
原因补充:

(最多只允许输入30个字)