Я видел, как оба термина используются почти взаимозаменяемо в различных онлайн-объяснениях, и в большинстве учебников, к которым я обращался, также не совсем ясно различие.
Может быть, есть ясный и простой способ объяснить разницу, о которой вы, ребята, знаете?
Преобразование типа (также иногда называемое приведением типа )
Использовать значение одного типа в контексте, ожидающем другого.
Неконвертирующее приведение типа (иногда известное как каламбур типа )
Изменение, которое не изменяет базовые биты.
Принуждение
Процесс, с помощью которого компилятор автоматически преобразует значение одного типа в значение другого типа, когда этот второй тип требуется для окружающего контекста.
Ответы:
Преобразование типов :
Итак, приведение является неявным, приведение - явным, а преобразование - любым из них.
Несколько примеров (из того же источника ):
Принуждение (неявное):
double d; int i; if (d > i) d = i;
Cast (явный):
double da = 3.3; double db = 3.3; double dc = 3.4; int result = (int)da + (int)db + (int)dc; //result == 9
источник
Как вы заметили, способы использования различаются.
Мое личное использование:
«Приведение» - это использование оператора приведения . Оператор приведения указывает компилятору, что либо (1) это выражение неизвестно, относится к данному типу, но я обещаю вам, что значение будет иметь этот тип во время выполнения; компилятор должен рассматривать выражение как имеющее данный тип, и среда выполнения выдаст ошибку, если это не так, или (2) выражение имеет совершенно другой тип, но существует хорошо известный способ связать экземпляры типа выражения с экземплярами преобразуемого типа. Компилятору предлагается сгенерировать код, выполняющий преобразование. Внимательный читатель заметит, что это противоположности, и я думаю, что это изящный трюк.
«Преобразование» - это операция, при которой значение одного типа обрабатывается как значение другого типа - обычно другого типа, хотя «преобразование идентичности» по-прежнему является преобразованием с технической точки зрения. Преобразование может быть «изменением представления», например int в double, или может быть «сохранением представления», например, строкой в объект. Преобразования могут быть «неявными», которые не требуют приведения, или «явными», которые требуют приведения.
«Принуждение» - это неявное преобразование, изменяющее представление.
источник
Data.Coerce.coerce :: Coercible a b => a -> b
работает для типов, для которых доказано одно и то же представление;Unsafe.Coerce.unsafeCoerce :: a -> b
работает для любых двух типов (и заставит демонов вылезти из вашего носа, если вы его неправильно используете).Приведение - это процесс, с помощью которого вы рассматриваете тип объекта как другой тип, принуждение - это преобразование одного объекта в другой.
Обратите внимание, что в первом процессе нет преобразования, у вас есть тип, который вы хотели бы рассматривать как другой, скажем, например, у вас есть 3 разных объекта, которые наследуются от базового типа, и у вас есть метод, который будет принимать это базовый тип, в любой момент, если вы знаете конкретный дочерний тип, вы можете CAST его к тому, что он есть, и использовать все определенные методы и свойства этого объекта, и это не создаст новый экземпляр объекта.
С другой стороны, принуждение подразумевает создание нового объекта в памяти нового типа, а затем исходный тип будет скопирован в новый, оставив оба объекта в памяти (до тех пор, пока сборщики мусора не уберут один или оба) .
В качестве примера рассмотрим следующий код:
class baseClass {} class childClass : baseClass {} class otherClass {} public void doSomethingWithBase(baseClass item) {} public void mainMethod() { var obj1 = new baseClass(); var obj2 = new childClass(); var obj3 = new otherClass(); doSomethingWithBase(obj1); //not a problem, obj1 is already of type baseClass doSomethingWithBase(obj2); //not a problem, obj2 is implicitly casted to baseClass doSomethingWithBase(obj3); //won't compile without additional code }
baseClass
baseClass
otherClass
вbaseClass
, который будет включать создание нового объекта типа baseClass и заполнение его путем копирования данных из obj3.Хорошим примером является класс Convert C #, где он предоставляет собственный код для преобразования между различными типами.
источник
Приведение сохраняет тип объектов. Принуждения нет.
Принуждение принимает значение типа, который НЕ совместим с присваиванием, и преобразуется в тип, совместимый с присваиванием. Здесь я выполняю принуждение, потому
Int32
что НЕ наследуется отInt64
... поэтому он НЕ совместим с присваиванием. Это расширяющееся принуждение (без потери данных). Расширяющееся принуждение - это неявное преобразование . Принуждение выполняет преобразование.void Main() { System.Int32 a = 100; System.Int64 b = a; b.GetType();//The type is System.Int64. }
Приведение позволяет рассматривать тип, как если бы он был другого типа, при этом сохраняя тип .
void Main() { Derived d = new Derived(); Base bb = d; //b.N();//INVALID. Calls to the type Derived are not possible because bb is of type Base bb.GetType();//The type is Derived. bb is still of type Derived despite not being able to call members of Test } class Base { public void M() {} } class Derived: Base { public void N() {} }
Источник: аннотированный стандарт общеязыковой инфраструктуры Джеймса С. Миллера.
Что странно, так это то, что документация Microsoft по Casting не согласуется с определением спецификации ecma-335 для Casting.
... Это похоже на принуждение не на кастинг.
Например,
object o = 1; int i = (int)o;//Explicit conversions require a cast operator i.GetType();//The type has been explicitly converted to System.Int32. Object type is not preserved. This meets the definition of Coercion not casting.
Кто знает? Может быть, Microsoft проверяет, читает ли кто-нибудь этот материал.
источник
Ниже приводится пост из следующей статьи :
Различием между принуждением и кастингом часто пренебрегают. Я понимаю почему; многие языки имеют одинаковый (или аналогичный) синтаксис и терминологию для обеих операций. Некоторые языки могут даже называть любое преобразование «преобразованием», но следующее объяснение относится к концепциям CTS.
Если вы пытаетесь присвоить значение некоторого типа местоположению другого типа, вы можете сгенерировать значение нового типа, которое имеет значение, аналогичное исходному. Это принуждение. Принуждение позволяет использовать новый тип, создавая новое значение, чем-то напоминающее исходное. Некоторые приведения могут отбрасывать данные (например, преобразование int 0x12345678 в короткий 0x5678), а другие - нет (например, преобразование int 0x00000008 в короткое 0x0008 или длинное 0x0000000000000008).
Напомним, что значения могут иметь несколько типов. Если ваша ситуация немного отличается, и вы хотите выбрать только другой тип значения, приведение типов - это инструмент для работы. Приведение просто указывает, что вы хотите работать с конкретным типом, который включает значение.
Разница на уровне кода варьируется от C # до IL. В C # и приведение, и приведение выглядят довольно похоже:
static void ChangeTypes(int number, System.IO.Stream stream) { long longNumber = number; short shortNumber = (short)number; IDisposable disposableStream = stream; System.IO.FileStream fileStream = (System.IO.FileStream)stream; }
На уровне IL они совсем другие:
ldarg.0 conv.i8 stloc.0 ldarg.0 conv.i2 stloc.1 ldarg.1 stloc.2 ldarg.1 castclass [mscorlib]System.IO.FileStream stloc.3
Что касается логического уровня, есть несколько важных отличий. Что наиболее важно помнить, так это то, что принуждение создает новое значение, а приведение - нет. Идентичность исходного значения и значение после преобразования одинаковы, а идентичность приведенного значения отличается от исходного значения; приведение создает новый отдельный экземпляр, а приведение - нет. Следствием этого является то, что результат преобразования и оригинал всегда будут эквивалентны (как по идентичности, так и по равенству), но приведенное значение может быть равно или не совпадать с оригиналом и никогда не разделяет исходную идентичность.
В приведенных выше примерах легко увидеть последствия принуждения, поскольку числовые типы всегда копируются по значению. Когда вы работаете со ссылочными типами, все становится немного сложнее.
class Name : Tuple<string, string> { public Name(string first, string last) : base(first, last) { } public static implicit operator string[](Name name) { return new string[] { name.Item1, name.Item2 }; } }
В приведенном ниже примере одно преобразование - это приведение, а другое - приведение.
Tuple<string, string> tuple = name; string[] strings = name;
После этих преобразований кортеж и имя равны, но строки не равны ни одному из них. Вы можете немного улучшить ситуацию (или немного запутать), реализовав Equals () и operator == () в классе Name для сравнения Name и строки []. Эти операторы «исправят» проблему сравнения, но у вас все равно останется два отдельных экземпляра; любая модификация строк не будет отражена в имени или кортеже, в то время как изменения одного из имени или кортежа будут отражаться в имени и кортеже, но не в строках.
Хотя приведенный выше пример предназначен для иллюстрации некоторых различий между приведением типов и приведением, он также служит отличным примером того, почему вы должны быть крайне осторожны при использовании операторов преобразования со ссылочными типами в C #.
источник
Из стандарта CLI :
источник
Согласно Википедии,
Разница между приведением типов и приведением типов заключается в следующем:
TYPE CASTING | TYPE COERCION | 1. Explicit i.e., done by user | 1. Implicit i.e., done by the compiler | 2. Types: | 2. Type: Static (done at compile time) | Widening (conversion to higher data | type) Dynamic (done at run time) | Narrowing (conversion to lower data | type) | 3. Casting never changes the | 3. Coercion can result in representation the actual type of object | as well as type change. nor representation. |
Примечание . Трансляция - это не преобразование. Это просто процесс, с помощью которого мы рассматриваем тип объекта как другой тип. Следовательно, фактический тип объекта, а также представление не изменяется во время литья.
Я согласен со словами @ PedroC88:
источник