12 C# 7.0 C# 6.0의경우컴파일러를완전히새롭게만드는작업으로인해이전버전대비중대한문법향상이없었던반면 C# 7.0부터는다시주요한변화를이끌어내기시작했는데바로함수형언어에서나제공하던패턴매칭구문을가능하게했다는점이다. 물론 C# 7.0에서는그밖의소소한간편표기구문도제공한다.

Similar documents
PowerPoint Presentation

PowerPoint Presentation

gnu-lee-oop-kor-lec06-3-chap7

Microsoft PowerPoint - CSharp-10-예외처리

슬라이드 1

Microsoft PowerPoint 자바-기본문법(Ch2).pptx

JAVA PROGRAMMING 실습 08.다형성

<4D F736F F F696E74202D20C1A63038C0E520C5ACB7A1BDBABFCD20B0B4C3BC4928B0ADC0C729205BC8A3C8AF20B8F0B5E55D>

PowerPoint Presentation

Microsoft PowerPoint - chap06-2pointer.ppt

Modern Javascript

JAVA 프로그래밍실습 실습 1) 실습목표 - 메소드개념이해하기 - 매개변수이해하기 - 새메소드만들기 - Math 클래스의기존메소드이용하기 ( ) 문제 - 직사각형모양의땅이있다. 이땅의둘레, 면적과대각

PowerPoint 프레젠테이션

Microsoft PowerPoint - 3ÀÏ°_º¯¼ö¿Í »ó¼ö.ppt

쉽게 풀어쓴 C 프로그래밍

11장 포인터

PowerPoint Presentation

OCW_C언어 기초

쉽게

금오공대 컴퓨터공학전공 강의자료

InsertColumnNonNullableError(#colName) 에해당하는메시지출력 존재하지않는컬럼에값을삽입하려고할경우, InsertColumnExistenceError(#colName) 에해당하는메시지출력 실행결과가 primary key 제약에위배된다면, Ins

C# Programming Guide - Types

JAVA PROGRAMMING 실습 05. 객체의 활용

1. auto_ptr 다음프로그램의문제점은무엇인가? void func(void) int *p = new int; cout << " 양수입력 : "; cin >> *p; if (*p <= 0) cout << " 양수를입력해야합니다 " << endl; return; 동적할

PowerPoint 프레젠테이션

[ 마이크로프로세서 1] 2 주차 3 차시. 포인터와구조체 2 주차 3 차시포인터와구조체 학습목표 1. C 언어에서가장어려운포인터와구조체를설명할수있다. 2. Call By Value 와 Call By Reference 를구분할수있다. 학습내용 1 : 함수 (Functi

슬라이드 1

Microsoft PowerPoint - ch07 - 포인터 pm0415

쉽게 풀어쓴 C 프로그래밍

PowerPoint 프레젠테이션

ThisJava ..

Contents. 1. PMD ㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍ 2. Metrics ㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍ 3. FindBugs ㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍ 4. ㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍㆍ

<322EBCF8C8AF28BFACBDC0B9AEC1A6292E687770>

JAVA PROGRAMMING 실습 09. 예외처리

JVM 메모리구조

Microsoft PowerPoint - chap02-C프로그램시작하기.pptx

Microsoft PowerPoint - chap03-변수와데이터형.pptx

다른 JSP 페이지호출 forward() 메서드 - 하나의 JSP 페이지실행이끝나고다른 JSP 페이지를호출할때사용한다. 예 ) <% RequestDispatcher dispatcher = request.getrequestdispatcher(" 실행할페이지.jsp");

Microsoft PowerPoint - CSharp-2-기초문법

PowerPoint Presentation

슬라이드 1

A Hierarchical Approach to Interactive Motion Editing for Human-like Figures

쉽게 풀어쓴 C 프로그래밍

학습목차 2.1 다차원배열이란 차원배열의주소와값의참조

쉽게 풀어쓴 C 프로그래밍

예제 2) Test.java class A intvar= 10; void method() class B extends A intvar= 20; 1"); void method() 2"); void method1() public class Test 3"); args) A

PowerPoint 프레젠테이션

11장 포인터

JAVA PROGRAMMING 실습 02. 표준 입출력

Microsoft PowerPoint - [2009] 02.pptx

학습목표 함수프로시저, 서브프로시저의의미를안다. 매개변수전달방식을학습한다. 함수를이용한프로그래밍한다. 2

C++ Programming

JAVA PROGRAMMING 실습 02. 표준 입출력

Microsoft PowerPoint - lec2.ppt

A Tour of Java V

Cluster management software

목차 BUG DEQUEUE 의 WAIT TIME 이 1 초미만인경우, 설정한시간만큼대기하지않는문제가있습니다... 3 BUG [qp-select-pvo] group by 표현식에있는컬럼을참조하는집합연산이존재하지않으면결괏값오류가발생할수있습니다... 4

PowerPoint Presentation

chap 5: Trees

Microsoft PowerPoint - chap10-함수의활용.pptx

제4장 기본 의미구조 (Basic Semantics)

Microsoft PowerPoint - 제11장 포인터(강의)

Tcl의 문법

Microsoft PowerPoint - C++ 5 .pptx

슬라이드 1

초보자를 위한 C# 21일 완성

<4D F736F F F696E74202D20B8AEB4AABDBA20BFC0B7F920C3B3B8AEC7CFB1E22E BC8A3C8AF20B8F0B5E55D>

Microsoft PowerPoint - ch09 - 연결형리스트, Stack, Queue와 응용 pm0100

PowerPoint Presentation

Chapter 4. LISTS

PowerPoint Template

제 14 장포인터활용 유준범 (JUNBEOM YOO) Ver 본강의자료는생능출판사의 PPT 강의자료 를기반으로제작되었습니다.

PowerPoint 프레젠테이션

PowerPoint Presentation

JUNIT 실습및발표

Microsoft PowerPoint - chap12-고급기능.pptx

Microsoft PowerPoint - Java7.pptx

1. 객체의생성과대입 int 형변수 : 선언과동시에초기화하는방법 (C++) int a = 3; int a(3); // 기본타입역시클래스와같이처리가능 객체의생성 ( 복습 ) class CPoint private : int x, y; public : CPoint(int a

금오공대 컴퓨터공학전공 강의자료

5장.key

02 C h a p t e r Java

Frama-C/JESSIS 사용법 소개

(8) getpi() 함수는정적함수이므로 main() 에서호출할수있다. (9) class Circle private double radius; static final double PI= ; // PI 이름으로 로초기화된정적상수 public

UI TASK & KEY EVENT

Microsoft PowerPoint - 제11장 포인터

선형대수학 Linear Algebra

Microsoft PowerPoint - 04-UDP Programming.ppt

Microsoft Word - FunctionCall

PowerPoint Presentation

<4D F736F F F696E74202D2036C0CFC2B05FB0B4C3BCC1F6C7E2C7C1B7CEB1D7B7A1B9D62E707074>

Lab 3. 실습문제 (Single linked list)_해답.hwp

설계란 무엇인가?

쉽게 풀어쓴 C 프로그래밍

Microsoft PowerPoint - additional01.ppt [호환 모드]

PowerPoint Presentation

q 이장에서다룰내용 1 객체지향프로그래밍의이해 2 객체지향언어 : 자바 2

<4D F736F F D20BEBEBCA520C4DAB5F920BFACBDC0202D20B8D6C6BC20BEB2B7B9B5E5BFCD20C0CCBAA5C6AE2E646F6378>

