Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 6개 있습니다.)

C# 7.3 - 초기화 식에서 변수 사용 가능(expression variables in initializers)

이번 글로써 C# 7.3 기능 명세에 나온,

Language Feature Status
; https://github.com/dotnet/roslyn/blob/master/docs/Language%20Feature%20Status.md

모든 내용을 다루게 됩니다.

C# 7.3 (1) - 개선된 문법 4개(Support == and != for tuples, Ref Reassignment, Constraints, Stackalloc initializers)
; https://www.sysnet.pe.kr/2/0/11552

C# 7.3 (2) - 개선된 메서드 선택 규칙 3가지(Improved overload candidates)
; https://www.sysnet.pe.kr/2/0/11553

C# 7.3 (3) - 자동 구현 속성에 특성 적용 가능(Attribute on backing field)
; https://www.sysnet.pe.kr/2/0/11554

C# 7.3 (4) - 사용자 정의 타입에 fixed 적용 가능(Custom fixed)
; https://www.sysnet.pe.kr/2/0/11555

C# 7.3 (5) - 구조체의 고정 크기를 갖는 fixed 배열 필드에 대한 직접 접근 가능(Indexing movable fixed buffers)
; https://www.sysnet.pe.kr/2/0/11556

C# 7.3 (6) - blittable 제네릭 제약(blittable)
; https://www.sysnet.pe.kr/2/0/11558

C# 7.3 (7) - 초기화 식에서 변수 사용 가능(expression variables in initializers)
; https://www.sysnet.pe.kr/2/0/11560

이 중에서 문서에 나오지 않는 것으로 "strongname"이 있는데 딱히 이렇다 할 부가적인 문서가 없어 현시점(2018-06-25)에서 다룰 수가 없습니다. 여기저기 흩어진 내용으로 볼 때 어셈블리에 대한 "강력한 형식 이름" 부여와 관련해 C# 컴파일러의 옵션이 추가되는 듯한데, 따라서 문법적인 변화와는 상관이 없는 것 같습니다. 결국 실질적인 변화로는 위의 7개 글에서 모두 다룬 것입니다.




자, 그럼 마지막 항목을 가볍게 다뤄볼까요? ^^ 아래의 코드는 속성, 필드 및 생성자의 초기화를 비롯해 LINQ 쿼리 절에서 식(expression)을 사용하고 있는 예제입니다.

// 코드 참조: https://github.com/dotnet/csharplang/issues/32
using System;

class UserType
{
    // 속성 초기화에 식 사용 가능
    int Number { get; set; } = int.TryParse("5", out int _) ? 0 : -1;

    // 필드 초기화에 식 사용 가능
    private readonly bool _field1 = int.TryParse("5", out _);
}

class Base
{
    bool _isFive;

    public Base(bool isFive)
    {
        _isFive = isFive;
    }
}

class UserCtor : Base
{
    // 생성자의 초기화 구문에 식 사용 가능
    public UserCtor(string s) : base(int.TryParse("5", out _))
    {
    }
}

class Program
{
    static void Main()
    {
        string[] strings = new string[] { "test", "is", "good" };

        // LINQ 쿼리 내에 식 사용 가능
        var query1 = from text in strings
                     where int.TryParse(text, out int _)
                     select text;
    }
}

그런데, 위와 같은 식 안에서 변수를 사용하는 경우(expression variables)에는 컴파일 오류가 발생합니다.

using System;

class UserType
{
    // 속성 초기화 시 변수 사용 불가능
    // C# 7.2까지 컴파일 오류 - Error CS8200 Out variable and pattern variable declarations are not allowed within constructor initializers, field initializers, or property initializers.
    int Number { get; set; } = int.TryParse("5", out int i) ? i : -1;

    // 필드 초기화 시 변수 사용 불가능
    // C# 7.2까지 컴파일 오류 - Error CS8200 Out variable and pattern variable declarations are not allowed within constructor initializers, field initializers, or property initializers.
    private readonly int _field1 = int.TryParse("5", out int value) ? value : -1;
}

class Base
{
    int _number;

    public Base(int number)
    {
        _number = number;
    }
}

class UserCtor : Base
{
    // 생성자의 초기화 식에서 변수 사용 불가능
    // C# 7.2까지 컴파일 오류 - Error CS8200 Out variable and pattern variable declarations are not allowed within constructor initializers, field initializers, or property initializers.
    public UserCtor(string s) : base(int.TryParse("5", out int i) ? i : -1)
    {
    }
}

class Program
{
    static void Main()
    {
        string[] strings = new string[] { "5", "6", "good" };

        // LINQ Query 절에서 변수 사용 불가능
        // C# 7.2까지 컴파일 오류 - Error CS8201 Out variable and pattern variable declarations are not allowed within a query clause.
        {
            var numbers = from text in strings
                         let t = (int.TryParse(text, out var v), v)
                         where t.Item1
                         select t.Item2;

            foreach (var item in numbers)
            {
                Console.WriteLine(item);
            }
        }
    }
}

그리고 이러한 제약이 C# 7.3부터 풀렸다고 합니다.

expression variables in initializers
; https://github.com/dotnet/csharplang/blob/master/proposals/csharp-7.3/expression-variables-in-initializers.md

We extend the features introduced in C# 7 to permit expressions containing expression variables (out variable declarations and declaration patterns) in field initializers, property initializers, ctor-initializers, and query clauses.

