Наиболее эффективный способ обновления LINQ to SQL
Могу ли я обновить свою запись сотрудника, как указано в приведенной ниже функции, или мне нужно сначала сделать запрос коллекции сотрудников, а затем обновить данные?
public int updateEmployee(App3_EMPLOYEE employee)
{
DBContextDataContext db = new DBContextDataContext();
db.App3_EMPLOYEEs.Attach(employee);
db.SubmitChanges();
return employee.PKEY;
}
Или мне нужно сделать следующее?
public int updateEmployee(App3_EMPLOYEE employee)
{
DBContextDataContext db = new DBContextDataContext();
App3_EMPLOYEE emp = db.App3_EMPLOYEEs.Single(e => e.PKEY == employee.PKEY);
db.App3_EMPLOYEEs.Attach(employee,emp);
db.SubmitChanges();
return employee.PKEY;
}
Но я не хочу использовать второй вариант. Есть ли эффективный способ обновления данных?
Я получаю эту ошибку, используя оба способа:
Была предпринята попытка вставить или добавить объект, который не является новым, возможно, был загружен из другого DataContext. Это не поддерживается.
Ответы
Ответ 1
Я нашел следующую проблему для этой проблемы:
1) fetch и update entity (я буду использовать этот путь, потому что это нормально для меня)
public int updateEmployee(App3_EMPLOYEE employee)
{
AppEmployeeDataContext db = new AppEmployeeDataContext();
App3_EMPLOYEE emp = db.App3_EMPLOYEEs.Single(e => e.PKEY == employee.PKEY);
emp.FIRSTNAME = employee.FIRSTNAME;//copy property one by one
db.SubmitChanges();
return employee.PKEY;
}
2) отключить ObjectTrackingEnabled следующим образом
// but in this case lazy loading is not supported
public AppEmployeeDataContext() :
base(global::LinqLibrary.Properties.Settings.Default.AppConnect3DBConnectionString, mappingSource)
{
this.ObjectTrackingEnabled = false;
OnCreated();
}
3) Отсоедините все связанные объекты
partial class App3_EMPLOYEE
{
public void Detach()
{
this._APP3_EMPLOYEE_EXTs = default(EntityRef<APP3_EMPLOYEE_EXT>);
}
}
public int updateEmployee(App3_EMPLOYEE employee)
{
AppEmployeeDataContext db = new AppEmployeeDataContext();
employee.Detach();
db.App3_EMPLOYEEs.Attach(employee,true);
db.SubmitChanges();
return employee.PKEY;
}
4) используйте метку времени в столбце
http://www.west-wind.com/weblog/posts/135659.aspx
5) Создайте хранимую процедуру для обновления ваших данных и вызовите ее с помощью контекста db
Ответ 2
Вы не можете присоединить модифицированный объект к DataContext, если нет столбца RowVersion. Вместо этого вы можете хранить исходный объект в своем приложении, если поддерживается копия изменений данных. Затем, когда необходимо сохранить изменения, вы можете привязать исходный объект к DataContext, изменить его значения в соответствии с измененными значениями сущности и представить изменения.
Вот пример:
public int updateEmployee(App3_EMPLOYEE employee, App3_EMPLOYEE originalEmployee)
{
DBContextDataContext db = new DBContextDataContext();
db.App3_EMPLOYEEs.Attach(originalEmployee);
// TODO: Copy values from employee to original employee
db.SubmitChanges();
return employee.PKEY;
}
Update:
В базе данных есть таблица с идентификаторами столбцов, Name, Notes
// fetch an employee which will not be changed in the application
Employee original;
using(var db = new TestDbDataContext())
{
original = db.Employees.First(e => e.ID == 2);
}
// create an instance to work with
var modified = new Employee {ID = original.ID, Name = original.Name, Notes = original.Notes};
// change some info
modified.Notes = string.Format("new notes as of {0}", DateTime.Now.ToShortTimeString());
// update
using(var db = new TestDbDataContext())
{
db.Employees.Attach(original);
original.Notes = modified.Notes;
db.SubmitChanges();
}
Ответ 3
В этом разделе обсуждается здесь, в MSDN s
рекомендуется использовать поле IsVersion и метод Attach
Ответ 4
Вы можете прикрепить непривязанный измененный объект, используя эту перегрузку:
db.App3_EMPLOYEEs.Attach(employee, true);//Attach as modfieied
Обратите внимание, что для этого вам понадобится в вашей таблице столбец "Версия" типа "timestamp"
Ответ 5
Это функция в моем классе репозитория, которую я использую для обновления объектов
protected void Attach(TEntity entity)
{
try
{
_dataContext.GetTable<TEntity>().Attach(entity);
_dataContext.Refresh(RefreshMode.KeepCurrentValues, entity);
}
catch (DuplicateKeyException ex) //Data context knows about this entity so just update values
{
_dataContext.Refresh(RefreshMode.KeepCurrentValues, entity);
}
}
Где TEntity - ваш класс DB, и в зависимости от вас вы можете просто хотеть делать
_dataContext.Attach(entity);
Ответ 6
Используйте этот метод расширения для обновления всех свойств, которые являются атрибутами столбца:
public static void SaveToOriginal<T>(this T original, T actual)
{
foreach (var prop in typeof(T).GetProperties(BindingFlags.Public | BindingFlags.Instance)
.Where(info => info.GetCustomAttribute<System.Data.Linq.Mapping.ColumnAttribute>() != null))
{
prop.SetValue(original, prop.GetValue(actual));
}
}
Я имею в виду, что сначала вы восстанавливаете оригинал из базы данных, используйте метод для сопоставления всех атрибутов столбцов от нового элемента до оригинала и, наконец, для отправки.
Надеюсь, это поможет.