02장.배열과 클래스

4 장클래스와객체 클래스와객체 public과 private 구조체와클래스객체의생성과생성자객체의소멸과소멸자생성자와소멸자의호출순서디폴트생성자와디폴트소멸자멤버초기화멤버함수의외부정의멤버함수의인라인함수선언 C++ 프로그래밍입문

Transcription:

12 C# 7.0 C# 6.0의경우컴파일러를완전히새롭게만드는작업으로인해이전버전대비중대한문법향상이없었던반면 C# 7.0부터는다시주요한변화를이끌어내기시작했는데바로함수형언어에서나제공하던패턴매칭구문을가능하게했다는점이다. 물론 C# 7.0에서는그밖의소소한간편표기구문도제공한다. C# 7.0에대응하는닷넷프레임워크의버전은 4.7이고, 주요개발환경은비주얼스튜디오 2017이다. 하지만 C# 7.0 컴파일러가출시되는시점의닷넷프레임워크버전은 4.6.2였고이로인해 C# 7.0의새로운두가지기능을사용하는데문제가발생한다. 먼저새로운문법인튜플은 System.ValueTuple 타입을요구하지만닷넷 4.6.2의 BCL에서는제공하지않기때문에 NuGet 패키지관리자로부터 System.ValueTuple을직접설치해야만했다. System.ValueTuple 타입 https://www.nuget.org/packages/system.valuetuple/ 다행히이후에출시된닷넷 4.7의 BCL은 System.ValueTuple을기본적으로포함하고있다. 문제가되는다른하나는 async 메서드의성능향상을위해추가된 System.Threading.Tasks. ValueTask 타입인데, 이는닷넷코어버전의초기 1.0 버전부터이미 System.Threading.Tasks. Extensions.dll을통해제공하고있었는데도불구하고닷넷프레임워크 4.7의 BCL에서는여전히포 728 2 부 _ C# 고급문법

함하고있지않는타입이다. 따라서이타입이들어간 async 구문을테스트하려면반드시 NuGet 패 키지관리자로부터 ValueTask 를구현한 System.Threading.Tasks.Extensions.dll 을다운로드해 참조추가해야한다. ValueTask 타입을포함하는 System.Threading.Tasks.Extensions 패키지 https://www.nuget.org/packages/system.threading.tasks.extensions/ 12.1 더욱편리해진 out 매개변수사용 out 매개변수가정의된대표적인메서드가바로 TryParse 다. C# 6.0 이전에는 out 매개변수가정의 된메서드를사용하려는경우반드시인자로전달될인스턴스를미리선언해야했다. int result; // 변수를미리선언 int.tryparse("5", out result); C# 7 부터는변수선언없이변수의타입과함께 out 예약어를쓸수있다. int.tryparse("5", out int result); 물론 C# 7 컴파일러는위의구문을컴파일하는경우개발자대신 C# 6 이전의코드로변환해소스코 드를컴파일한다. 따라서당연히다음과같은식으로코드를작성하면컴파일오류가발생한다. int.tryparse("5", out int result); int.tryparse("5", out int result); // 컴파일오류! 왜냐하면 C# 7 컴파일러가바꾼구문은다음과같기때문이다. 12 장. C# 7.0 729

int result; int.tryparse("5", out result); int result; // 중복변수선언 int.tryparse("5", out result) 이뿐만아니라타입추론을컴파일러에게맡기는 var 예약어도사용할수있다. int.tryparse("5", out var result); // out int result 로컴파일러가대신처리 심지어값을무시하는구문까지추가되어값이필요하지않은상황에대해서는변수명까지생략할수 있게됐다. int.tryparse("5", out int _ ); // 변수명대신밑줄 (underline: _) 로생략 int.tryparse("5", out var _ ); 12.2 반환값및로컬변수에 ref 기능추가 (ref returns and locals) out 예약어의사용법이 C# 7.0에서는쉽게개선된경우라면 ref 예약어는사용법을확장시킨경우에해당한다. C# 6.0 이전까지는 ref 예약어를오직메서드의매개변수로만사용할수있었지만이제는로컬변수와반환값에대해서도참조관계를맺을수있게개선됐다. 우선로컬변수에대한 ref 예약어의사용여부에따른차이점을알아보자. ref가지정되지않은경우다음코드는 int a = 5; int b = a; 730 2 부 _ C# 고급문법

변수 a 와 b 모두값형식이므로메모리에는그림 12.1 과같은식으로할당된다. 그림 12.1 스택영역에별도의공간을점유한 a, b 변수 int a; 0x2000 int b; 5 0x1600 스택영역 5 0x1200 메모리 0x0000 이상태에서다음과같이 a 변수에새로운값을대입하면 int a = 5; int b = a; a = 6; Console.WriteLine(a); // 6 Console.WriteLine(b); // 5 예상할수있듯이 a 와 b 의값은그림 12.2 와같이독립적으로유지된다. 그림 12.2 a 의값과독립적인변수 b int a; 0x2000 int b; 6 0x1600 스택영역 5 0x1200 0x0000 메모리 12 장. C# 7.0 731

그럼이번에는 ref 예약어를사용해 a 변수와 b 변수가참조관계를맺도록수정해보자. int a = 5; ref int b = ref a; 이제 a 변수와 b 변수는그림 12.3 과같이동일한메모리를공유하게된다. 그림 12.3 변수 a 와 b 가같은주소를공유 int a; 0x2000 int b; 5 0x1600 스택영역 0x1200 메모리 0x0000 즉, 4.5.1.5절 구조체 에서메서드의매개변수가 ref 사용여부에따라다르게동작했던것과같은결과를보이는것이다. 반환값에 ref를사용할수있는경우에는더욱극적인표현이가능하다. 예를들어배열의특정요소의값을바꾸려면배열인스턴스자체를넘겨주거나원하는배열요소만을바꿀수있는메서드를일부러정의해야했다 ( 정보은닉입장에서는후자의방법이권장되지만아래예제에서는일부러전자의방법을사용했다 ). IntList intlist = new IntList(); int[] list = intlist.getlist(); // IntList 에정의된 list 요소를바꾸기위해목록반환 list[0] = 5; // list[0] 의요소를변경 intlist.print(); // 출력결과 : 5,2, class IntList 732 2 부 _ C# 고급문법

int[] list = new int[2] 1, 2 ; public int [] GetList() return list; internal void Print() Array.ForEach(list, (e) => Console.Write(e + ",")); Console.WriteLine(); 하지만참조 return 을사용하면원하는요소의참조만반환하는것이가능하다 ( 이때 ref 예약어를네 곳에서명시해야한다 ). IntList intlist = new IntList(); ref int item = ref intlist.getfirstitem(); item = 5; // 참조값이므로값을변경하면원래의 int [] 배열에도반영 intlist.print(); // 출력결과 : 5,2, class IntList int[] list = new int[2] 1, 2 ; public ref int GetFirstItem() return ref list[0]; internal void Print() [ 생략 ] 12 장. C# 7.0 733

