Here we are talking about filtering list items again :). I got two comment suggesting the use of yield return.
James Curran said:
Don’t build a new list if you don’t have to.
private static IEnumerable EvenSlicker_FilterPositiveItems(IEnumerable t)
{
foreach(int n in t)
if (n > 0)
yield return n;}
The advantage is if you try using the method this this:
foreach(int n in FilterPositiveItems(myList) {…}
Your way is O(2N) while mine is O(N)
and Configurator said:
Once again, I have to reiterate what I’ve stated here before.
I know you don’t like the ‘yield’ keyword, but it’s things like this that it’s meant for.
All of your examples create a copy of your list in memory, filtering only the needed results.
However, there are two methods that are a lot better unless you actually need a List:
1. Use the yield keyword directly. This function will return an IEnumreable object that when enumerated finds the results you want.
private static IEnumerable FilterPositives(IEnumerable numbers) {
foreach (int num in numbers)
if (num > 0)
yield return num;
}2. Use LINQ’s Where method. The following two functions will return the same result as my previous example.
private static List FilterPositives2(List numbers) {
return numbers.Where(intpredicate); // intpredicate is your method from earlier
}private static List FilterPositives3(List numbers) {
return numbers.Where(num => num > 0);
}
First of all thanks to Configurator and James for opening up a discussion. As configurator said "I am not fond of the Yield keyword" but I do recognize it must be used sometimes or we will never have had the opportunity to enjoy LINQ. However It must be said that I don’t like Yield for 2 reasons.
- Code Readability. I think that using yield makes debugging and understanding of the code mush harder.
- Its not Free. The compiler works pretty hard on creating many variables in the background, check it out below.
I took the code that was suggested by both of them and created a simple console application:
1: class Program
2: {
3: static void Main(string[] args)
4: {
5: List<int> t = new List<int> { 3, 4, -3, -4, 6, 1, -65, 2, -5, 2, -4 };
6: t = (List<int>)FilterPositives(t);
7: }
8:
9: private static IEnumerable FilterPositives(IEnumerable<int> numbers)
10: {
11: foreach (int num in numbers)
12: if (num > 0)
13: yield return num;
14: }
15: }
I then took the .EXE file and threw it into the reflector. Here is the code that was created by the compiler:
1: internal class Program
2: {
3: // Methods
4: private static IEnumerable FilterPositives(IEnumerable<int> numbers)
5: {
6: <FilterPositives>d__1 d__ = new <FilterPositives>d__1(-2);
7: d__.<>3__numbers = numbers;
8: return d__;
9: }
10:
11: private static void Main(string[] args)
12: {
13: List<int> <>g__initLocal0 = new List<int>();
14: <>g__initLocal0.Add(3);
15: <>g__initLocal0.Add(4);
16: <>g__initLocal0.Add(-3);
17: <>g__initLocal0.Add(-4);
18: <>g__initLocal0.Add(6);
19: <>g__initLocal0.Add(1);
20: <>g__initLocal0.Add(-65);
21: <>g__initLocal0.Add(2);
22: <>g__initLocal0.Add(-5);
23: <>g__initLocal0.Add(2);
24: <>g__initLocal0.Add(-4);
25: List<int> t = <>g__initLocal0;
26: t = (List<int>) FilterPositives(t);
27: }
28:
29: // Nested Types
30: [CompilerGenerated]
31: private sealed class <FilterPositives>d__1 : IEnumerable<object>
32: , IEnumerable
33: , IEnumerator<object>
34: , IEnumerator
35: , IDisposable
36: {
37: // Fields
38: private int <>1__state;
39: private object <>2__current;
40: public IEnumerable<int> <>3__numbers;
41: public IEnumerator<int> <>7__wrap3;
42: private int <>l__initialThreadId;
43: public int <num>5__2;
44: public IEnumerable<int> numbers;
45:
46: // Methods
47: [DebuggerHidden]
48: public <FilterPositives>d__1(int <>1__state)
49: {
50: this.<>1__state = <>1__state;
51: this.<>l__initialThreadId = Thread.CurrentThread.ManagedThreadId;
52: }
53:
54: private void <>m__Finally4()
55: {
56: this.<>1__state = -1;
57: if (this.<>7__wrap3 != null)
58: {
59: this.<>7__wrap3.Dispose();
60: }
61: }
62:
63: private bool MoveNext()
64: {
65: bool CS$1$0000;
66: try
67: {
68: switch (this.<>1__state)
69: {
70: case 0:
71: break;
72:
73: case 2:
74: goto Label_0081;
75:
76: default:
77: goto Label_009F;
78: }
79: this.<>1__state = -1;
80: this.<>7__wrap3 = this.numbers.GetEnumerator();
81: this.<>1__state = 1;
82: while (this.<>7__wrap3.MoveNext())
83: {
84: this.<num>5__2 = this.<>7__wrap3.Current;
85: if (this.<num>5__2 <= 0)
86: {
87: continue;
88: }
89: this.<>2__current = this.<num>5__2;
90: this.<>1__state = 2;
91: return true;
92: Label_0081:
93: this.<>1__state = 1;
94: }
95: this.<>m__Finally4();
96: Label_009F:
97: CS$1$0000 = false;
98: }
99: fault
100: {
101: this.System.IDisposable.Dispose();
102: }
103: return CS$1$0000;
104: }
105:
106: [DebuggerHidden]
107: IEnumerator<object> IEnumerable<object>.GetEnumerator()
108: {
109: Program.<FilterPositives>d__1 d__;
110: if ((Thread.CurrentThread.ManagedThreadId == this.<>l__initialThreadId)
111: && (this.<>1__state == -2))
112: {
113: this.<>1__state = 0;
114: d__ = this;
115: }
116: else
117: {
118: d__ = new Program.<FilterPositives>d__1(0);
119: }
120: d__.numbers = this.<>3__numbers;
121: return d__;
122: }
123:
124: [DebuggerHidden]
125: IEnumerator IEnumerable.GetEnumerator()
126: {
127: return this.System.Collections.Generic.IEnumerable<System.Object>.GetEnumerator();
128: }
129:
130: [DebuggerHidden]
131: void IEnumerator.Reset()
132: {
133: throw new NotSupportedException();
134: }
135:
136: void IDisposable.Dispose()
137: {
138: switch (this.<>1__state)
139: {
140: case 1:
141: case 2:
142: break;
143:
144: default:
145: break;
146: try
147: {
148: }
149: finally
150: {
151: this.<>m__Finally4();
152: }
153: break;
154: }
155: }
156:
157: // Properties
158: object IEnumerator<object>.Current
159: {
160: [DebuggerHidden]
161: get
162: {
163: return this.<>2__current;
164: }
165: }
166:
167: object IEnumerator.Current
168: {
169: [DebuggerHidden]
170: get
171: {
172: return this.<>2__current;
173: }
174: }
175: }
176: }
177:
Look at the amount of CXXp that was created. where as if you take the third method I proposed:
1: static void Main(string[] args)
2: {
3: List<int> t = new List<int> { 3, 4, -3, -4, 6, 1, -65, 2, -5, 2, -4 };
4: t = TheSlick_FilterPositiveItems(t);
5: }
6:
7: private static List<int> TheSlick_FilterPositiveItems(List<int> t)
8: {
9: return t.FindAll(n => n > 0);
10: }
This is what will be created by the compiler:
1: internal class Program
2: {
3: // Methods
4: private static void Main(string[] args)
5: {
6: List<int> <>g__initLocal0 = new List<int>();
7: <>g__initLocal0.Add(3);
8: <>g__initLocal0.Add(4);
9: <>g__initLocal0.Add(-3);
10: <>g__initLocal0.Add(-4);
11: <>g__initLocal0.Add(6);
12: <>g__initLocal0.Add(1);
13: <>g__initLocal0.Add(-65);
14: <>g__initLocal0.Add(2);
15: <>g__initLocal0.Add(-5);
16: <>g__initLocal0.Add(2);
17: <>g__initLocal0.Add(-4);
18: List<int> t = <>g__initLocal0;
19: t = TheSlick_FilterPositiveItems(t);
20: }
21:
22: private static List<int> TheSlick_FilterPositiveItems(List<int> t)
23: {
24: return t.FindAll(delegate (int n) {
25: return n > 0;
26: });
27: }
28: }
29:
No extra variables, classes, switch cases and other things you just don’t need. and this is only for such a simple program, imagine what we will get when things get complicated.
If you haven’t read the first article you can find it here.
Let me know what do you think.
Amit
Copyright © 2008
This feed is for personal, non-commercial use only.
The use of this feed on other websites breaches copyright. If this content is not in your news reader, it makes the page you are viewing an infringement of the copyright. (Digital Fingerprint:
)