我有一个CodeIgniter / PHP模型,我想在数据库中插入一些数据。
但是,我在“原始”SQL查询中设置了这个:
ON DUPLICATE KEY UPDATE duplicate=duplicate+1
我正在使用CodeIgniter并将我之前的所有控制器内SQL查询转换为 ActiveRecord 。有没有办法在基于ActiveRecord的模型中执行此操作?
谢谢!
杰克
答案 0 :(得分:52)
您可以添加“ON DUPLICATE”语句而无需修改任何核心文件。
$sql = $this->db->insert_string('table', $data) . ' ON DUPLICATE KEY UPDATE duplicate=LAST_INSERT_ID(duplicate)';
$this->db->query($sql);
$id = $this->db->insert_id();
我希望它会帮助别人。
答案 1 :(得分:4)
您可以通过最少的添加来调整活动记录功能:
DB_driver.php在类中添加:
protected $OnlyReturnQuery = false;
public function onlyReturnQuery($return = true)
{
$this->OnlyReturnQuery = $return;
}
查找函数查询(...并在最开头添加:
if ($this->OnlyReturnQuery) {
$this->OnlyReturnQuery = false;
return $sql;
}
最后在DB_active_rec.php中添加功能:
public function insert_or_update($table='', $data=array())
{
$this->onlyReturnQuery();
$this->set($data);
$insert = $this->insert($table);
$this->onlyReturnQuery();
$this->set($data);
$update = $this->update($table);
$update = preg_replace('/UPDATE.*?SET/',' ON DUPLICATE KEY UPDATE',$update);
return $this->query($insert.$update);
}
现在您可以将其用作:
$ this-> db-> insert_or_update('table',array $ data);
优点: 使用所有活动记录验证 缺点: 它不是扩展功能的最佳(最合适)方式,因为如果您计划更新这些文件,则必须重做该过程。
答案 2 :(得分:3)
以下过程在Codeigniter 3.0.6中为我工作
public function updateOnDuplicate($table, $data ) {
if (empty($table) || empty($data)) return false;
$duplicate_data = array();
foreach($data AS $key => $value) {
$duplicate_data[] = sprintf("%s='%s'", $key, $value);
}
$sql = sprintf("%s ON DUPLICATE KEY UPDATE %s", $this->db->insert_string($table, $data), implode(',', $duplicate_data));
$this->db->query($sql);
return $this->db->insert_id();
}
答案 3 :(得分:2)
上面论坛帖子的链接已损坏。我不知道比调用db->查询更好的方法,如果有人有更好的解决方案,请发布。
$result = $this->CI->db->query(
"INSERT INTO tablename (id, name, duplicate) VALUES (1, 'Joe', 1) ".
"ON DUPLICATE KEY UPDATE duplicate=duplicate+1");
我希望这有助于某人寻找解决方案。
答案 4 :(得分:2)
以下是我每次都在重复更新时使用的代码段:
$sql = "insert into table_name (id, name) values(".$id.",'".$name."') on duplicate key update id=".$id.",name='".$name."'";
//Executing queries
$result = $this->db->query($sql, array($id, $name));
注意:列id
必须是主要的或唯一的
答案 5 :(得分:1)
根据Pickett链接的片段,我做了一些修改:
1)更新它以使用新的查询生成器(CI 3),以前称为活动记录。
2)您现在可以传递关联数组(key => value)或关联数组数组。在第二种形式中,它执行多次更新。
我只用mysqli测试过它,效果很好。
这段代码放在system / database / drivers / mysqli / mysqli_driver.php中
function _duplicate_insert($table, $values)
{
$updatestr = array();
$keystr = array();
$valstr = array();
foreach($values as $key => $val)
{
$updatestr[] = $key." = ".$val;
$keystr[] = $key;
$valstr[] = $val;
}
$sql = "INSERT INTO ".$table." (".implode(', ',$keystr).") ";
$sql .= "VALUES (".implode(', ',$valstr).") ";
$sql .= "ON DUPLICATE KEY UPDATE ".implode(', ',$updatestr);
return $sql;
}
function _multi_duplicate_insert($table, $values)
{
$updatestr = array();
$keystr = array();
$valstr = null;
$entries = array();
$temp = array_keys($values);
$first = $values[$temp[0]];
foreach($first as $key => $val)
{
$updatestr[] = $key." = VALUES(".$key.")";
$keystr[] = $key;
}
foreach($values as $entry)
{
$valstr = array();
foreach($entry as $key => $val)
{
$valstr[] = $val;
}
$entries[] = '('.implode(', ', $valstr).')';
}
$sql = "INSERT INTO ".$table." (".implode(', ',$keystr).") ";
$sql .= "VALUES ".implode(', ',$entries);
$sql .= "ON DUPLICATE KEY UPDATE ".implode(', ',$updatestr);
return $sql;
}
这会进入/system/database/DB_query_builder.php文件:
function on_duplicate($table = '', $set = NULL )
{
if ( ! is_null($set))
{
$this->set($set);
}
if (count($this->qb_set) == 0)
{
if ($this->db_debug)
{
return $this->display_error('db_must_use_set');
}
return FALSE;
}
if ($table == '')
{
if ( ! isset($this->qb_from[0]))
{
if ($this->db_debug)
{
return $this->display_error('db_must_set_table');
}
return FALSE;
}
$table = $this->qb_from[0];
}
$is_multi = false;
foreach (array_keys($set) as $k => $v) {
if ($k === $v) {
$is_multi = true; //is not assoc
break;
}
}
if($is_multi)
{
$sql = $this->_multi_duplicate_insert($this->protect_identifiers($table, TRUE, NULL, FALSE), $this->qb_set );
}
else
{
$sql = $this->_duplicate_insert($this->protect_identifiers($table, TRUE, NULL, FALSE), $this->qb_set );
}
$this->_reset_write();
return $this->query($sql);
}
然后你可以为单行插入/更新执行此操作:
$this->db->on_duplicate('table', array('column1' => 'value', 'column2' => 'value'));
或多次插入/更新:
$this->db->on_duplicate('table', array(
array('column1' => 'value', 'column2' => 'value'),
array('column1' => 'value', 'column2' => 'value')
));
答案 6 :(得分:0)
使用x = new()
x.add(1, 2) # returns 3, x got passed as self
和参数,我就是这样做的。前4个参数用于插入部分,后3个参数用于$this->db->query()
部分。
on duplicate key update
答案 7 :(得分:0)
简单完成-
$updt_str = '';
foreach ($insert_array as $k => $v) {
$updt_str = $updt_str.' '.$k.' = '.$v.',';
}
$updt_str = substr_replace($updt_str,";", -1);
$this->db->query($this->db->insert_string('table_name', $insert_array).' ON DUPLICATE KEY UPDATE '.$updt_str);