이밖에도참조 return 덕분에가능한구문이바로메서드에값을대입하는구문이다. 가령다음과같 은구문으로값을설정할수있는데, C# 6.0 이전에는불가능한문법이다. MyMatrix matrix = new MyMatrix(); matrix.put(0, 0) = 1; class MyMatrix int[,] _matrix = new int[100, 100]; public ref int Put(int column, int row) return ref _matrix[column, row]; 다소가독성이떨어지는억지스러운구문이지만메서드에값을설정하는구문과동시에값을받는것 도가능하다. MyMatrix matrix = new MyMatrix(); matrix.put(0, 0) = 1; int result = matrix.put(1, 1) = 1; Console.WriteLine(result); // 출력결과 : 1 반환및로컬변수에사용할수있는 ref 예약어는두가지제약이있는데모두컴파일오류가발생하 므로실수할여지는없다. 지역변수를 return ref로반환해서는안된다. 지역변수의유효범위가스택상에있을때로한정되기때문에메서드 의실행이끝나호출측으로넘어가는시점에스택이해제되어 return ref 로반환받은인스턴스가남아있을거라는보 장을할수없기때문이다. ref 예약어를지정한지역변수는다시다른변수를가리키도록변경할수없다. 734 2 부 _ C# 고급문법

두가지모두다음과같은상황으로테스트할수있다. public ref int RefReturnOfLocalValue() int x = 5; return ref x; // 컴파일에러 CS8168: 지역변수를 return ref로반환할수없음 public void ChangeRefLocalVar() int a = 5; ref int b = ref a; int c = 10; b = ref c; // 컴파일에러 : 이미 a 를가리키는 b 참조변수를다른참조로바꿀수없음 ref b = ref c; // 컴파일에러 : 이런식의사용법도허용되지않음 12.3 튜플 튜플 (Tuple) 이란유한개의원소로이뤄진졍렬된리스트를의미하는자료구조로서일반적으로 n-tuple이라고일컬을때의 n은요소의수를의미하며가령 3개의요소를갖는경우 3-tuple이라고한다. 어려운수학적정의와는달리 C# 에도입된튜플의경우메서드의인자또는반환에대해다중값을한번에전달할수있는약식구문이라고여기면된다. 이절의내용을실습하려면비주얼스튜디오에서프로젝트의대상프레임워크값을반드시 4.7로설정하거나 NuGet 패키지관리자로부터 System.ValueTuple을참조추가해야한다. 그렇지않으면컴파일할때다음과같은예외메시지가발생한다. error CS8179: Predefined type System.ValueTuple`2 is not defined or imported CS8179 미리정의된형식 System.ValueTuple`2 을 ( 를 ) 정의하지않았거나가져오지않았습니다. 튜플의필요성을이해하기위해우선튜플이없었던시절부터 2개이상의반환값을돌려줘야했던 int.tryparse의예를살펴보자. int.tryparse는입력문자열을숫자로변환한값을 out 매개변수로 12 장. C# 7.0 735

넘기는것과함께성공여부를 return 으로반환한다. 만약 out 으로처리하지않았다면별도의클래스 를만들어반환값을표현해야하는전형적인사례다. public class IntResult public bool Parsed get; set; public int Number get; set; class Program static void Main(string[] args) Program pg = new Program(); IntResult result = pg.parseinteger("15"); Console.WriteLine(result.Parsed); // True Console.WriteLine(result.Number); // 15 IntResult ParseInteger(string text) IntResult result = new IntResult(); try result.number = Int32.Parse(text); result.parsed = true; catch result.parsed = false; return result; 736 2 부 _ C# 고급문법

보다시피 2개이상의반환값이필요할때마다저런식으로별도의클래스를만들어야하는것은여간귀찮은작업이아닐수없다. 물론위의코드정도는 out 매개변수를사용하는것으로바꿀수있지만 3개, 4개의매개변수를이같은식으로처리해야한다고가정하면 out 매개변수로처리하는방식이직관적인메서드구현의관점에서보면바람직하지않을수있다. 매번정의되는클래스를없애기위해 C# 3.0의익명타입과 C# 4.0의 dynamic 예약어를이용할수도있다. dynamic result = ParseInteger("20"); Console.WriteLine(result.Parsed); Console.WriteLine(result.Number); dynamic ParseInteger(string text) int number = 0; try number = Int32.Parse(text); return new Number = number, Parsed = true ; catch return new Number = number, Parsed = false ; 하지만 dynamic의도입으로정적형식검사를할수없어나중에필드이름이바뀌어도컴파일시문제를알아낼수없다는문제점이발생한다. 또다른해결방법으로는닷넷프레임워크 4.0의 BCL부터제공되는 System.Tuple 제네릭타입을이용하는것이다. 이타입은 7개까지의인자를기본처리할수있도록미리정의돼있으며, public class Tuple<T1> : IStructuralEquatable, [ 생략 ], ITuple public class Tuple<T1, T2> : IStructuralEquatable, [ 생략 ], ITuple public class Tuple<T1, T2, T3> : IStructuralEquatable, [ 생략 ], ITuple 12 장. C# 7.0 737

public class Tuple<T1, T2, T3, T4> : IStructuralEquatable, [ 생략 ], ITuple public class Tuple<T1, T2, T3, T4, T5> : IStructuralEquatable, [ 생략 ], ITuple public class Tuple<T1, T2, T3, T4, T5, T6> : IStructuralEquatable, [ 생략 ], ITuple public class Tuple<T1, T2, T3, T4, T5, T6, T7> : IStructuralEquatable, [ 생략 ], ITuple 8 개이상은또다른 System.Tuple 로이어서처리할수있도록구현돼있다. public class Tuple<T1, T2, T3, T4, T5, T6, T7, TRest> : IStructuralEquatable, [ 생략 ], ITuple 따라서 System.Tuple 을이용해다음과같이문제를해결할수있다. Tuple<bool, int> result = pg.parseinteger("40"); Console.WriteLine(result.Item1); // 첫번째제네릭인자를 Item1 로접근 Console.WriteLine(result.Item2); // 두번째제네릭인자를 Item2 로접근 Tuple<bool, int> ParseInteger(string text) int number = 0; bool result = false; try number = Int32.Parse(text); result = true; catch return Tuple.Create(result, number); dynamic 예약어와는달리이번에는정적형식검사를제공하지만변수의이름이무조건 Item1, Item2, 와같은식으로정해진다는아쉬움이남고문법역시다른언어 ( 예 : 파이썬 ) 에비해복잡하다는문제가있다. 이러한모든문제를해결하기위해마이크로소프트는 C# 언어차원에서튜플을지원하기로결정했고이로인해메서드의정의구문이확장된다. 4.1.2절 메서드 에나오는예제를하나만들어보자. 738 2 부 _ C# 고급문법

