코드를 상상해보십시오.
public class obj
{
// elided
}
public static Dictionary<string, obj> dict = new Dictionary<string, obj>();
방법 1
public static obj FromDict1(string name)
{
if (dict.ContainsKey(name))
{
return dict[name];
}
return null;
}
방법 2
public static obj FromDict2(string name)
{
try
{
return dict[name];
}
catch (KeyNotFoundException)
{
return null;
}
}
두 함수의 성능에 차이가 있는지 궁금합니다. 첫 번째 함수는 두 번째 함수보다 느려 야합니다. 한 번만 와우, 그것은 실제로 반대입니다.
1 만개의 값에 대한 루프 (기존의 10 만개와 존재하지 않는 9 만개의 경우) :
첫 번째 기능 : 306 밀리 초
두 번째 기능 : 20483 밀리 초
왜 그런 겁니까?
편집 :이 질문 아래 주석에서 알 수 있듯이 두 번째 기능의 성능은 실제로 기존 키가 0이 아닌 경우 첫 번째 기능보다 약간 우수합니다. 그러나 기존 키가 아닌 하나 이상의 키가 있으면 두 번째 키의 성능이 급격히 떨어집니다.
답변
한편, 예외를 던지는 것은 본질적으로 스택이 풀려야하기 때문에 본질적으로 비싸다 .
반면에, 키로 사전의 값에 액세스하는 것은 값이 빠르다. 왜냐하면 O (1) 연산이 빠르기 때문이다.
BTW :이를 수행하는 올바른 방법은 TryGetValue
obj item;
if(!dict.TryGetValue(name, out item))
return null;
return item;
이것은 두 번이 아닌 한 번만 사전에 액세스합니다. 키가 존재하지 않으면
실제로 반환 null
하려면 위 코드를 더 단순화 할 수 있습니다.
obj item;
dict.TryGetValue(name, out item);
return item;
이 때문에, 작동 TryGetValue
세트 item
에 null
아무 키는 경우 name
가 없습니다.
답변
사전은 특히 초고속 키 조회를 수행하도록 설계되었습니다. 그것들은 해시 테이블로 구현되며 항목이 많을수록 다른 방법에 비해 빠릅니다. 예외 엔진을 사용하는 것은 메소드가 설계 한 작업을 수행하지 못한 경우에만 수행되어야합니다. 이는 메소드가 오류 처리를위한 많은 기능을 제공하는 큰 객체 세트이기 때문입니다. try catch 블록으로 둘러싸인 모든 항목으로 한 번 전체 라이브러리 클래스를 작성했으며 600 개가 넘는 예외 중 하나에 대한 별도의 행이 포함 된 디버그 출력을 보도록 놀라게되었습니다!