미해결
[C#과 유니티로 만드는 MMORPG 게임 개발 시리즈] Part8: Entity Framework Core
EntityState.Unchanged로 처리 후 데이터베이스 조회시 다른 필드값이 null인 경우에 대한 질문
안녕하세요? Rookiss님.
몬스터가 드랍하는 아이템중에 stackable 아이템이 있어서 DB 조회없이 바로 갯수를 업데이트하고자 하여 아래와 같이 코드를 작성했습니다. 그 후에 다시 인벤토리 아이템을 클라에 통보하기 위하여 조회를 했으나 Stackable Item의 다른 Property가 모두 NULL이 되어 버리네요. 그래서 결국 아이템을 조회한 후에 갯수를 추가하니 정상적으로 처리가 되네요.
질문1 : 이런 경우 entityState를 사용하지 못하고 꼭 데이터베이스에 조회를 해야할 까요?
질문2 : stackable Item의 경우라면 어떤 로직으로 처리를 하는 게 바람직할 지 조언을 부탁드리고자 합니다.
------------------------------------------------------------------------------------
구글링을 통해 stackoverflow에서 찾아본 결과는 아래와 같습니다.
stackoverflow의 답변
https://stackoverflow.com/questions/28479933/after-setting-entitystate-unchanged-the-entity-properties-are-lost
=> 채택된 답변은 아이템의 갯수를 업데이트하기 전에 먼저 조회를 하라고 합니다.
------------------------------------------------------------------------------------
제가 작성중인 Dbtransaction 클래스의 메서드입니다.
------------------------------------------------------------------------------------
public static void RewardStackableItem(Player player, GameRoom room, int itemDbId, int totalCount)
{
if (player == null || room == null || itemDbId == 0)
return;
ItemDb oldItemDb = new ItemDb()
{
ItemDbId = itemDbId,
Count = totalCount
};
// YOU
Instance.Push(() =>
{
S_UpdateItem updateItemPacket = new S_UpdateItem();
using (AppDbContext db = new AppDbContext())
{
// 방법1. EntityState 로 처리
//db.Entry(oldItemDb).State = EntityState.Unchanged;
//db.Entry(oldItemDb).Property(nameof(ItemDb.Count)).IsModified = true;
// 방법2 : 데이터베이스에 조회를 한 후에 처리
ItemDb item = db.Items
.Single(i => i.OwnerDbId == player.PlayerDbId && i.ItemDbId == itemDbId);
item.Count = totalCount;
bool success = db.SaveChangesEx();
if (success)
{
//문제가 발생하는 부분으로 인벤토리의 해당 아이템을 수정하지 않고 그냥 조회 후 인벤토리를 clear한 후에 다시 인벤토리의 아이템을 만들어서 클라에 통보하고자 하였습니다.
List<ItemDb> items = db.Items
.Where(i => i.OwnerDbId == player.PlayerDbId).ToList();
player.Inven.Items.Clear();
// Me
room.Push(() =>
{
foreach (ItemDb itemDb in items)
{
Item item = Item.MakeItem(itemDb);
if (item != null)
{
player.Inven.Add(item);
ItemInfo info = new ItemInfo();
info.MergeFrom(item.Info);
updateItemPacket.Items.Add(info);
}
}
player.Session.Send(updateItemPacket);
});
}
}
});
}