메서드가값을반환하는경우반환타입메서드명 ([ 타입명 ] [ 매개변수명 ], ) // 코드 : 메서드의본문 return [ 반환타입에해당하는표현식 ]; 튜플이나오기전에는반환타입과타입명으로단일타입의이름만가능했지만이제는괄호와함께 2 개이상의타입과원한다면각각의이름까지지정할수있다. 튜플의반환타입 ( 반환타입 [ 필드명 ], 반환타입 [ 필드명 ], ) 메서드명 ([ 타입명 ] [ 매개변수명 ], ) // 코드 : 메서드의본문 return ([ 반환타입에해당하는표현식, ]); 튜플의입력타입반환타입메서드명 (([ 타입명 ] [ 매개변수명 ], [ 타입명 ] [ 매개변수명 ], )) 반환타입메서드명 ([ 타입명 ] [ 매개변수명 ], ([ 타입명 ] [ 매개변수명 ], [ 타입명 ] [ 매개변수명 ], )) 반환타입메서드명 (([ 타입명 ] [ 매개변수명 ], [ 타입명 ] [ 매개변수명 ], ), [ 타입명 ] [ 매개변수명 ]) 괄호를사용해 2개이상의타입및그이름을지정할수있고반환타입뿐만아니라매개변수로도전달할수있다. 튜플의각요소는 Item1부터차례대로 Item2, Item3, 의순으로자동명명되지만원하다면이름을직접지정 10 하는것도가능하다. 예 ) int와 string을반환하는경우 (int, string) GetResult() 예 ) bool, Dictionary<int, string> 을반환하는경우 (bool, Dictionary<int, string>) GetResult() 예 ) bool, int를반환하고각각의이름으로 Parsed, Number를반환하는경우 (bool Parsed, int Number) GetResult() 예 ) (bool, int) 튜플을인자로전달하는경우 void SetResult((bool, int) arg1) 1 C# 7.1 부터튜플에대해타입추론이추가되어이름지정을생략할수있는경우가있다 (13.3 절참고 ). 12 장. C# 7.0 739

즉, 괄호를이용해다중값을처리할수있는구문을 C# 7.0 수준에서지원하도록추가한것이다. 이 를이용하면 ParseInteger 메서드의처리를다음과같이간단하게바꿀수있다. (bool, int) result = pg.parseinteger("50"); Console.WriteLine(result.Item1); // 튜플의첫번째인자를 Item1 로접근 Console.WriteLine(result.Item2); // 튜플의두번째인자를 Item2 로접근 (bool, int) ParseInteger(string text) int number = 0; bool result = false; try number = Int32.Parse(text); result = true; catch return (result, number); 반환받은튜플의요소에접근하기위해 System.Tuple 에서처럼 Item1, Item2 로접근하는데, 원한다 면다음과같이직접이름을지정하고 var 예약어로받아접근할수있다. var result = pg.parseinteger("50"); Console.WriteLine(result.Parsed); // 튜플의첫번째인자를 Parsed 로접근 Console.WriteLine(result.Number); // 튜플의두번째인자를 Number 로접근 (bool Parsed, int Number) ParseInteger(string text) // [ 생략 ] return (result, number); 740 2 부 _ C# 고급문법

만약튜플을반환하는메서드가지정한튜플의이름들을원하지않거나또는이름이지정되지않은튜 플인경우라도호출하는측에서강제로이름을지정하는것또한가능하다. (bool success, int n) result = pg.parseinteger("50"); Console.WriteLine(result.success); // 튜플의첫번째인자에 success 로접근 Console.WriteLine(result.n); // 튜플의두번째인자에 n 으로접근 심지어튜플로받지않고개별필드를분해해서받는구문도지원한다. (var success, var number) = pg.parseinteger("50"); Console.WriteLine(success); // 튜플의첫번째인자의값을담은 success 변수 Console.WriteLine(number); // 튜플의두번째인자의값을담은 number 변수 또한 out 매개변수처리에서지원했던생략기호 ( 밑줄 ) 도튜플의반환값을분해하는구문에사용할수 있다. (var _, var _) = pg.parseinteger("70"); // 2 개의값을모두생략 (var _, var n) = pg.parseinteger("70"); // 마지막값만 n 으로받음 Console.WriteLine(n); 앞에서설명한모든튜플구문은 C# 7.0 컴파일러가소스코드를컴파일하는시점에 System. ValueTuple 제네릭타입으로변경해서처리한다. 참고로닷넷프레임워크 4에서제공하는 System. Tuple 타입은 class로정의된반면닷넷프레임워크 4.7에서새롭게제공하는 System.ValueTuple 은 struct로정의돼있다는차이점이있다 ( 다른말로하면, System.Tuple은참조형식이고 System. ValueTuple은값형식이다 ). 12 장. C# 7.0 741

12.4 Deconstruct 메서드 튜플의반환값을분해하는구문은원한다면여러분이만든타입에도 Deconstruct 라는이름의특별한 메서드를 1 개이상정의해구현할수있다. 이때분해가되는개별값을 out 매개변수를이용해처리 하면된다. 문법은다음과같다. 접근 _ 제한자 void Deconstruct(out T1 x1,..., out Tn xn)... T1 ~ Tn 은분해될값을담을타입. out 매개변수이므로반드시값을채워서반환해야함. 다음은분해메서드를정의한클래스이고, class Rectangle public int X get; public int Y get; public int Width get; public int Height get; public Rectangle(int x, int y, int width, int height) X = x; Y = y; Width = width; Height = height; public void Deconstruct(out int x, out int y) x = X; y = Y; public void Deconstruct(out int x, out int y, out int width, out int height) x = X; y = Y; width = Width; 742 2 부 _ C# 고급문법

height = Height; 튜플의분해에서본것과같이그대로사용할수있다. Rectangle rect = new Rectangle(5, 6, 20, 25); (int x, int y) = rect; Console.WriteLine($"x == x, y == y"); // 출력결과 : x == 5, y == 6 (int _, int _) = rect; // 의미없는구문이지만. (int _, int y) = rect; Console.WriteLine($"y == y"); // 출력결과 : y == 6 (int x, int y, int width, int height) = rect; Console.WriteLine($"x == x, y == y, width = width, height = height"); // 출력결과 : x == 5, y == 6, width = 20, height = 25 (int _, int _, int _, int _) = rect; // 의미없는구문이지만. (int _, int _, int w, int h) = rect; Console.WriteLine($"w == w, h == h"); // 출력결과 : w == 20, h == 25 (var _, var _, var _, var last) = rect; Console.WriteLine($"last == last"); // 출력결과 : last == 25 12 장. C# 7.0 743

12.5 람다식을이용한메서드정의확대 (Expression-bodied members) C# 6.0 의 11.2 절 람다식을이용한메서드, 속성및인덱서정의 에서람다식으로메서드의정의가 가능했던유형은다음과같다. 일반메서드 속성의 get 접근자 ( 읽기전용으로처리됨 ) 인덱서의 get 접근자 ( 읽기전용으로처리됨 ) C# 7.0 부터는람다식의접근이다음의메서드정의까지확장됐다. 생성자 (Constructor) 소멸자 (Destructor) 이벤트의 add/remove 속성의 set 구문 인덱서의 set 구문 하지만 C# 6.0에서와마찬가지로단일식 (expression) 인경우에만가능하다는제약은그대로존재한다. 다음은예제 11.1 람다식을이용한메서드본문구현 에서는불가능했던메서드유형에대한람다식메서드정의를추가한것이다. public class Vector double x; double y; public Vector(double x) => this.x = x; // 생성자정의가능 public Vector(double x, double y) // 생성자이지만 2개의문장이므로람다식으로정의불가 this.x = x; this.y = y; 744 2 부 _ C# 고급문법

~Vector() => Console.WriteLine("~dtor()"); // 소멸자정의가능 public double X get => x; set => x = value; // 속성의 set 접근자정의가능 public double Y get => y; set => y = value; // 속성의 set 접근자정의가능 public double this[int index] get => (index == 0)? x : y; set => _ = (index == 0)? x = value : y = value; // 인덱서의 set 접근자정의가능 private EventHandler positionchanged; public event EventHandler PositionChanged // 이벤트의 add/remove 접근자정의가능 add => this.positionchanged += value; remove => this.positionchanged -= value; public Vector Move(double dx, double dy) x += dx; y += dy; positionchanged?.invoke(this, EventArgs.Empty); return this; public void PrintIt() => Console.WriteLine(this); public override string ToString() => string.format("x = 0, y = 1", x, y); 12 장. C# 7.0 745