즉 다음의 4가지 구문에서,

  1. field initializers
  2. property initializers
  3. ctor-initializers
  4. query clauses

"out variable declarations and declaration patterns"의 사용이 가능해진 것입니다. 보니까, out 변수 말고도 "declaration patterns"도 사용 가능하다고 하는데요, 테스트는 다음과 같이 할 수 있습니다.

// 초기화 구문에서 패턴 매칭 시 변수 사용 불가능
// C# 7.2까지 컴파일 오류 - Error CS8200 Out variable and pattern variable declarations are not allowed within constructor initializers, field initializers, or property initializers.
int Number3 { get; set; } = 5 is int value ? value : 0;

// LINQ Query 절에서 패턴 매칭 시 변수 사용 불가능
// C# 7.2까지 컴파일 오류 - Error CS8201 Out variable and pattern variable declarations are not allowed within a query clause.
{
    object[] list = new object[] { 5, 6, "good" };
    var numbers =
        from item in list
        let t = item is int value ? (true, value) : (false, 0)
        where t.Item1
        select t.Item2;

    foreach (var item in numbers)
    {
        Console.WriteLine(item);
    }
}

물론, 위의 코드는 C# 7.3부터 정상적으로 컴파일이 됩니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 3/2/2019]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 61  62  63  64  65  66  67  68  [69]  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12211정성태4/27/202019310개발 환경 구성: 486. WSL에서 Makefile로 공개된 리눅스 환경의 C/C++ 소스 코드 빌드
12210정성태4/20/202020781.NET Framework: 903. .NET Framework의 Strong-named 어셈블리 바인딩 (1) - app.config을 이용한 바인딩 리디렉션 [1]파일 다운로드1
12209정성태4/13/202017457오류 유형: 614. 리눅스 환경에서 C/C++ 프로그램이 Segmentation fault 에러가 발생한 경우 (2)
12208정성태4/12/202016036Linux: 29. 리눅스 환경에서 C/C++ 프로그램이 Segmentation fault 에러가 발생한 경우
12207정성태4/2/202015941스크립트: 19. Windows PowerShell의 NonInteractive 모드
12206정성태4/2/202018523오류 유형: 613. 파일 잠금이 바로 안 풀린다면? - The process cannot access the file '...' because it is being used by another process.
12205정성태4/2/202015154스크립트: 18. Powershell에서는 cmd.exe의 명령어를 지원하진 않습니다.
12204정성태4/1/202015178스크립트: 17. Powershell 명령어에 ';' (semi-colon) 문자가 포함된 경우
12203정성태3/18/202018028오류 유형: 612. warning: 'C:\ProgramData/Git/config' has a dubious owner: '...'.
12202정성태3/18/202021248개발 환경 구성: 486. .NET Framework 프로젝트를 위한 GitLab CI/CD Runner 구성
12201정성태3/18/202018492오류 유형: 611. git-credential-manager.exe: Using credentials for username "Personal Access Token". [1]
12200정성태3/18/202018579VS.NET IDE: 145. NuGet + Github 라이브러리 디버깅 관련 옵션 3가지 - "Enable Just My Code" / "Enable Source Link support" / "Suppress JIT optimization on module load (Managed only)"
12199정성태3/17/202016228오류 유형: 610. C# - CodeDomProvider 사용 시 Unhandled Exception: System.IO.DirectoryNotFoundException: Could not find a part of the path '...\f2_6uod0.tmp'.
12198정성태3/17/202019588오류 유형: 609. SQL 서버 접속 시 "Cannot open user default database. Login failed."
12197정성태3/17/202018935VS.NET IDE: 144. .NET Core 콘솔 응용 프로그램을 배포(publish) 시 docker image 자동 생성 - 두 번째 이야기 [1]
12196정성태3/17/202016024오류 유형: 608. The ServicedComponent being invoked is not correctly configured (Use regsvcs to re-register).
12195정성태3/16/202018336.NET Framework: 902. C# - 프로세스의 모든 핸들을 열람 - 세 번째 이야기
12194정성태3/16/202021035오류 유형: 607. PostgreSQL - Npgsql.NpgsqlException: sorry, too many clients already
12193정성태3/16/202018019개발 환경 구성: 485. docker - SAP Adaptive Server Enterprise 컨테이너 실행 [1]
12192정성태3/14/202020042개발 환경 구성: 484. docker - Sybase Anywhere 16 컨테이너 실행
12191정성태3/14/202021109개발 환경 구성: 483. docker - OracleXE 컨테이너 실행 [1]
12190정성태3/14/202015724오류 유형: 606. Docker Desktop 업그레이드 시 "The process cannot access the file 'C:\Program Files\Docker\Docker\resources\dockerd.exe' because it is being used by another process."
12189정성태3/13/202021333개발 환경 구성: 482. Facebook OAuth 처리 시 상태 정보 전달 방법과 "유효한 OAuth 리디렉션 URI" 설정 규칙
12188정성태3/13/202026094Windows: 169. 부팅 시점에 실행되는 chkdsk 결과를 확인하는 방법
12187정성태3/12/202015699오류 유형: 605. NtpClient was unable to set a manual peer to use as a time source because of duplicate error on '...'.
12186정성태3/12/202017456오류 유형: 604. The SysVol Permissions for one or more GPOs on this domain controller and not in sync with the permissions for the GPOs on the Baseline domain controller.
... 61  62  63  64  65  66  67  68  [69]  70  71  72  73  74  75  ...