我相信不少人对这个再熟悉不过了。对已经修改的集合进行操做就会出现这个错。spa
好比有下面的一段代码,咱们建立一个集合,并向集合中添加10个数,而后,咱们循环再将这些数移除了。code
static void Main(string[] args) { List<int> lst = new List<int>(); for (int i = 0; i < 10; i++) { lst.Add(i); } foreach (var item in lst) { lst.Remove(item); } Console.Read(); }
出现了.....blog
是否是被泛型集合提供的方法坑了?我记得好久以前我也被坑过。很疑惑吧,其实也很简单,由于你若是移除了一项,集合的元素个数是变化的。这个时候元素会重排,第二个元素的索引由1变为0,后面的依次往前移动。索引
static void Main(string[] args) { List<int> lst = new List<int>(); for (int i = 0; i < 10; i++) { lst.Add(i); } var result = lst; Console.WriteLine("lst的count" + lst.Count); lst.Remove(0); Console.WriteLine("lst的count" + lst.Count); for (int i = 0; i < lst.Count; i++) { Console.WriteLine("索引:{0},值:{1}", i, lst[i]); } Console.Read(); }
上面的代码为集合添加10个元素。而后输出当前集合的count,接着将索引为0的元素移除。这个时候集合中应该没有元素0了。而后输出集合的元素个数。输出此时的集合中索引和对应的值。如图所示string
能够看到,原本索引为1的1,往前移动了,此时他的索引变为了0.因此在使用foreach移除的时候,集合是变化的,是不容许的。难道就没办法操做了吗?固然有,它不是移除一个集合就少一个吗》it
此时,咱们能够经过for循环,从集合的队尾移除,这个时候移除队尾的元素,虽然集合的count变了,但他们的索引没有变化。for循环
static void Main(string[] args) { List<int> lst = new List<int>(); for (int i = 0; i < 10; i++) { lst.Add(i); } var result = lst; Console.WriteLine("lst的count" + lst.Count); lst.Remove(0); Console.WriteLine("lst的count" + lst.Count); for (int i = lst.Count - 1; i >= 0; i--) { Console.WriteLine("索引:{0},值:{1}", i, lst[i]); Console.WriteLine("移除了元素:{0}", lst[i]); lst.RemoveAt(i); } Console.Read(); }
那么咱们只移除知足条件的是否也能够经过for循环呢?固然能够,foreach你不是不让吗?又不是只有你一个能够循环。class
static void Main(string[] args) { List<int> lst = new List<int>(); for (int i = 0; i < 10; i++) { lst.Add(i); } var result = lst; Console.WriteLine("lst的count" + lst.Count); lst.Remove(0); Console.WriteLine("lst的count" + lst.Count); for (int i = 0; i < lst.Count; i++) { Console.WriteLine("索引:{0},值:{1}", i, lst[i]); if (lst[i] % 2 == 0) { Console.WriteLine("移除了元素:{0}", lst[i]); lst.RemoveAt(i); } } Console.Read(); }
前不久刚有人遇到,这里仍是记录一下吧。泛型