12.6 지역함수 (Local functions) 메서드의편리한정의를위해 C# 에서는이미다음과같은구문을정의하고있다. C# 2.0 7.7절의익명메서드 C# 3.0 8.8절의람다식 이런노력에도불구하고 C# 7.0에는메서드안에서만호출가능한메서드를 1개이상정의할수있는지역함수문법을추가했다. 문법은메서드정의와완전히같고단지다른메서드의내부에서정의한다는차이점만있다. 예를들어, 정수나눗셈을수행하는기능을우선익명메서드로정의해보자. delegate (bool, int) MyDivide(int a, int b); // 사용할익명메서드에대한델리게이트정의 void AnonymousMethodTest() MyDivide func = delegate (int a, int b) if (b == 0) return (false, 0); return (true, a / b); ; Console.WriteLine(func(10, 5)); Console.WriteLine(func(10, 0)); // 출력결과 (True, 2) (False, 0) 이코드의문제점은익명메서드를정의할때마다해당메서드에부합하는델리게이트가반드시정의 돼있어야한다는점이다. 이를람다식으로바꾼다해도코드만약간간편해질뿐여전히델리게이트 가있어야한다는사실에는변함이없다. 746 2 부 _ C# 고급문법

delegate (bool, int) MyDivide(int a, int b); // 사용할람다식에대한델리게이트정의 void LambdaMethodTest() MyDivide func = (a, b) => if (b == 0) return (false, 0); return (true, a / b); ; Console.WriteLine(func(10, 5)); 하지만이것을지역함수로바꾸면델리게이트정의와상관없이사용할수있다는편리함이있다. void LocalFuncTest() (bool, int) func(int a, int b) if (b == 0) return (false, 0); return (true, a / b); ; Console.WriteLine(func(10, 5)); 또한메서드가단일식으로정의될수있으면 11.2 절의내용대로람다식으로정의하는것도가능하다. void LocalLambdaFuncTest() (bool, int) func(int a, int b) => (b == 0)? (false, 0) : (true, a / b); Console.WriteLine(func(10, 5)); 참고로지역함수에대해 C# 컴파일러는소스코드를 internal 접근자를가진메서드로정의해타입내 에자동으로추가한다. 단지이때의메서드이름이컴파일러에의해정해진임의의이름을사용하기 12 장. C# 7.0 747

때문에다른곳에서부를수없는것뿐이다. 따라서리플렉션을이용해다소억지스럽지만원한다면 호출이가능하다. 12.7 사용자정의 Task 타입을 async 메서드의반환타입으로사용가능 C# 5.0부터구현된 async 예약어가붙는메서드는반환타입이반드시 void, Task, Task<T> 중의하나여야만했다. 달리말하면수많은비동기시나리오에서개발자가최적화할수있는여지를닫아버린것이다. 실제로마이크로소프트에서구현한 async 메서드는성능상불이익이발생할수있다. 예를들어, async 예약어가적용되면 C# 컴파일러는자동으로 Task 객체를사용하도록코드를변환하는데 async 메서드내에서 await을호출하지않아비동기로처리될필요가없을때조차도 Task 객체가생성되어성능상불이익을받게된다는문제점이발생한다. 이번절의내용을실습하려면반드시 NuGet 패키지관리자로부터 System.Threading.Tasks.Extensions 를참조추가해야한다. 이를위해 도구 (Tools) NuGet 패키지관리자 (NuGet Package Manager) 패키지관리자콘솔 (Package Manager Console) 메뉴를선택하면나오는창에서다음과같은명령을실행한다. PM> Install-Package System.Threading.Tasks.Extensions -Version 4.4.0 C# 7.0부터는사용자정의 Task 타입을구현하고이를 async의반환타입으로사용할수있도록허용한다. 물론이런타입을직접만드는방법이그리쉽지는않기때문에범용적으로확산되지는않겠지만확장이가능하다는면에서충분히의미를둘만하다. 게다가마이크로소프트는사용자정의 Task 타입의한사례로값형식 (struct) 의 ValueTask<T> 타입을구현했는데바로이타입이 async 메서드내에서 await을호출하지않았을때불필요한 Task 객체생성을생략함으로써성능향상을이뤄낸대표적인경우다. 그럼 ValueTask<T> 의구체적인차이점을이해하기위해 async 메서드에서값을반환하는메서드를추가하는예제로시작해보자. using System; using System.IO; using System.Threading; using System.Threading.Tasks; 748 2 부 _ C# 고급문법

class Program static void Main(string[] args) Task<(string, int tid)> result = FileReadAsync(@"C:\windows\system32\drivers\etc\HOSTS"); int tid = Thread.CurrentThread.ManagedThreadId; Console.WriteLine($"MainThreadID: tid, AsyncThreadID: result.result.tid"); private static async Task<(string, int)> FileReadAsync(string filepath) string filetext = await ReadAllTextAsync(filePath); return (filetext, Thread.CurrentThread.ManagedThreadId); static Task<string> ReadAllTextAsync(string filepath) return Task.Factory.StartNew(() => return File.ReadAllText(filePath); ); // 출력결과 MainThreadID: 1, AsyncThreadID: 3 보다시피 FileReadAsync 메서드의 return 문은그전의 await 호출로인해비동기처리되어 3 번스 레드에서실행됐고이후 Main 메서드의 Console.WriteLine 코드는원래의 1 번스레드에서실행됐 다. 이제 FileReadAsync 코드를다음과같이고쳐보자. static string _filecontents = string.empty; private static async Task<(string, int)> FileReadAsync(string filepath) if (string.isnullorempty(_filecontents) == false) 12 장. C# 7.0 749

return (_filecontents, Thread.CurrentThread.ManagedThreadId); _filecontents = await ReadAllTextAsync(filePath); return (_filecontents, Thread.CurrentThread.ManagedThreadId); // FileReadAsync 메서드를두번호출해출력한결과 MainThreadID: 1, AsyncThreadID: 3 MainThreadID: 1, AsyncThreadID: 1 위의메서드는첫번째로호출한경우에는 await의호출로비동기처리되지만두번째호출부터는 await 코드를실행하지않고동기방식으로호출을완료한다. 하지만그래도 Task<T> 객체가생성된다는점에는변함이없다. C# 7.0부터이메서드를다음과같이개선할수있다. ValueTask<(string, int tid)> result = FileReadAsync(@"C:\windows\system32\drivers\etc\HOSTS"); int tid = Thread.CurrentThread.ManagedThreadId; Console.WriteLine($"MainThreadID: tid, AsyncThreadID: result.result.tid"); private static async ValueTask<(string, int)> FileReadAsync(string filepath) if (string.isnullorempty(_filecontents) == false) return (_filecontents, Thread.CurrentThread.ManagedThreadId); _filecontents = await ReadAllTextAsync(filePath); return (_filecontents, Thread.CurrentThread.ManagedThreadId); 750 2 부 _ C# 고급문법

