[C#] ‘T’유형의 값은로 변환 할 수 없습니다

이것은 초보자 질문 일 가능성이 있지만 Google은 놀랍게도 답변을 제공하지 않았습니다.

이 인공적인 방법이 있습니다

T HowToCast<T>(T t)
{
    if (typeof(T) == typeof(string))
    {
        T newT1 = "some text";
        T newT2 = (string)t;
    }

    return t;
}

C ++ 배경에서 오는 것은 이것이 효과가 있다고 예상했습니다. 그러나 위의 두 가지 할당 모두에 대해 ” ‘T’유형을 문자열로 암시 적으로 변환 할 수 없음”및 ” ‘T’유형을 문자열로 변환 할 수 없음”으로 컴파일되지 않습니다.

개념적으로 잘못된 일을하거나 구문이 잘못되었습니다. 이걸 정리해주세요.

감사합니다!



답변

그것이 내부 비록 if블록, 컴파일러는 모르는 T것입니다 string.
따라서 캐스팅 할 수 없습니다. (당신이 캐스팅 할 수없는 것과 같은 이유 DateTimestring)

당신은에 캐스트 할 필요가 object(이 어떤, T위해 할 수있는 캐스트), 거기에에서 string(이후 object에 할 수 있습니다 캐스트 string).
예를 들면 다음과 같습니다.

T newT1 = (T)(object)"some text";
string newT2 = (string)(object)t;


답변

두 줄 모두 같은 문제가 있습니다

T newT1 = "some text";
T newT2 = (string)t;

컴파일러는 T가 문자열임을 알지 못하므로이를 할당하는 방법을 알 수 없습니다. 그러나 당신이 확인했기 때문에 당신은 그것을 강제로 할 수 있습니다.

T newT1 = "some text" as T;
T newT2 = t; 

이미 문자열이기 때문에 t를 캐스팅 할 필요가 없으며 제약 조건을 추가해야합니다.

where T : class


답변

OP 가이 질문에 일반 파서에서 게시 한 유사한 코드를 알고 있습니다. 성능 측면 Unsafe.As<TFrom, TResult>(ref TFrom source)에서 System.Runtime.CompilerServices.Unsafe NuGet 패키지에있는를 사용해야 합니다. 이 시나리오에서는 값 유형에 대한 권투를 피합니다. 또한 Unsafe.AsJIT에서 생성 한 머신 코드가 두 번 캐스팅하는 것 ()을 사용하는 것보다 적은 기계 코드를 얻는다고 생각 (TResult) (object) actualString하지만 확인하지 않았습니다.

public TResult ParseSomething<TResult>(ParseContext context)
{
    if (typeof(TResult) == typeof(string))
    {
        var token = context.ParseNextToken();
        string parsedString = token.ParseToDotnetString();
        return Unsafe.As<string, TResult>(ref parsedString);
    }
    else if (typeof(TResult) == typeof(int))
    {
        var token = context.ParseNextToken();
        int parsedInt32 = token.ParseToDotnetInt32();
        // This will not box which might be critical to performance
        return Unsafe.As<int, TResult>(ref parsedInt32);
    }
    // other cases omitted for brevity's sake
}

Unsafe.As 공식 CoreFX 저장소에서 볼 수 있듯이 효율적인 기계 코드 명령어가있는 JIT로 대체됩니다.

안전하지 않은 소스 코드


답변

명시 적 유형을 확인하는 경우 왜 해당 변수를로 선언 T합니까?

T HowToCast<T>(T t)
{
    if (typeof(T) == typeof(string))
    {
        var newT1 = "some text";
        var newT2 = t;  //this builds but I'm not sure what it does under the hood.
        var newT3 = t.ToString();  //for sure the string you want.
    }

    return t;
}


답변

클래스와 메서드 모두에 대한 일반적인 선언이있는 경우에도이 오류가 발생합니다. 예를 들어 아래 코드는이 컴파일 오류를 나타냅니다.

public class Foo <T> {

    T var;

    public <T> void doSomething(Class <T> cls) throws InstantiationException, IllegalAccessException {
        this.var = cls.newInstance();
    }

}

이 코드는 컴파일합니다 (메소드 선언에서 T가 제거됨).

public class Foo <T> {

    T var;

    public void doSomething(Class <T> cls) throws InstantiationException, IllegalAccessException {
        this.var = cls.newInstance();
    }

}


답변

이 줄을 바꾸십시오 :

if (typeof(T) == typeof(string))

이 줄의 경우 :

if (t.GetType() == typeof(string))


답변