Winform 多个窗口编辑同一条数据同步的实现
场景: 一个主窗口中,可以在列表(DataGridView)里选中一条记录编辑,打开一个编辑窗口(非模态窗口),编辑窗口保存后需要刷新父窗口,由于编辑窗口是非模态窗口,如果打开了多个窗口,并且都是编辑同一条数据,那么一个窗口保存(并关闭)后,需要通知其它正在打开的窗口“数据有更改,需要刷新”
首先,刷新父窗口,如果是打开编辑窗口是模态窗口,那么可以类似如下的实现(伪代码):
FormEdit frm = new FormEdit();
frm.EditId = 选中数据行对应的id;
if(frm.ShowDialog() == DialogResult.OK)
{
UpdateThisForm();
}
非模态窗口是Form.Show(); 由于该方法是void修饰,因此不能像上面那样去实现,此时可以在编辑窗口类中公开一个事件,当父窗口new这个编辑窗口后,可以注册这个事件,然后编辑窗口中如果保存了可以调用该事件方法达到通知的效果。
下面是例子,主窗口有一个DataGridView控件,数据绑定是Person的集合,Person实体类有Id,Name属性,选中某一行并点击编辑,可以打开编辑界面; 编辑界面有一个文本框显示编辑Person的Name,有一个保存按钮,点击保存之后将修改的Name更新到Person集合中(此处Person集合通过Xml序列化和反序列化实现保存于读取)
主窗口核心代码:
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
private void button1_Click(object sender, EventArgs e)
{
if (dataGridView1.SelectedRows.Count == 1)
{
int personId = (int)dataGridView1.SelectedRows[0].Cells["Id"].Value;
Form2 frm = new Form2();
frm.personId = personId;
frm.UpdateParentEvent += Frm_UpdateParentEvent;
frm.Show();
}
}
private void Frm_UpdateParentEvent()
{
LoadData();
}
private void Form1_Load(object sender, EventArgs e)
{
LoadData();
}
private void LoadData()
{
List<Person> personList = XmlSerializeHelper.DeserializeObject<List<Person>>("persons.xml");
dataGridView1.DataSource = personList;
}
编辑窗口核心代码:
public partial class Form2 : Form
{
public int personId;
/// <summary>
/// 刷新父窗口的事件
/// </summary>
public event Action UpdateParentEvent;
private Person p = null;
private List<Person> persons;
public Form2()
{
InitializeComponent();
}
private void Form2_Load(object sender, EventArgs e)
{
persons = XmlSerializeHelper.DeserializeObject<List<Person>>("persons.xml");
p = persons.Where(ps => ps.Id == personId).SingleOrDefault();
if (p != null)
{
txtName.Text = p.Name;
}
}
private void btnSave_Click(object sender, EventArgs e)
{
if (p != null)
{
p.Name = txtName.Text;
XmlSerializeHelper.SerializeObject(persons, "persons.xml");
UpdateParentEvent?.Invoke();
//获取所有打开的窗口
var openForms = Application.OpenForms;
Type thisType = this.GetType();
this.Close();
foreach (var item in openForms)
{
Type itemType = item.GetType();
//如果都是当前窗口的类的实例,但不是当前实例(证明打开了多个窗口)
if (itemType == thisType && !object.ReferenceEquals(item,this))
{
int itemPersonId = (int)itemType.GetField("personId").GetValue(item);
//证明编辑的是同一条记录,需要通知其它窗口刷新页面
if (itemPersonId == this.personId)
{
MethodInfo mInfo = itemType.GetMethod("ChangeHandle",BindingFlags.NonPublic | BindingFlags.Instance);
mInfo?.Invoke(item,null);
}
}
}
}
}
private void ChangeHandle()
{
if (MessageBox.Show("其它窗口修改了本条数据,需要重新加载","提示",MessageBoxButtons.OK,MessageBoxIcon.Information) == DialogResult.OK)
{
//重新加载数据
Form2_Load(this, null);
}
}
}
测试:
下面是打开了两个编辑窗口,并且都是编辑同一条数据,当编辑其中一个的Name,并保存后,另一个提示需要刷新
示例中使用了Application.OpenForms;得到当前所有打开的窗口,遍历并通过反射获取她们的“类型”(Type,下同),如果“类型”与当前窗口的“类型”相同,并且不是当前窗口,且又是编辑同一条数据时,反射获取方法并调用,以达到通知的效果。