사용법은단순히 Task<T> 를 ValueTask<T> 로바꾼것뿐이지만 FileReadAsync 메서드를첫번째로호출한경우에만 ValueTask<T> 가내부적으로 Task<T> 객체를생성해처리하고두번째호출의경우 await 호출을수행하지않으므로 Task<T> 객체생성없이 ValueTask<T> 단독으로처리함으로써비동기호출의성능을향상시킨다. 12.8 자유로워진 throw 사용 throw 구문은 3.5절의제어문과마찬가지로식 (expression) 이아닌문 (statement) 에해당한다. 이때문에표현식에서의사용이제한됐으므로 C# 6.0 이전에는이럴때 throw를포함한별도의메서드를만들어호출하는방법으로해결해야만했다. 가령다음코드는컴파일시 잘못된식의항 throw 입니다 (Invalid expression term throw ) 라는오류가발생하므로 public bool Assert(bool result) => #if DEBUG result == true? result : throw new ApplicationException("ASSERT"); #else result; #endif 다음과같이메서드를이용해우회해서 throw 를해야만했다. public bool Assert(bool result) => #if DEBUG result == true? result : AssertException("ASSERT"); #else result; #endif public bool AssertException(string msg) throw new ApplicationException(msg); 12 장. C# 7.0 751

하지만 C# 7.0 부터는 throw 가의미있게사용될만한식에서허용이되도록바뀌었다. 따라서위코 드를다음과같이변경해서컴파일할수있다. public void Assert(bool result) => #if DEBUG _ = result == true? result : throw new ApplicationException("ASSERT"); #else _ = result; #endif 이밖에도다음예제코드에서볼수있는것처럼 null 병합연산자 (??) 와람다식을사용할수있는곳 에서 throw 를사용할수있다. (C# 6.0 에서는모두컴파일오류가발생하는코드다 ). class Person public string Name get; // 7.2 절의 null 병합연산자 (??) 에서 throw 사용가능 public Person(string name) => Name = name?? throw new ArgumentNullException(nameof(name)); // 11.2 절의람다식으로정의된메서드에서사용가능 public string GetLastName() => throw new NotImplementedException(); public void Print() // 8.8.1.1절의단일람다식을이용한델리게이트정의에서사용가능 Action action = () => throw new Exception(); action(); 그렇다고해서 throw 가문에서식으로완전히바뀐것은아니므로표현식이허용되는모든곳에서사 용할수는없다. 대신 throw 가허용되지않는곳에서는명시적으로다음과같은컴파일에러가발생 한다. error CS8115: 이컨텍스트에서는 throw 식을사용할수없습니다. error CS8115: A throw expression is not allowed in this context. 752 2 부 _ C# 고급문법

12.9 리터럴에대한표현방법개선 (Literal improvements) 보통숫자데이터가길어지면코드에적거나읽을때상대적으로가독성이떨어진다. 가령정수로 1천만을담는변수를정의하려고할때 10000000 값을기입하는것보다 10,000,000과같이쓰는것이가독성측면에서더좋을수밖에없다. 아쉽게도콤마 (,) 기호를쓸수는없지만 C# 7.0부터숫자내의임의의위치에밑줄을추가할수있도록허용하고있다. int number1 = 10000000; // 1천만인지한눈에인식이안됨 int number2 = 10_000_000; // 세자리마다띄어한눈에 1천만임을알수있음 int number3 = 1_0_0_0_0_000; // 잘쓰진않겠지만이런식으로도가능 숫자데이터에적용할수있기때문에 16 진수로표현했을때도사용가능하다. uint hex1 = 0xFFFFFFFF; uint hex2 = 0xFF_FF_FF_FF; // 1 바이트마다띄어서표현 uint hex3 = 0xFFFF_FFFF; // 2 바이트마다띄어서표현 이와함께 2 진비트열의리터럴표현도추가됐다. 접두사로 0b 를사용하며, 역시밑줄과함께쓰면가 독성높은 2 진비트열을정의하는것이가능하다. // 숫자 4369 를표현한 2 진비트열 uint bin1 = 0b0001000100010001; // 2 진비트열로숫자데이터정의 uint bin2 = 0b0001_0001_0001_0001; // 4 자리마다구분자를사용해가독성을높임 12.10 패턴매칭 패턴매칭이란단어는흔히다음과같이사용되곤한다. 이미지에대한패턴매칭 문자열에대한패턴매칭 C# 언어에서의패턴매칭이라는단어를이해하려면위와같이 ~ 에대한 으로생각하면된다. 실제로 이번절의패턴매칭제목을다음과같이정의하고바라보면 12 장. C# 7.0 753

객체에대한패턴매칭 이후설명할내용을좀더직관적으로이해하는데도움될것이다. 우선 C# 7.0 부터추가된패턴매칭유형은크게다음과같이세가지 2 로나뉜다. 상수패턴 (Constant patterns) 타입패턴 (Type patterns) Var 패턴 (Var patterns) 각패턴은 객체에대해 상수값인지아니면주어진어떤타입과일치 (match) 하는지테스트할수있 다. 그리고위의세가지패턴을 C# 코드에적용할수있도록기존의 is 예약어와 switch/case 구문이 확장된다. 12.10.1 is 연산자의패턴매칭 is 연산자는기존기능에서패턴매칭을지원하기위해 as 연산자의기능을흡수했다. 예를들어, as 연산자는기존의 is 연산자와는달리변환결과를포함할수있어서변환여부만알수있었던 is 연산자보다더자주사용되곤했다. object obj = new List<string>(); List<string> list = obj as List<string>; list?.foreach((e) => Console.WriteLine(e)); 하지만 C# 7.0 부터는 is 연산자가위에서했던 as 연산자와동일한역할을수행할수있다. if (obj is List<string> list) list.foreach((e) => Console.WriteLine(e)); 물론기존의 is 연산자처럼타입만비교하는것도여전히가능하다. 2 패턴매칭이가능한유형은이후의 C# 버전에서늘어날예정이다. 일례로 C# 7.1 에는이미제네릭에대한패턴매칭을포함하고있다 (13.4 절참고 ). 754 2 부 _ C# 고급문법

if (obj is List<string>) [ 생략 ] 즉, is 연산자가대상타입에대한비교를할수있었던것과함께그뒤에변수명을추가할수있게만듦으로써 as 연산자의기능까지포함하게된것이다. 이렇게강화된 is 연산자는세가지패턴매칭유형을모두사용할수있지만그중에서상수패턴과타입패턴만이의미가있다. 다음은그두가지패턴유형을사용한예다. 예제 12.1 is 연산자를이용한상수패턴과타입패턴사례 object[] objlist = new object[] 100, null, DateTime.Now, new ArrayList() ; foreach (object item in objlist) if (item is 100) // 상수패턴 Console.WriteLine(item); else if (item is null) // 상수패턴 Console.WriteLine("null"); else if (item is DateTime dt) // 타입패턴 ( 값형식 ) 필요없다면 dt 변수생략가능 Console.WriteLine(dt); else if (item is ArrayList arr) // 타입패턴 ( 참조형식 ) 필요없다면 arr 변수생략가능 Console.WriteLine(arr.Count); // 출력결과 100 null 2017-08-20 오후 7:07:58 0 의미가없다고했던 Var 패턴의사용법을알아보자. 참고로 Var 패턴의경우반드시변수명을함께써 야한다. 12 장. C# 7.0 755

