public static T With<T, U>(this T obj, Expression<Func<T, U>> property, U value)
where T : ICloneable {
if (obj == null)
throw new ArgumentNullException("obj");
if (property == null)
throw new ArgumentNullException("property");
var memExpr = property.Body as MemberExpression;
if (memExpr == null || !(memExpr.Member is PropertyInfo))
throw new ArgumentException("Must refer to a property", "property");
var copy = (T)obj.Clone();
var propInfo = (PropertyInfo)memExpr.Member;
propInfo.SetValue(copy, value, null);
return copy;
}
public class Foo : ICloneable {
public int Id { get; set; }
public string Bar { get; set; }
object ICloneable.Clone() {
return new Foo { Id = this.Id, Bar = this.Bar };
}
}
public static void Test() {
var foo = new Foo { Id = 1, Bar = "blah" };
var newFoo = foo.With(x => x.Bar, "boo-ya");
Console.WriteLine(newFoo.Bar); //boo-ya
}
या, एक प्रति निर्माता का उपयोग कर:
public class Foo {
public Foo(Foo other) {
this.Id = other.Id;
this.Bar = other.Bar;
}
public Foo() { }
public int Id { get; set; }
public string Bar { get; set; }
}
public static void Test() {
var foo = new Foo { Id = 1, Bar = "blah" };
var newFoo = new Foo(foo) { Bar = "boo-ya" };
Console.WriteLine(newFoo.Bar);
}
और जॉर्ज उत्कृष्ट सुझाव पर एक मामूली बदलाव, जो एक से अधिक कार्य के लिए अनुमति देता है:
public static T With<T>(this T obj, params Action<T>[] assignments)
where T : ICloneable {
if (obj == null)
throw new ArgumentNullException("obj");
if (assignments == null)
throw new ArgumentNullException("assignments");
var copy = (T)obj.Clone();
foreach (var a in assignments) {
a(copy);
}
return copy;
}
public static void Test() {
var foo = new Foo { Id = 1, Bar = "blah" };
var newFoo = foo.With(x => x.Id = 2, x => x.Bar = "boo-ya");
Console.WriteLine(newFoo.Bar);
}
मैं शायद दूसरे के बाद से उपयोग करूंगा (1) किसी भी सामान्य प्रयोजन समाधान को अनावश्यक रूप से धीमा और गड़बड़ाना होगा; (2) यह आपके इच्छित चीज़ों के निकटतम वाक्यविन्यास है (और वाक्यविन्यास जो आप उम्मीद करते हैं); (3) एफ # कॉपी-एंड-अपडेट एक्सप्रेशन समान रूप से कार्यान्वित किए जाते हैं।
मैं इन सवालों के और अधिक देखने के रूप में कार्यात्मक भाषाओं मुख्यधारा बन उम्मीद है। एक सामान्य बचना "मैं [मेरी पसंदीदा भाषा] में [कुछ कार्यात्मक विशेषता] कैसे कर सकता हूं?" – Daniel