-
Notifications
You must be signed in to change notification settings - Fork 862
Insert or Update
中文 | English
IFreeSql
defines the InsertOrUpdate
method, which uses the characteristics and functions of the database to implement the added or modified functions. (since 1.5.0)
Database | Features | Database | Features | |
---|---|---|---|---|
MySql | on duplicate key update | Dameng | merge into | |
PostgreSQL | on conflict do update | Kingbase ES | on conflict do update | |
SqlServer | merge into | Shentong Database | merge into | |
Oracle | merge into | GBase | merge into | |
Sqlite | replace into | MsAccess | not support | |
Firebird | merge into |
fsql.InsertOrUpdate<T>()
.SetSource(items) //Data to be processed
//.IfExistsDoNothing() //If the data exists, do nothing (that means, insert the data if and only if the data does not exist)
//.UpdateSet((a, b) => a.Count == b.Count + 10)
.ExecuteAffrows();
//or..
var sql = fsql.Select<T2, T3>()
.ToSql((a, b) => new
{
id = a.id + 1,
name = "xxx"
}, FieldAliasOptions.AsProperty);
fsql.InsertOrUpdate<T>()
.SetSource(sql)
.ExecuteAffrows();
When the entity class has auto-increment properties, batch InsertOrUpdate
can be split into two executions at most. Internally, FreeSql will calculate the data without self-increment and with self-increment, and execute the two commands of insert into
and merge into
mentioned above (using transaction execution).
Note: the common repository in FreeSql.Repository
also has InsertOrUpdate
method, but their mechanism is different.
var dic = new Dictionary<string, object>();
dic.Add("id", 1);
dic.Add("name", "xxxx");
fsql.InsertOrUpdateDict(dic).AsTable("table1").WherePrimary("id").ExecuteAffrows();
//The generated SQL is the same as above
To use this method, you need to reference the FreeSql.Repository
or FreeSql.DbContext
extensions package.
var repo = fsql.GetRepository<T>();
repo.InsertOrUpdate(YOUR_ENTITY);
State management is a data replica that is automatically generated during queries. Only when using DbContext/Repository queries can state management be implemented.
If there is data in the internal state management, then update it.
If there is no data in the internal state management, query the database to determine whether it exists.
Update if it exists, insert if it doesn't exist
Disadvantages: does not support batch operations
package name | method | desc (v3.2.693) |
---|---|---|
FreeSql.Provider.SqlServer | ExecuteSqlBulkCopy | |
FreeSql.Provider.MySqlConnector | ExecuteMySqlBulkCopy | |
FreeSql.Provider.Oracle | ExecuteOracleBulkCopy | |
FreeSql.Provider.Dameng | ExecuteDmBulkCopy | 达梦 |
FreeSql.Provider.PostgreSQL | ExecutePgCopy | |
FreeSql.Provider.KingbaseES | ExecuteKdbCopy | 人大金仓 |
Principle: Use BulkCopy to insert data into a temporary table, and then use MERGE INTO to join the table.
Tip: When the number of updated fields exceeds 3000, the benefits are greater.
fsql.InsertOrUpdate<T1>().SetSource(list).ExecuteSqlBulkCopy();
SELECT ... INTO #temp_T1 FROM [T1] WHERE 1=2
MERGE INTO [T1] t1 USING (select * from #temp_user1) t2 ON (t1.[id] = t2.[id])
WHEN MATCHED THEN
update set ...
WHEN NOT MATCHED THEN
insert (...)
values (...);
DROP TABLE #temp_user1
var list = fsql.Select<T>().Where(...).ToList();
var repo = fsql.GetRepository<T>();
repo.BeginEdit(list); //Start editing `list`
list.Add(new T { Name = "Category2_3" });
list[0].Name = "123123";
list.RemoveAt(1);
repo.EndEdit(); //override repo.EndEdit(newlist)
class T
{
public Guid Id { get; set; }
public string Name { get; set; }
}
When the above code EndEdit
method is executed, 3 SQL statements are generated, as follows:
INSERT INTO "T"("Id", "Name") VALUES('5f26bf07-6ac3-cbe8-00da-7dd74818c3a6', 'Category2_3')
UPDATE "T" SET "Name" = '123123'
WHERE ("Id" = '5f26bf00-6ac3-cbe8-00da-7dd01be76e26')
DELETE FROM "T" WHERE ("Id" = '5f26bf00-6ac3-cbe8-00da-7dd11bcf54dc')
Use case: In winform, after loading the table data, perform some operations such as adding, modifying, and deleting, and then click [Save]
Note: This operation is only valid for the variable list
, not for comparison and update of the entire table.
FreeSql.Provider.MySql
and FreeSql.Provider.MySqlConnector
support MySql's unique function On Duplicate Key Update
.
This function can also insert or update data, and supports batch operations.
class TestOnDuplicateKeyUpdateInfo {
[Column(IsIdentity = true)]
public int id { get; set; }
public string title { get; set; }
public DateTime time { get; set; }
}
var item = new TestOnDuplicateKeyUpdateInfo { id = 100, title = "title-100", time = DateTime.Parse("2000-01-01") };
fsql.Insert(item)
.NoneParameter()
.OnDuplicateKeyUpdate().ToSql();
//INSERT INTO `TestOnDuplicateKeyUpdateInfo`(`id`, `title`, `time`) VALUES(100, 'title-100', '2000-01-01 00:00:00.000')
//ON DUPLICATE KEY UPDATE
//`title` = VALUES(`title`),
//`time` = VALUES(`time`)
Methods that can be called after OnDuplicateKeyUpdate()
:
Method | Description |
---|---|
IgnoreColumns | Ignore updated columns, the mechanism is the same as IUpdate.IgnoreColumns
|
UpdateColumns | Specify updated columns, the mechanism is the same as IUpdate.UpdateColumns
|
Set | Manually specify the updated column, the same function as IUpdate.Set
|
SetRaw | As a supplement to the Set method, SQL strings can be passed in. |
ToSql | Return the SQL statement to be executed |
ExecuteAffrows | Execute and return the number of rows affected |
Both IInsert
and OnDuplicateKeyUpdate
have IgnoreColumns
and UpdateColumns
methods.
When inserting an entity or a set of entities, the time
column is ignored, the code is as follows:
fsql.Insert(item)
.IgnoreColumns(a => a.time)
.NoneParameter()
.OnDuplicateKeyUpdate().ToSql();
//INSERT INTO `TestOnDuplicateKeyUpdateInfo`(`id`, `title`) VALUES(200, 'title-200')
//ON DUPLICATE KEY UPDATE
//`title` = VALUES(`title`),
//`time` = '2000-01-01 00:00:00.000'
We found that the UPDATE time
part became a constant instead of VALUES(`time`). The mechanism is as follows:
When there are columns in the insert
part, they will be set in the form of VALUES(`field`) in the update
;
When a column that does not exist in the insert
part, it will be set as a constant in the update
. When manipulating the entity array, this constant is executed for case when ... end
(same as IUpdate
);
FreeSql.Provider.PostgreSQL
supports PostgreSQL 9.5+ unique function On Conflict(id) Do Update
.
The usage method is roughly the same as that of MySql's OnDuplicateKeyUpdate
.
class TestOnConflictDoUpdateInfo {
[Column(IsIdentity = true)]
public int id { get; set; }
public string title { get; set; }
public DateTime? time { get; set; }
}
var items = new [] {
new TestOnConflictDoUpdateInfo { id = 200, title = "title-200", time = DateTime.Parse("2000-01-01") },
new TestOnConflictDoUpdateInfo { id = 201, title = "title-201", time = DateTime.Parse("2000-01-01") },
new TestOnConflictDoUpdateInfo { id = 202, title = "title-202", time = DateTime.Parse("2000-01-01") }
};
fsql.Insert(items)
.IgnoreColumns(a => a.time)
.NoneParameter()
.OnConflictDoUpdate().ToSql();
//INSERT INTO ""testonconflictdoupdateinfo""(""id"", ""title"") VALUES(200, 'title-200'), (201, 'title-201'), (202, 'title-202')
//ON CONFLICT(""id"") DO UPDATE SET
//""title"" = EXCLUDED.""title"",
//""time"" = CASE EXCLUDED.""id""
//WHEN 200 THEN '2000-01-01 00:00:00.000000'
//WHEN 201 THEN '2000-01-01 00:00:00.000000'
//WHEN 202 THEN '2000-01-01 00:00:00.000000' END::timestamp