if (item is var elem) // 타입패턴과는달리변수명을생략할수없다. Console.WriteLine(elem); // 단지변수가필요없는경우밑줄로대체가능 if (item is var _ ) var 예약어의의미상 item 변수는무조건 var 타입에일치하게되어 if 문에사용하는경우항상 true 로평가된다. 위코드를 C# 7.0 컴파일러가변환한코드를보면왜 Var 패턴이 is 연산자에서의미가 없는지더욱분명하게알수있다. object elem = item; if (true) Console.WriteLine(elem); 예상했던대로항상 true 가되는조건이기때문에 is 연산자에서 Var 패턴은적당한사용용도를찾을 수없다. 아무리좋게말해도단순히변수명을 item 에서 elem 으로바꿔주는역할만할뿐이다. 12.10.2 switch/case 문의패턴매칭 3.5.1.3 절에서배운 switch 문은 C# 7.0 에서패턴매칭의적용으로문법이바뀐다. switch ( 인스턴스 ) case 패턴 _ 매칭 _ 식1 : 구문 ; break; [ 임의의 case 문반복 ] case 패턴 _ 매칭 _ 식n: 구문 ; break; 756 2 부 _ C# 고급문법

default: 구문 ; break; 설명 : 인스턴스의값과컴파일또는실행시에결정되는 case 의패턴 _ 매칭 _ 식결괏값이일치하는경우해당 case 에속한구 문을실행한다. 나열된 case 의패턴 _ 매칭 _ 식에일치하는값이없다면 default 에지정된구문의코드를실행한다. 인스턴스의타입유형에는제약이없다. 이로인해 C# 6.0 이전까지는일부 if 문의특수한경우에만 switch 문으로변경할수있었는데 C# 7.0 부터는모든 if 문을 switch 문으로변경할수있고그반대도가능하다. case 조건의패턴매칭문법은기본적으로 is 연산자와같다. 예제 12.1의 is 연산자예제를 switch/ case로바꾸는경우 is만뺐을뿐패턴매칭을위한코드가완전히같음을알수있다. foreach (object item in objlist) switch (item) case 100: // 상수패턴 break; case null: // 상수패턴 break; case DateTime dt: // 타입패턴 ( 값형식 ) 필요없다면 dt 변수생략가능 break; case ArrayList arr: // 타입패턴 ( 참조형식 ) 필요없다면 arr 변수생략가능 break; case var elem: // Var 패턴 ( 이렇게사용하면 default 와동일 ) break; 12 장. C# 7.0 757

그런데 case 조건의패턴매칭에는한가지혜택이더있는데바로 when 예약어를추가해조건을한번더검사할수있다는것이다. case 조건에서의 when 예약어기능은 11.8절 예외필터 에서다룬 when의사용법과일치한다. when의추가로 if 문의다음과같은조건도 int j = GetIntegerResult(); if (j > 300) // j 값이 300보다큰경우 else switch 에서처리할수있게됐다. switch (j) case int i when i > 300: // int 타입이고그값이 300보다큰경우 break; default: break; case 조건의 when 예약어추가는 (is 연산자에서는쓸모없었던 ) Var 패턴에또다른생명력을불어넣어주는데, 바로사용자정의패턴매칭구현을가능하게한다는점이다. 즉, 일단 Var 패턴으로받고 when을사용해어떠한조건이라도설정할수있게만들수있다. 가령, 어떤문자열이네이버의홈페이지에포함돼있는지다음 (daum) 의홈페이지에포함돼있는지에대한조건을다음과같이설정할수있다. // 기존에는아래와같은조건은 if 문을통해서만가능했지만 // 이젠 switch/case로도처리할수있다. string text = " "; switch (text) 758 2 부 _ C# 고급문법

case var item when (ContainsAt(item, "http://www.naver.com")): Console.WriteLine("In Naver"); break; case var item when (ContainsAt(item, "http://www.daum.net")): Console.WriteLine("In Daum"); break; default: Console.WriteLine("None"); break; private static bool ContainsAt(string item, string url) WebClient wc = new WebClient(); wc.encoding = Encoding.UTF8; string text = wc.downloadstring(url); return text.indexof(item)!= -1; 이정도면 case 에만 when 을허용한것이 is 연산자로써는아쉬울따름이다. 이쯤에서함수형프로그래밍언어인 F# 의패턴매칭과살짝비교해보자. // F# 예제코드 // https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/pattern-matching let detectzeroor point = match point with (0, 0) (0, _) (_, 0) -> printfn "Zero found." _ -> printfn "Both nonzero." detectzeroor (0, 0) detectzeroor (1, 0) detectzeroor (0, 10) detectzeroor (10, 15) 12 장. C# 7.0 759

위의 F# 코드는 2 개의값중에서하나라도 0 을포함한다면 Zero found. 를출력하고, 없다면 Both nonzero. 를출력한다. 이를 C# 의패턴매칭으로바꾸면다음과같이구현할수있다. Action<(int, int)> detectzeroor = (arg) => switch (arg) case var r when r.equals((0, 0)): case var r1 when r1.item1 == 0: case var r2 when r2.item2 == 0: Console.WriteLine("Zero found."); return; ; Console.WriteLine("Both nonzero."); detectzeroor((0, 0)); detectzeroor((1, 0)); detectzeroor((0, 10)); detectzeroor((10, 15)); 코드의간결함에있어서는 F# 을따라갈순없지만그런대로 C# 에서도유사하게처리할수있어 C# 6.0 이전보다더유연한프로그래밍이가능해진다. 마지막으로, 패턴매칭이적용되면서 case 의평가순서를다시한번확인해둘필요가있다. default 절은그위치에상관없이항상마지막에평가된다. case의순서가중요해지는데상위의 case 조건에서이미만족한다면그절의코드가실행되고그하위의 case 조건 에대한평가는무시된다. 760 2 부 _ C# 고급문법

13 C# 7.1 C# 언어에대한로드맵문서 1 에따르면 C# 7.1 에는 5 가지기능이예정돼있다. Async Main Default Expressions Ref Assemblies Infer tuple names Pattern-matching with generics 이가운데 Ref Assemblies 를제외한 4개의기능에대해서는 2017년 8월 18일 2 에업데이트된비주얼스튜디오 2017의 15.3.1 버전부터제공된다. 하지만기본컴파일러로는아직 7.1로설정돼있지않으므로프로젝트설정창에서별도로지정해야만한다. 따라서이번장을실습하려면 C# 프로젝트를만든후프로젝트설정창의 빌드 (Build) 고급 (Advanced) 버튼을눌러나오는 고급빌드설정 (Advanced Build Settings) 창에서다음과같이 언어버전 (Language version) 값을 C# 7.1 로바꿔야한다. 1 https://github.com/dotnet/roslyn/blob/master/docs/language%20feature%20status.md 2 비주얼스튜디오 15.3.1 의릴리스노트를통해 C# 7.1 의 4 가지기능이추가됐다는사실을 8 월 20 일에이책의 7.0 원고를마치며알게됐다. 13 장. C# 7.1 761

참고로컴퓨터에설치된비주얼스튜디오의버전을알고싶다면 도움말 (Help) Microsoft Visual Studio 정보 (About Microsoft Visual Studio) 메뉴를선택해서확인할수있다. 762 2 부 _ C# 고급문법

