瀏覽代碼

sqlite支持优化处理

tags/6.5.6
tianya 1 周之前
父節點
當前提交
d5a8ae0cc2
共有 5 個文件被更改,包括 138 次插入288 次删除
  1. +1
    -16
      src/install/index.php
  2. +2
    -0
      src/install/install.html
  3. +0
    -213
      src/install/update.txt
  4. +119
    -7
      src/system/common.func.php
  5. +16
    -52
      src/system/database/dedesqlite.class.php

+ 1
- 16
src/install/index.php 查看文件

@@ -170,22 +170,7 @@ else if ($step==2) {
$query .= $line."\n";
$query = str_replace('#@__', $dbprefix, $query);
if ($dbtype == 'sqlite') {
$query = preg_replace('/character set (.*?) /i','', $query);
$query = preg_replace('/unsigned/i','', $query);
$query = str_replace('TYPE=MyISAM','', $query);
$query = preg_replace ('/TINYINT\(([\d]+)\)/i','INTEGER', $query);
$query = preg_replace ('/mediumint\(([\d]+)\)/i','INTEGER', $query);
$query = preg_replace ('/smallint\(([\d]+)\)/i','INTEGER', $query);
$query = preg_replace('/int\(([\d]+)\)/i','INTEGER', $query);
$query = preg_replace('/auto_increment/i','PRIMARY KEY AUTOINCREMENT', $query);
$query = preg_replace('/,([\t\s ]+)KEY(.*?)MyISAM;/','', $query);
$query = preg_replace('/,([\t\s ]+)KEY(.*?);/',');', $query);
$query = preg_replace('/,([\t\s ]+)UNIQUE KEY(.*?);/',');', $query);
$query = preg_replace('/set\(([^\)]*?)\)/','varchar', $query);
$query = preg_replace('/enum\(([^\)]*?)\)/','varchar', $query);
if (preg_match("/PRIMARY KEY AUTOINCREMENT/", $query)) {
$query = preg_replace('/,([\t\s ]+)PRIMARY KEY([\t\s ]+)\(`([0-9a-zA-Z]+)`\)/i','', $query);
}
$query = ConvertMysqlToSqlite($query);
$db->exec($query);
} else {
if (preg_match('#CREATE#i', $query)) {


+ 2
- 0
src/install/install.html 查看文件

@@ -113,8 +113,10 @@
$("#dbtype").change(function() {
if ($(this).val() === 'sqlite') {
$(".form-group.server").hide();
$("#dbpwd").prop("required", false);
} else {
$(".form-group.server").show();
$("#dbpwd").prop("required", true);
}
});
</script>


+ 0
- 213
src/install/update.txt
文件差異過大導致無法顯示
查看文件


+ 119
- 7
src/system/common.func.php 查看文件

@@ -365,14 +365,24 @@ function ShowMsg($msg, $gourl, $onlymsg = 0, $limittime = 0)
*/
function TableHasField($tablename, $field)
{
global $dsql;
$dsql->GetTableFields($tablename,"tfd");
while ($r = $dsql->GetFieldObject("tfd")) {
if ($r->name === $field) {
return true;
global $dsql,$cfg_dbtype;
if ($cfg_dbtype === 'mysql') {
$dsql->GetTableFields($tablename,"tfd");
while ($r = $dsql->GetFieldObject("tfd")) {
if ($r->name === $field) {
return true;
}
}
return false;
} elseif ($cfg_dbtype === 'sqlite') {
$k = $dsql->GetTableFields($tablename);
while ($r = $dsql->GetFieldObject($k)) {
if ($r->name === $field) {
return true;
}
}
return false;
}
return false;
}
function GetSimpleServerSoftware()
{
@@ -459,7 +469,7 @@ function GetUpdateSQL()
$result = array();
$query = '';
$sql4tmp = "ENGINE=MyISAM DEFAULT CHARSET=".$cfg_db_language;
$fp = fopen(DEDEROOT.'/install/update.txt','r');
$fp = fopen(DEDEDATA.'/admin/update.txt','r');
$sqls = array();
$current_ver = '';
while(!feof($fp))
@@ -692,6 +702,108 @@ function DedeSearchAPIURL($action, $parms=array())

return $finalUrl;
}

function ConvertMysqlToSqlite($mysqlQuery) {
// 移除 CHARACTER SET 和 COLLATE
$query = preg_replace('/character set \S+/i', '', $mysqlQuery);
$query = preg_replace('/collate \S+/i', '', $query);
// 移除 unsigned 关键字
$query = str_replace('unsigned', '', $query);
// 替换 MySQL 的整数类型为 SQLite 的 INTEGER
$query = preg_replace('/\b(TINY|SMALL|MEDIUM|BIG)?INT\(\d+\)/i', 'INTEGER', $query);
// 替换 AUTO_INCREMENT 为 PRIMARY KEY AUTOINCREMENT (仅适用于 INTEGER 类型)
$query = preg_replace('/\bINTEGER\s+NOT NULL\s+PRIMARY KEY AUTOINCREMENT/i', 'INTEGER PRIMARY KEY AUTOINCREMENT', $query);
$query = preg_replace('/\bAUTO_INCREMENT\b/i', '', $query);
// 移除 MySQL 特有的 ENGINE、CHARSET、COLLATE、TYPE 选项
$query = preg_replace('/ENGINE=\S+/i', '', $query);
$query = preg_replace('/DEFAULT CHARSET=\S+/i', '', $query);
$query = preg_replace('/COLLATE=\S+/i', '', $query);
$query = preg_replace('/TYPE=MyISAM;/i', '', $query);
// 移除 COMMENT 语法(SQLite 不支持)
$query = preg_replace('/COMMENT\s+\'[^\']*\'/i', '', $query);
// 移除 KEY 和 UNIQUE KEY 定义(SQLite 会自动管理索引),同时处理 USING BTREE
$query = preg_replace('/,?\s*KEY\s+\S+\s*\([^)]*\)\s*(USING BTREE)?/i', '', $query);
$query = preg_replace('/,?\s*UNIQUE KEY\s+\S+\s*\([^)]*\)\s*(USING BTREE)?/i', '', $query);
// 移除不完整的 UNIQUE 定义
$query = preg_replace('/,?\s*UNIQUE\s*(?!\()/', '', $query);
// 替换 ENUM 和 SET 为 TEXT
$query = preg_replace('/\b(ENUM|SET)\([^)]*\)/i', 'TEXT', $query);
// 替换 MEDIUMTEXT 为 TEXT
$query = preg_replace('/\bMEDIUMTEXT\b/i', 'TEXT', $query);
// 处理 DEFAULT CURRENT_TIMESTAMP
$query = preg_replace('/DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP/i', 'DEFAULT CURRENT_TIMESTAMP', $query);
// 处理 DEFAULT 值
$query = preg_replace('/DEFAULT\s+\'([^\']+)\'/i', 'DEFAULT "$1"', $query);
// 处理 PRIMARY KEY 只能用于 INTEGER
if (preg_match('/PRIMARY KEY \(`(\w+)`\)/', $query, $matches)) {
$primaryKeyColumn = $matches[1];
$query = preg_replace('/,?\s*PRIMARY KEY\s*\(`' . $primaryKeyColumn . '`\)/i', '', $query);
$query = preg_replace('/(`' . $primaryKeyColumn . '`\s+INTEGER)/i', '$1 PRIMARY KEY', $query);
}
// 处理 CONCAT 替换为 SQLite 兼容形式
if (preg_match('/CONCAT\(([^)]*?)\)/i', $query, $matches)) {
$query = preg_replace('/CONCAT\(([^)]*?)\)/i', str_replace(",", "||", $matches[1]), $query);
$query = str_replace("'||'", "','", $query);
}
// 修正 FIND_IN_SET 替换
$query = preg_replace("/FIND_IN_SET\('([\w]+)', arc.flag\)>0/i", "(',' || arc.flag || ',') LIKE '%,\\1,%'", $query);
$query = preg_replace("/FIND_IN_SET\('([\w]+)', arc.flag\)<1/i", "(',' || arc.flag || ',') NOT LIKE '%,\\1,%'", $query);

// 修正 FIND_IN_SET 替换(允许列名包含点号)
$query = preg_replace_callback(
"/FIND_IN_SET\s*\(\s*'([^']+)'\s*,\s*([a-zA-Z0-9_`\.]+)\s*\)/i",
function ($matches) {
// 返回 SQLite 兼容的 LIKE 语法
return "(',' || " . $matches[2] . " || ',' LIKE '%," . $matches[1] . ",%')";
},
$query
);
// 替换 FIELD 函数为 CASE 表达式
$query = preg_replace_callback(
'/\bFIELD\s*\(\s*([^,]+)\s*,\s*((?:\'[^\']+\'|`[^`]+`|[^),]+(?:,\s*)?)+)\s*\)/i',
function ($matches) {
$field = trim($matches[1]);
$values = trim($matches[2]);
// 更精确分割值列表(支持带引号、反引号及无空格分隔的数值)
preg_match_all('/\'[^\']+\'|`[^`]+`|\d+|\w+/', $values, $valueParts);
$cases = [];
$position = 1;
foreach ($valueParts[0] as $value) {
$cases[] = "WHEN $field = $value THEN $position";
$position++;
}
return "(CASE " . implode(' ', $cases) . " ELSE 0 END)";
},
$query
);
// 新增的转换逻辑
$query = preg_replace("/SHOW fields FROM `([\w]+)`/i", "PRAGMA table_info('\\1') ", $query);
$query = preg_replace("/SHOW CREATE TABLE `([\w]+)`/i", "SELECT 0,sql FROM sqlite_master WHERE name='\\1'; ", $query);
$query = preg_replace("/Show Tables/i", "SELECT name FROM sqlite_master WHERE type = \"table\"", $query);
$query = str_replace("\'", "\"", $query);
$query = str_replace('\t\n', "", $query);
$query = str_ireplace('rand', 'RANDOM', $query);
return trim($query);
}

//自定义函数接口
if (file_exists(DEDEINC.'/extend.func.php')) {
require_once(DEDEINC.'/extend.func.php');


+ 16
- 52
src/system/database/dedesqlite.class.php 查看文件

@@ -56,7 +56,7 @@ class DedeSqlite
var $isInit = false;
var $pconnect = false;
var $_fixObject;
var $_fieldIdx = 1; //这里最好是数组,对应id,但由于用的地方不多,暂时先这样处理
var $_fieldIdx = array(); //这里最好是数组,对应id,但由于用的地方不多,暂时先这样处理
//用外部定义的变量初始类,并连接数据库
function __construct($pconnect = FALSE, $nconnect = FALSE)
{
@@ -253,7 +253,7 @@ class DedeSqlite
CheckSql($this->queryString);
}
$t1 = ExecTime();
//var_dump($this->queryString);
// var_dump($this->queryString);
$this->result[$id] = $this->linkID->query($this->queryString);
if (!$this->result[$id]) {
$this->DisplayError("执行SQL错误:{$this->linkID->lastErrorMsg()}");
@@ -386,7 +386,7 @@ class DedeSqlite
return $sqlite_version;
}
//获取特定表的信息
function GetTableFields($tbname, $id = "me")
function GetTableFields($tbname)
{
global $dsqlite;
if (!$dsqlite->isInit) {
@@ -395,29 +395,27 @@ class DedeSqlite
$prefix = "#@__";
$tbname = str_replace($prefix, $GLOBALS['cfg_dbprefix'], $tbname);
$query = "SELECT * FROM {$tbname} LIMIT 1";
$this->result[$id] = $this->linkID->query($query);
$result = $this->linkID->query($query);
$key = spl_object_hash($result);
$this->result[$key] = $result;
$this->_fieldIdx[$key] = 0;
return $key ;
}
//获取字段详细信息
function GetFieldObject($id = "me")
{
if (!$this->result[$id]) {
if (!isset($this->result[$id]) || !$this->result[$id]) {
return false;
}
$cols = $this->result[$id]->numColumns();
if ($this->_fieldIdx >= $cols) {
$this->_fieldIdx = 1;
if ($this->_fieldIdx[$id] >= $cols) {
return false;
}
for ($i = 1; $i <= $cols; $i++) {
$field = new stdClass;
$n = $this->result[$id]->columnName($i);
$field->name = $n;
if ($this->_fieldIdx === $i) {
$this->_fieldIdx++;
return $field;
}
}
return false;
$n = $this->result[$id]->columnName($this->_fieldIdx[$id]);
$field = new stdClass();
$field->name = $n;
$this->_fieldIdx[$id]++;
return $field;
}
//获得查询的总记录数
function GetTotalRow($id = "me")
@@ -460,41 +458,7 @@ class DedeSqlite
{
$prefix = "#@__";
$sql = str_replace($prefix, $GLOBALS['cfg_dbprefix'], $sql);
$this->queryString = $sql;
//$this->queryString = preg_replace("/CONCAT\(',', arc.typeid2, ','\)/i","printf(',%s,', arc.typeid2)",$this->queryString);
if (preg_match("/CONCAT\(([^\)]*?)\)/i", $this->queryString, $matches)) {
$this->queryString = preg_replace("/CONCAT\(([^\)]*?)\)/i", str_replace(",", "||", $matches[1]), $this->queryString);
$this->queryString = str_replace("'||'", "','", $this->queryString);
}
$this->queryString = preg_replace("/FIND_IN_SET\('([\w]+)', arc.flag\)>0/i", "(',' || arc.flag || ',') LIKE '%,\\1,%'", $this->queryString);
$this->queryString = preg_replace("/FIND_IN_SET\('([\w]+)', arc.flag\)<1/i", "(',' || arc.flag || ',') NOT LIKE '%,\\1,%'", $this->queryString);
if (preg_match("/CREATE TABLE/i", $this->queryString)) {
$this->queryString = preg_replace("/[\r\n]/", '', $this->queryString);
$this->queryString = preg_replace('/character set (.*?) /i', '', $this->queryString);
$this->queryString = preg_replace('/unsigned/i', '', $this->queryString);
$this->queryString = str_replace('TYPE=MyISAM', '', $this->queryString);
$this->queryString = preg_replace('/TINYINT\(([\d]+)\)/i', 'INTEGER', $this->queryString);
$this->queryString = preg_replace('/mediumint\(([\d]+)\)/i', 'INTEGER', $this->queryString);
$this->queryString = preg_replace('/smallint\(([\d]+)\)/i', 'INTEGER', $this->queryString);
$this->queryString = preg_replace('/int\(([\d]+)\)/i', 'INTEGER', $this->queryString);
$this->queryString = preg_replace('/auto_increment/i', 'PRIMARY KEY AUTOINCREMENT', $this->queryString);
$this->queryString = preg_replace('/, KEY(.*?)MyISAM;/i', '', $this->queryString);
$this->queryString = preg_replace('/, KEY(.*?);/i', ');', $this->queryString);
$this->queryString = preg_replace('/, UNIQUE KEY(.*?);/i', ');', $this->queryString);
$this->queryString = preg_replace('/set\(([^\)]*?)\)/', 'varchar', $this->queryString);
$this->queryString = preg_replace('/enum\(([^\)]*?)\)/', 'varchar', $this->queryString);
if (preg_match("/PRIMARY KEY AUTOINCREMENT/", $this->queryString)) {
$this->queryString = preg_replace('/,([\t\s ]+)PRIMARY KEY \(`([0-9a-zA-Z]+)`\)/i', '', $this->queryString);
$this->queryString = str_replace(', PRIMARY KEY (`id`)', '', $this->queryString);
}
}
$this->queryString = preg_replace("/SHOW fields FROM `([\w]+)`/i", "PRAGMA table_info('\\1') ", $this->queryString);
$this->queryString = preg_replace("/SHOW CREATE TABLE .([\w]+)/i", "SELECT 0,sql FROM sqlite_master WHERE name='\\1'; ", $this->queryString);
//var_dump($this->queryString);
$this->queryString = preg_replace("/Show Tables/i", "SELECT name FROM sqlite_master WHERE type = \"table\"", $this->queryString);
$this->queryString = str_replace("\'", "\"", $this->queryString);
$this->queryString = str_replace('\t\n', "", $this->queryString);
$this->queryString = str_ireplace('rand', 'RANDOM', $this->queryString);
$this->queryString = ConvertMysqlToSqlite($sql);
//var_dump($this->queryString);
}
function SetSql($sql)


Loading…
取消
儲存