简体   繁体   English

MVC 5实体框架6 - 删除嵌套/分层数据

[英]MVC 5 Entity Framework 6 - deleting nested / hierarchical data

i've run into some trouble, when playing around with hierarchical data and trying to delete a node that has children. 当玩弄分层数据并尝试删除有孩子的节点时,我遇到了一些麻烦。

I have set up a sample tree similar to this article and decorated the ul-li items with a little bit of bootstrap (add, delete and edit icons). 我已经设置了一个类似于本文的示例树,并用一些引导程序(添加,删除和编辑图标)修饰了ul-li项目。

I'm struggeling now at the recursive deletion, as i am using an IEnumerable (Children), which is the inverse result of parentNodeID. 我现在正在进行递归删除,因为我正在使用IEnumerable(Children),这是parentNodeID的反向结果。 So when deleting a node, the children were updated and the system throws an enumeration error (collection has changed). 因此,在删除节点时,子节点已更新,系统抛出枚举错误(集合已更改)。 I had such an error a few years ago, but i don't remember, how i fixed it. 几年前我有这样的错误,但我不记得了,我是如何修理它的。 Any advice? 有什么建议?

So here is what i got: 所以这就是我得到的:

~/Models/Tree.cs 〜/型号/ Tree.cs

public class Tree
{
    [Key]
    public int NodeID { get; set; }

    [Display(Name ="Element name")]
    public string NodeName { get; set; }

    [Display(Name ="Element identifier")]
    public string NodeIdentifier { get; set; }

    public int? ParentNodeID { get; set; }
    public virtual Tree Parent { get; set; }
    public virtual ICollection<Tree> Children { get; set; }
}

~/Controllers/TreesController.cs 〜/控制器/ TreesController.cs

// Partial - only Delete Actions

// GET: Trees/Delete/5
public async Task<ActionResult> Delete(int? id)
{
    if (id == null)
    {
        return new HttpStatusCodeResult(HttpStatusCode.BadRequest);
    }
    Tree tree = await db.Trees.FindAsync(id);
    if (tree == null)
    {
        return HttpNotFound();
    }

    if (tree.Children.Count > 0)
    {
        ViewBag.HasChildren = true;
    }
    else
    {
        ViewBag.HasChildren = false;
    }

    return View(tree);
}

// POST: Trees/Delete/5
[HttpPost, ActionName("Delete")]
[ValidateAntiForgeryToken]
public async Task<ActionResult> DeleteConfirmed(int id)
{
    Tree tree = await db.Trees.FindAsync(id);
    DeleteRecursive(tree);
    db.Trees.Remove(tree);
    await db.SaveChangesAsync();
    return RedirectToAction("Index");
}

public void DeleteRecursive(Tree tree)
{
    foreach(var child in tree.Children)
    {
        DeleteRecursive(child);
    }
    db.Trees.Remove(tree);
}

~/Views/Trees/Index.cshtml 〜/查看/树/ Index.cshtml

<div class="container">
@helper BuildTree(IEnumerable<MVCMusicStore.Models.Tree> tree, int? parentID = null)
{
    var nodes = tree.Where(t => t.ParentNodeID == parentID).OrderBy(n => n.NodeIndexOrder);
    if (nodes.Any())
    {
        if (nodes.First().ParentNodeID == null)
        {
                <ul id="tree">
                    @foreach (var node in nodes)
                    {
                        <li>
                            @node.NodeIndexOrder - @node.NodeName &nbsp; <a href="@Url.Action("Edit","Trees",new { id = node.NodeID })" data-toggle="tooltip" data-placement="right" title="Bearbeiten"><i class="glyphicon glyphicon-edit"></i></a>&nbsp; <a href="@Url.Action("Delete","Trees",new { id = node.NodeID })" data-toggle="tooltip" data-placement="right" title="Löschen"><i class="glyphicon glyphicon-trash"></i></a>
                            @BuildTree(tree, node.NodeID)
                        </li>
                    }
                </ul>
        }
        else
        {
                <ul>
                    @foreach (var node in nodes)
                    {
                        <li>
                            @node.NodeIndexOrder - @node.NodeName &nbsp; <a href="@Url.Action("Edit","Trees",new { id = node.NodeID })" data-toggle="tooltip" data-placement="right" title="Bearbeiten"><i class="glyphicon glyphicon-edit"></i></a>&nbsp; <a href="@Url.Action("Delete","Trees",new { id = node.NodeID })" data-toggle="tooltip" data-placement="right" title="Löschen"><i class="glyphicon glyphicon-trash"></i></a>
                            @BuildTree(tree, node.NodeID)
                        </li>
                    }
                </ul>
        }         
    }
    else
    {

        <p>
            <a href="@Url.Action("Create","Trees", new { parentNodeID = parentID })" data-toggle="tooltip" data-placement="right" title="Neu"><i class="glyphicon glyphicon-plus"></i></a>
        </p>
    }
 }

@BuildTree(Model,null)
</div>

Edit - Solution: 编辑 - 解决方案:

public void DeleteRecursive(Tree tree)
{
    foreach(var child in tree.Children.ToArray<Tree>())
    {
        DeleteRecursive(child);
    }
    db.Trees.Remove(tree);
}

Using an array to delete from does not alter the original collection. 使用要删除的数组不会更改原始集合。

Credits to Johnathon Sullinger: Delete item in nested collections of Nth level 致Johnathon Sullinger的信用:删除第N级嵌套集合中的项目

public void DeleteRecursive(Tree tree)
{
    foreach(var child in tree.Children.ToArray<Tree>())
    {
        DeleteRecursive(child);
    }
    db.Trees.Remove(tree);
}

Using an array to delete from does not alter the original collection. 使用要删除的数组不会更改原始集合。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM