C#.net에서 제공하는 4가지 키워드는 개발에서 자주 사용되는 in, out, ref, paras 인데 어떻게 사용하면 될까요?
in은 대리자 및 인터페이스에서만 사용됩니다.
예:
//测试模型 class Model { public int a { get; set; } public Model(int a) { this.a = a; } }//创建3个实例List<Model> modelList= new List<Model>() { new Model(1), new Model(4), new Model(6) };//调用foreach接口,试着操作3个实例,赋值为nullmodelList.ForEach(e=>e=null); //查看结果://modelList的取值不变。
이유를 분석하면 ForEach의 매개변수는 다음과 같습니다. Delegate함수 :
//ForEach方法:public void ForEach(Action<T> action);//委托声明:public delegate void Action<in T>(T obj);
Delegate는 generic이며 T 유형 앞에 키워드 in이 추가됩니다. 키워드 in 때문에 T obj를 수정할 수 없습니다.
테스트해 보세요:
//修改元素e的属性amodelList.ForEach(e=>{e.a*=2;});
결과적으로 각 요소는 2를 곱하여 2,8,12가 됩니다. 객체의 속성이 수정될 수 있음을 알 수 있습니다.
out 키워드 사용법 참고 사항:
1) out이 없는 형식 매개변수의 경우 함수 정의 시 return 값 이전에 기능에 할당되어야 합니다.
2) 함수를 호출할 때, 없는 매개변수에는 초기값을 할당할 필요가 없습니다.
3) out 형식 매개변수 값 전달 은 참조에 의한 것입니다(by reference).
out 사용 시나리오:
함수가 여러 값을 반환할 때 일반적으로 out은 하나의 값을 반환하는 데 사용됩니다.
public bool Operation(out Model updateMod) { updateMode = new Model(5); try{ // my operation ... // return true; } catch{ //写入日志 return false; } } //使用 Model um; //未初始化 bool rtnMsg = Operation(out um); //如果初始化,传值通过reference //分析: //返回um,如果rntMsg为ture,则um按照预想逻辑被赋值, //如果rntMsg为false 则um未按照预想逻辑被赋值。
C#.net에는 out의 또 다른 중요한 응용 프로그램인 TryParse 함수 유형이 있습니다. 관심이 있으시면 Parse 및 TryParse를 통해: Try-Parse 및 Tester-Doer 모드
ref 키워드를 사용하여 매개변수 전달을 변경합니다. 값을 참조로 변경합니다. 원래 값은 참조로 전달됩니다. 효과는 참조가 있든 없든 동일합니다.
예:
public void reviseModel(int a) { a = 12; } Model model = new Model(10); //调用reviseModel reviseModel(model.a); //model.a仍然=10;by-value reviseMode(ref model.a); //编译不过,提示ref后的参数不归类与变量 int a; reviseMode(ref a); //如果不给变量a赋一个初始值, //编译器也是提示:调用前未被赋值的错误 //因此赋值 int a= model.a; //变量a初始值为10; reviseMode(ref a);//修改变量a=12;但是model.a的值仍然为10
객체 모델의 a 속성을 12로 변경하려면 어떻게 수정해야 하나요?
//直接将参数设为Model对象,则函数调用时,传值通过by referencepublic void reviseModel(Model md) { md.a = 12; } reviseModel(model );//传值通过by reference
따라서 ref 키워드 사용을 요약하면 다음과 같습니다.
ref는 기본 유형, 구조 등과 같은 값 변수를 처리하는 데 사용됩니다. 새 것일 필요는 없으며 값은 전송은 가치 복사를 기반으로 합니다.
1) ref 뒤의 변수가 값 유형인 경우 ref를 추가하면 참조로 전달된 값이 됩니다.
2) ref 뒤의 변수가 Reference인 경우 type(참조 유형)이면 ref를 추가하든 안하든 차이가 없습니다.
3) ref 뒤의 변수는 사용하기 전에 값을 할당해야 합니다.
4) ref 뒤의 변수는 사용할 수 없습니다. a reference Type 속성
위 내용은 기본적인 분석으로, 이 문제를 좀 더 자세히 분석하고 싶다면 계속 진행하시기 바랍니다.
4 out ref에 대한 심층 논의
out ref의 사용과 이를 사용하지 않을 경우 어떤 영향을 미칠지 주로 분석합니다.
1) C#에는 Int.TryParse와 같은 Try...라는 메서드 클래스가 있습니다. 이는 부울 값을 반환하고 문자열을 정수로 구문 분석하려고 시도합니다. true를 반환하고 결과 정수는 다음과 같습니다. int가 두 번째 출력으로 전달됩니다.
분석 기사 보기
예외 설계 지침
Parse 및 TryParse를 통한: Try-Parse 및 Tester-Doer 모드
기사에서 볼 수 있듯이 매개변수가 없는 하위 메서드 Parse와 비교 , 구문 분석된 경우 문자열이 실패하면 매개변수 오류 예외가 발생합니다.
Try... 메서드를 사용하여 작성된 코드는 try...catch 메서드를 사용하는 것보다 더 간결하므로 이는 out 매개 변수를 사용하는 일반적인 시나리오가 되었습니다.
2) Java와 C# 비교
// HashMap<K, V> map; // K key; V val = map.get(key);if (val != null) { // ...}
그러나 val == null, 둘 다 맵에 해당 키가 포함된 키-값 쌍이 없거나, 키-값 쌍이 이미 존재하지만 해당 값이 null일 수 있습니다.
이 둘을 구별하기 위해 HashMap은 containKey() 메서드를 제공합니다. 따라서 이를 작성하는 올바른 방법은 다음과 같습니다.
// HashMap<K, V> map; // K key;if (map.containsKey(key)) { V val = map.get(key); // ...}
containKey()와 get()의 내부 작업은 둘 다 해시 검색을 수행해야 하지만 서로 다른 부분을 반환할 뿐입니다. 검색결과 중 즉, 이 "올바른 작성 방법"에 따라 작성하면 HashMap에 한 번 액세스하는 데 비용이 두 배로 늘어납니다. 컵!
C#에는 Java보다 세심한 디자인이 많이 있습니다. C#에서 out 키워드를 사용하여 이 문제를 개선하는 방법을 알아보세요.
System.Collections.Generic.Dictionary
TryGetValue: Dictionary(TKey, TValue).TryGetValue Method (TKey, TValue) (System.Collections.Generic)public bool TryGetValue( TKey key, out TValue value )ParameterskeyType: TKey The key of the value to get. valueType: TValue
이 방법을 사용하면 위 Java 코드에 해당하는 C# 버전을 다음과 같이 작성할 수 있습니다.
// Dictionary<TKey, TValue> dict; // TKey key; TValue val;if (dict.TryGetValue(key, out val)) { // ...}
ContainsKey와 Item을 비교합니다. [ 키 ] 의미론을 결합하여 해시 검색에서 찾을 수 있는 모든 정보를 한 번에 반환함으로써 소스에서 "2회 검색"의 중복 작업을 피하고 이는 프로그램 성능에 유리합니다.
C#.net에서는 매개변수라는 키워드를 제공합니다. 이전에는 이 키워드가 존재하는지 몰랐습니다. 한 번은 동료가 내 오버로드된 함수의 여러 버전을 보고 침착하게 나에게 '그래, 매개변수를 사용해도 돼'라고 말했습니다. 나중에 확인해보니 이제 익숙해지네요. 그냥 이전 버전을 모두 제거하고 매개변수를 사용하여 리팩토링했습니다.
그럼 params의 사용법과 제가 겪었던 과정에 대해 말씀드리겠습니다. <… 다른 쿼리 필드를 추가합니다.
4개 핵심 필드 기반 쿼리 방법:
public void GetPlansByInputControl(string planState, string contactno,DatePair dp) { string planStat = ""; switch (planState) { case "...": planStat = "..."; break; case "...": planStat = "..."; break; } plans = getPlansWithCondition(Convert.ToDateTime(dp.startValue), Convert.ToDateTime(dp.endValue), planStat, contactno); }
입니다.
private List<MPartPlan> getMPartPlansWithCondition(DateTime dateTime, DateTime dateEndTime, string planStat, string contactNo) { var conditions = new CslSqlBaseSingleTable(); conditions.AddCondition("RequireStartDate", dateTime, DataCompareType.GreaterOrEqual); conditions.AddCondition("RequireStartDate", dateEndTime, DataCompareType.LessOrEqual); conditions.AddCondition("OrderCode", contactNo, DataCompareType.Equal); if (!string.IsNullOrEmpty(planStat)) { conditions.AddCondition("PlanState", planStat, DataCompareType.Equal); } return _cslMPartPlan.QueryListInSingleTable(typeof(MPartPlan), conditions); } }
问题来了,当查询再新加1个字段时,你难道还再重载一个版本吗?
5.2 应用params
private List<MPartPlan> getMPartPlansWithCondition(DateTime dateTime, DateTime dateEndTime, string planStat, string contactNo,string newField);
当C#提供了params后,当然不用,直接将getMPartPlansWithCondition改写为如下
private List<MPartPlan> getMPartPlansWithCondition(params object[] queryConditions); { queryConditions[0] queryConditions[1] queryConditions[2] queryConditions[3] queryConditions[4] //放到字典中dict sqlQuery(dict); }
以后随意添加查询字段,只要修改下这个函数就行了,不用增删重载版本!!!
客户端调用,直接加一个字段就行
_bsl.GetPlansByInputControl(field1, field2,field3,field4,field5);
5.3 总结
queryFun(params object[] objs),带有这个参数的函数,只需要一个版本,这样解决了因为个数不一致而导致的多个重载版本,
在客户端调用时,将属性参数一一列数即可。
위 내용은 .NET Framework - in, out, ref, paras에 사용되는 코드 요약의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!