php连接到oracle的配置,本文不进行过多阐述,如有不懂,请看文章:Oracle11g客户端安装教程。本文主要的工作是探讨在thinkPHP框架下MySQL迁移到Oracle后数据库操作方法的改写,不讨论迁移细节。
本文使用Navicat快速将MySQL迁移到Oracle,具体操作本文不再细述,请自行搜索相关教程。
迁移后在执行新增和删除操作时,出现问题。定位到数据库方法,发现是表的问题。
用于测试的sql语句(mysql版本),导入后请利用navicat传输到Oracle。
“`sql
– phpMyAdmin SQL Dump
– version 4.0.10.11
– 主机: 127.0.0.1:3306
– 生成日期: 2018-07-02 21:53:02
– 服务器版本: 5.5.54-log
– PHP 版本: 5.6.14
SET SQL_MODE = “NO_AUTO_VALUE_ON_ZERO”;
SET time_zone = “+00:00”;
/!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT /;
/!40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS /;
/!40101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION /;
/!40101 SET NAMES utf8 /;
–
ray
–
– 表的结构 ray_user
–
CREATE TABLE IF NOT EXISTS ray_user
(
user_id
int(11) unsigned NOT NULL AUTO_INCREMENT,
user_name
varchar(10) NOT NULL,
user_pwd
varchar(40) NOT NULL,
PRIMARY KEY (user_id
)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=18 ;
–
– 转存表中的数据 ray_user
–
INSERT INTO ray_user
(user_id
, user_name
, user_pwd
) VALUES
(1, ‘updatename’, ‘ray’),
(2, ‘testname’, ‘testpwd’),
/!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT /;
/!40101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_RESULTS /;
/!40101 SET COLLATION_CONNECTION=@OLD_COLLATION_CONNECTION /;
“`
mysql环境下的CURD操作(经简单测试后增删改查无问题)
数据库配置database.php
// +----------------------------------------------------------------------
// | ThinkPHP [ WE CAN DO IT JUST THINK ]
// +----------------------------------------------------------------------
// | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
// +----------------------------------------------------------------------
// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
// +----------------------------------------------------------------------
// | Author: liu21st
// +----------------------------------------------------------------------
return [
// 数据库类型
'type' => 'mysql',
// 服务器地址
'hostname' => '127.0.0.1',
// 数据库名
'database' => 'ray',
// 用户名
'username' => 'root',
// 密码
'password' => '', // 你的密码
// 端口
'hostport' => '3306',
// 连接dsn
'dsn' => '',
// 数据库连接参数
'params' => [],
// 数据库编码默认采用utf8
'charset' => 'utf8',
// 数据库表前缀
'prefix' => 'ray_',
// 数据库调试模式
'debug' => true,
// 数据库部署方式:0 集中式(单一服务器),1 分布式(主从服务器)
'deploy' => 0,
// 数据库读写是否分离 主从式有效
'rw_separate' => false,
// 读写分离后 主服务器数量
'master_num' => 1,
// 指定从服务器序号
'slave_no' => '',
// 是否严格检查字段是否存在
'fields_strict' => true,
// 数据集返回类型
'resultset_type' => 'array',
// 自动写入时间戳字段
'auto_timestamp' => false,
// 时间字段取出后的默认时间格式
'datetime_format' => 'Y-m-d H:i:s',
// 是否需要进行SQL性能分析
'sql_explain' => false,
];
控制器User.php
namespace app\index\controller;
use think\Controller;
use app\index\model\User as US;
class User extends Controller
{
public function index() {
$obj_user = new US;
// 查找
$data = $obj_user->operateUser("find",null,"1");
var_dump($data);
// 更新
$updateData = [
'user_name' => 'updatename'
];
$result = $obj_user->operateUser("update",$updateData,"1");
var_dump($result);
// 新增
$insertData = [
'user_name' => 'testname',
'user_pwd' => 'testpwd'
];
$result = $obj_user->operateUser("insert",$insertData);
var_dump($result);
// 删除
$result = $obj_user->operateUser("delete",null,'2');
var_dump($result);
}
}
模型User.php
namespace app\index\model;
use think\Model;
class User extends Model
{
public function operateUser($directive,$data = null,$user_id = null) {
if($directive == "find" && $user_id != null) {
return User::where('user_id',$user_id)->find();
} else if($directive == "insert" && $data != null) {
return User::save($data) ? 1 : 0;
} else if($directive == "update" && $data != null && $user_id != null) {
return User::where('user_id',$user_id)->find()->save($data) ? 1 : 0;
} else if($directive == "delete" && $user_id != null) {
return User::where('user_id',$user_id)->delete() ? 1 : 0;
} else {
return null;
}
}
}
oracle环境下的CURD操作
数据库配置文件database.php
// +----------------------------------------------------------------------
// | ThinkPHP [ WE CAN DO IT JUST THINK ]
// +----------------------------------------------------------------------
// | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
// +----------------------------------------------------------------------
// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
// +----------------------------------------------------------------------
// | Author: liu21st
// +----------------------------------------------------------------------
return [
// 数据库类型
'type' => '\think\oracle\Connection',
// 服务器地址
'hostname' => '127.0.0.1',
// 数据库名
'database' => 'orcl',
// 用户名
'username' => 'Scott',
// 密码
'password' => '', // 你的密码
// 端口
'hostport' => '1521',
// 连接dsn
'dsn' => '',
// 数据库连接参数
'params' => [],
// 数据库编码默认采用utf8
'charset' => 'utf8',
// 数据库表前缀
'prefix' => 'ray_',
// 数据库调试模式
'debug' => true,
// 数据库部署方式:0 集中式(单一服务器),1 分布式(主从服务器)
'deploy' => 0,
// 数据库读写是否分离 主从式有效
'rw_separate' => false,
// 读写分离后 主服务器数量
'master_num' => 1,
// 指定从服务器序号
'slave_no' => '',
// 是否严格检查字段是否存在
'fields_strict' => true,
// 数据集返回类型
'resultset_type' => 'array',
// 自动写入时间戳字段
'auto_timestamp' => false,
// 时间字段取出后的默认时间格式
'datetime_format' => 'Y-m-d H:i:s',
// 是否需要进行SQL性能分析
'sql_explain' => false,
];
根据指定ID查询记录
由于Oracle表名和字段名均需加上双引号,故改写thinkphp\library\db\Builder.php
中的parseSqlTable
和parseWhereItem
方法。改写完成后根据ID查询记录OK。
...
/**
* 将SQL语句中的__TABLE_NAME__字符串替换成带前缀的表名(小写)
* @access protected
* @param string $sql sql语句
* @return string
*/
protected function parseSqlTable($sql)
{
return '"'. strtoupper($this->query->parseSqlTable($sql)).'"'; 前后加上双引号并将表明设置为大写
}
......
// where子单元分析
protected function parseWhereItem($field, $val, $rule = '', $options = [], $binds = [], $bindName = null)
{
// 字段分析
$key = $field ? '"'. $this->parseKey($field, $options, true) .'"' : ''; 前后加上双引号
// 查询规则和条件
if (!is_array($val)) {
$val = is_null($val) ? ['null', ''] : ['=', $val];
}
list($exp, $value) = $val;
...
改写了控制器和模型层方法:
控制器Users.php
namespace app\index\controller;
use think\Controller;
use app\index\model\Users as US;
class Users extends Controller
{
public function index() {
// 查询
$obj_users = new US;
$data = $obj_users->operateUser("find",null,"1");
var_dump($data);
// 更新
$updateData = [
'NAME' => "updateora",
'PWD' => "newpwd"
];
$result = $obj_users->operateUser("update",$updateData,"1");
var_dump($result);
// 插入
$insertData = [
'NAME' => 'testname',
'PWD' => 'testpwd'
];
$result = $obj_users->operateUser("insert",$insertData);
var_dump($result);
// 删除
$result = $obj_users->operateUser("delete",null,'18');
var_dump($result);
}
}
模型Users.php
namespace app\index\model;
use think\Model;
class Users extends Model
{
public function operateUser($directive,$data = null,$ID = null) {
if($directive == "find" && $ID != null) {
return Users::where('ID',$ID)->find();
} else if($directive == "insert" && $data != null) {
/*$id = Users::getLastInsID('SEQUSERS')-2;
var_dump($id);
$data['ID'] = $id;*/
return Users::save($data,[],'SEQUSERS') ? 1 : 0; // 注意这里传参
} else if($directive == "update" && $data != null && $ID != null) {
return Users::where('ID',$ID)->find()->save($data) ? 1 : 0;
} else if($directive == "delete" && $ID != null) {
return Users::where('ID',$ID)->delete() ? 1 : 0;
} else {
return null;
}
}
}
经测试更新数据通过,接下来是最为头疼的新增。因为MySQL主键自增通过给PK添加A-I属性即可,而Oracle则需要通过触发器来实现。下面采用了简单的实现方法。
触发器,序列实现Oracle主键自增。
CREATE OR REPLACE TRIGGER TRIUSERS BEFORE
INSERT ON SCOTT.USERS FOR EACH ROW
WHEN (
new.id is null
)
begin
select SEQUSERS.nextval into:new.id from dual;
end;
create sequence SEQUSERS
minvalue 1
maxvalue 999999999999999999999999999
start with 1
increment by 1
nocache;
需要改写think-oracle\src\Connection.php里面的getLastInsId()方法
/**
* 获取最近插入的ID
* @access public
* @param string $sequence 自增序列名
* @return string
*/
public function getLastInsID($sequence = null)
{
$pdo = $this->linkID->query("select {$sequence}.nextval as id from dual");
$pdo = $this->linkID->query("select {$sequence}.currval as id from dual");
$result = $pdo->fetchColumn();
$pdo = $this->linkID->query("alter sequence {$sequence} increment by -1");
$pdo = $this->linkID->query("select {$sequence}.nextval as id from dual");
$pdo = $this->linkID->query("alter sequence {$sequence} increment by 1");
return $result;
}
注意代码中后面三行执行语句是修复每次自增不为1的bug。