C#集合遍历时删除和增加元素的方法

2019-12-30 13:11:15丽君

本文实例讲述了C#集合遍历时删除和增加元素的方法。,具体如下:

大多数时候,遍历集合元素的时候并不需要对元素进行增加或者删除操作,但有些时候则需要,比如,如果集合中盛放的元素是社会上所有的人,那么有人死亡则元素删除,有人出生则是集合元素的增加。对于这种情况,遍历不能按照原来那种方式去做了,而且C#中的集合对于这类有增删动作的遍历,也不支持foreach循环。

有三种办法可以解决这一问题。

第一种方法:使用C#的LinkedList<>双链表。我原来设想,把原来链表需要删除的元素直接remove掉,那些新添加的元素,先装入到一个临时链表中,等循环结束,再用Add把临时链表的头结点添加到原来链表的尾部即可,这样算法的复杂度也较低,但是,出乎意料的是,C#的双链表,无法将属于另外一个链表的结点添加到本链表中,其Next属性也只读。无奈,只能一边循环,一边在原链表尾端添加结点,这样就需要标记处循环结束的位置,即需要在原来的未改动的链表的尾部结点处结束循环,而不是在改动后的链表的尾部结点处结束。这样就要求在循环开始之前,先获得尾部结点的引用。程序如下(链表中有0-29的整数值结点,遍历时遇到3的整数倍,就在链表尾端添加一个0值结点,遇到2的整数倍就删除结点)


using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace 集合遍历时删除或增加元素
{
  public partial class Form1 : Form
  {
    public Form1()
    {
      InitializeComponent();
    }
    private LinkedList<int> list = new LinkedList<int>();
    //初始化,添加0-29的整数进入链表
    private void button1_Click(object sender, EventArgs e)
    {
      for (int i = 0; i < 30; i++)
      {
        this.list.AddLast(i);
      }
    }
    //遍历链表,做如下操作:
    //遇到能被3整除的,就在该链表后增加一个0元素,遇到能被2整除的就删除该元素
    private void button2_Click(object sender, EventArgs e)
    {
      LinkedListNode<int> nodeNow = this.list.First;//链表第一个元素
      LinkedListNode<int> nodeLast = this.list.Last;//原链表的最后一个元素,循环结束的标记
      LinkedListNode<int> nodeTmp;//临时结点
      //循环结束的条件是,等当前结点是原链表的最后一个结点
      while (nodeNow != nodeLast)
      {
        //如果能被3整除时,则在链表后加一个0
        if (nodeNow.Value % 3 == 0)
        {
          this.list.AddLast(0);
        }
        //如果能被2整除,则删除该元素
        if (nodeNow.Value % 2 == 0)
        {
          //如果nodeNow被删除了,那么一定不能用Next获取下一个要判断的元素
          //因为已经自动向下一个移动了,这是就要在删除nodeNow之前,
          //获取它的Next,赋给nodeTmp,待nodeNow删除之后,再把nodeTmp的内存赋给nodeNow
          nodeTmp = nodeNow.Next;
          this.list.Remove(nodeNow);
          nodeNow = nodeTmp;
        }
        else
        {
          //如果不能被2整除,则在链表中保留该元素,并获得下一个并进行判断
          nodeNow = nodeNow.Next;
        }
      }
      //最后不要忘记对nodeLast(原链表最后一个元素)本身进行处理,上面的while循环没有包括这个nodeLast
      if (nodeNow.Value % 3 == 0)
      {
        this.list.AddLast(0);
      }
      if (nodeNow.Value % 2 == 0)
      {
        this.list.Remove(nodeNow);
      }
    }
    //测试结果
    private void button3_Click(object sender, EventArgs e)
    {
      foreach (int i in this.list)
      {
        Console.WriteLine(i);
      }
    }
  }
}