13.1 Main 메서드에 async 예약어허용 C# 7.0 까지 Main 메서드에는 async 예약어를적용할수없었다. 왜냐하면 4.1.5.2 절에서설명한대 로 Main 메서드는다음과같은유형만가능하기때문이다. static void Main() static void Main(string[]) static int Main() static int Main(string[]) 즉 async 예약어를적용할수없는 Main 메서드는 await 호출을할수없어다음과같은식으로우회 해서사용해야만했다. static void Main(string[] args) MainAsync(); // 별도의 async 메서드를만들어호출 private static async Task MainAsync() WebClient wc = new WebClient(); string text = await wc.downloadstringtaskasync("http://www.microsoft.com"); Console.WriteLine(text); 그런데위코드를컴파일하면경고가발생한다. CS4014: 이호출이대기되지않으므로호출이완료되기전에현재메서드가계속실행됩니다. 호출결과에 'await' 연산자를적용해보세요. CS4014: Because this call is not awaited, execution of the current method continues before the call is completed. Consider applying the 'await' operator to the result of the call. 경고메시지에나오는조언과는달리 Main 메서드는 async 예약어가허용된메서드가아니므로경고 에따라 await 을 MainAsync 에붙이는것도불가능하다. 그래서이와같은경고를없애기위해다음 과같이코드를변경해실행하는방법을권장한다. 13 장. C# 7.1 763

static void Main(string[] args) MainAsync().GetAwaiter().GetResult(); 이같은우회방법을뒤로하고, 결국마이크로소프트는 C# 7.1 부터 async 를허용하도록바꾸어이러 한모든부자연스러운문제를깔끔하게해결한다. static async Task Main(string [] args) WebClient wc = new WebClient(); string text = await wc.downloadstringtaskasync("http://www.microsoft.com"); Console.WriteLine(text); 참고로표현식하나로도호출이가능하다면람다식유형으로정의하는것도가능하다. static async Task Main(string[] args) => Console.WriteLine(await new WebClient(). DownloadStringTaskAsync("http://www.microsoft.com")); 정리하자면, C# 7.1 부터다음과같은새로운유형의 Main 메서드정의가가능하다. static async Task Main() static async Task Main(string[]) static async Task<int> Main() static async Task<int> Main(string[]) 13.2 default 리터럴추가 C# 2.0 부터구현됐던 default 예약어에대해 C# 7.1 부터는타입추론이가능해져서리터럴형식으 로쓸수있게바뀐다. 실제로 7.3 절의 default 예약어를설명하기위해만든모든예제코드에대해 default(type) 형식이아닌 default 로바꿔도그대로컴파일이가능하다. 764 2 부 _ C# 고급문법

static void Main(string[] args) int intvalue = default; // int로대입된다는것을알수있으므로 BigInteger bigintvalue = default; // BigInteger로대입되므로 Console.WriteLine(intValue); // 출력결과 : 0 Console.WriteLine(bigIntValue); // 출력결과 : 0 string txt = default; // string 타입의기본값을반환 Console.WriteLine(txt?? "(null)"); // 출력결과 : (null) C# 컴파일러입장에서는 default 대상이되는타입을추론할수있으므로굳이타입을지정할필요가 없는것이다. 마찬가지로제네릭인자에도 default 리터럴을쓸수있다. static void Main(string[] args) GenericTest<int> t = new GenericTest<int>(); Console.WriteLine(t.GetDefaultValue()); // 출력결과 : 0 class GenericTest<T> public T GetDefaultValue() return default; // C# 7.0 이전에는 default(t) 로반환 13.3 타입추론을통한튜플의변수명자동지정 default 예약어에대한타입추론과함께 C# 7.1 부터는튜플의변수명에대해서도타입추론을활용 해사용의편의성을높였다. 가령 C# 7.0 에서다음과같이튜플을만드는경우, 13 장. C# 7.1 765

int age = 20; string name = "Kevin Arnold"; (int, string) person = (age, name); Console.WriteLine($"person.Item1, person.item2"); Item1, Item2 이름을없애기위해서는명시적으로이름을지정해야한다. (int age, string name) person = (age, name); Console.WriteLine($"person.age, person.name"); 하지만 C# 7.1 부터는튜플에대입된변수명을타입추론을통해알수있으므로다음과같이이름을 자동으로붙여주는것으로바뀌었다. int age = 20; string name = "Kevin Arnold"; var t = (age, name); Console.WriteLine($"t.age, t.name"); 물론추론이안되는필드에대해서는예전처럼 Item1, Item2, 순으로매겨진다. var person = new Age = 30, Name = "Winnie Cooper" ; var t = (25, person.name); Console.WriteLine($"t.Item1, t.name"); 튜플에대한이같은지원은 8.9 절에서배운 LINQ 구문을좀더자연스럽게만들어준다. 가령 LINQ 의 select 를통해익명타입을반환하는다음의예제를보자. List<Person> people = new List<Person> new Person Name = "Tom", Age = 63, Address = "Korea", new Person Name = "Winnie", Age = 40, Address = "Tibet", new Person Name = "Anders", Age = 47, Address = "Sudan", 766 2 부 _ C# 고급문법

; new Person Name = "Hans", Age = 25, Address = "Tibet", new Person Name = "Eureka", Age = 32, Address = "Sudan", new Person Name = "Hawk", Age = 15, Address = "Korea", var datelist = from person in people select new Name = person.name, Year = DateTime.Now.AddYears(-person.Age).Year ; foreach (var item in datelist) Console.WriteLine(string.Format("0-1", item.name, item.year)); class Person public string Name get; set; public int Age get; set; public string Address get; set; 우선익명타입의번거로움을튜플을사용하면다음과같이간결하게바꿀수있다. var datelist = from person in people select (person.name, DateTime.Now.AddYears(-person.Age).Year); foreach (var item in datelist) Console.WriteLine(string.Format("0-1", item.item1, item.item2)); 하지만대개의경우위의코드에서 Item1 과 Item2 대신이름을지정하고싶을텐데그럼코드가다소 번거로워진다. var datelist2 = from person in people select (Name: person.name, Year: DateTime.Now.AddYears(-person.Age).Year); foreach (var item in datelist2) 13 장. C# 7.1 767

Console.WriteLine(string.Format("0-1", item.name, item.year)); 바로이때 C# 7.1 부터는타입추론을통해튜플내에지정된속성의이름을그대로가져오므로다음 과같이쓸수있다. var datelist = from person in people select (person.name, DateTime.Now.AddYears(-person.Age).Year); foreach (var item in datelist) Console.WriteLine(string.Format("0-1", item.name, item.year)); 13.4 패턴매칭 제네릭추가 C# 7.0 컴파일러에추가된 12.10 절의패턴매칭은제네릭인자에대한패턴매칭구문을허용하지않 는다. 예를들어, 다음코드는 C# 7.0 에서컴파일오류가발생한다. static void Main(string[] args) WriteLog(DateTime.Now); WriteLog(DateTime.UtcNow); // 제네릭인자의객체는 is 연산자와 switch/case 패턴매칭구문에서허용되지않으므로 // 컴파일오류발생 (C# 7.0) public static void WriteLog<T>(T item) if (item is DateTime time) Console.WriteLine(time.ToString()); 768 2 부 _ C# 고급문법

switch (item) case DateTime dt when dt.kind == DateTimeKind.Utc: Console.WriteLine(dt.ToLocalTime()); break; case DateTime dt when dt.kind == DateTimeKind.Unspecified: Console.WriteLine("Invalid DateTime Kind"); break; case DateTime dt: Console.WriteLine(dt); break; 하지만약속했던대로 C# 7.1 부터제네릭에대한패턴매칭이가능하기때문에정상적으로컴파일 된다. 13 장. C# 7.1 769