К основному контенту

Почему нельзя наследовать структуры в C# ?

Введение - зачем используются структуры?

Все мы сильно любим структуры (идея, чуждая Java, не считая примитивных типов). Структуры, когда они не упакованы, зачастую, предоставляют прекрасную возможность обрабатывать динамические данные относительно небольшого размера.

Размещение и освобождение типов данных в целом проще, чем размещение и освобождение ссылочных типов. Это связано тем, что структуры размещаются в стеке, либо встроены в содержащиеся типы и освобождаются, при очищении стека либо когда встроенные типы перераспределяются, в то время, как ссылочные типы размещаются в куче и очищаются сборщиком мусора (garbage-collected)

Хм... Структуры не хотят наследоваться.

Объектно-Ориентированное Программирование дает нам возможность делать много полезных вещей. И основной коцепт - это, конечно же, наследование. Многие из нас, когда либо, пытались наследовать структуру только для того, что бы понять, что C# не позволит нам этого сделать.

Для примера рассмотрим код:

// Что может быть не так с этим простым кодом?
// DateTime это структура, если вы еще об этом не знаете, то Now() хорошее время для того, что // бы узнать это :)
struct MyDateTime : DateTime
{
    double SecondsFromMyBirthday { get; private set; }
}


// или с этим, довольно разумным запросом на наследование:
struct Point
{
   int x, y;
}

struct Point3D : Point
{
   int z;
}


Честно говоря, каждый программист желает  расширить структура DateTime, не так ли?

Это код выведет ошибку компиляции: Type 'DateTime' in interface list is not an interface.

Если структура будет перемещена в класс, то выйдет следующая ошибка компиляции: 'MyDateTime': cannot derive from sealed type 'System.DateTime'.

Теперь мы понимаем, что System.DateTime упакован. Но почему?

Замечание: фактически, все типы структур унаследованы от класса System.ValueType, который, в свою очередь унаследован от класса object.

Ответ

В памяти структуре выделен стек фиксированного размера.

Давайте рассмотрим код примера Point/Point3D структуры, который указан выше. Рассмотрим следующую строку кода: 

// Предположительно, у нас есть p2d - Point, 
// и p3d - Point3D
p2d = p3d; 

Что должно произойти? Стек p2d расширит свою память для поддержки присвоения?

Кроме того, структуры, являясь типами значений, не используют ссылки (кроме упакованных). Это сделало бы полиморфизм бессмысленным, поскольку отсутствуют пути через ссылочные указатели.

Что же касается массивов, массивы типов данных хранятся  "inline" (для повышения производительности и сборки мусора). Если мы будем использовать структуру без понимания этого, то "обрезанная структура" приведет к повреждению памяти.

В заключении - мы увидели неразумность наследования структур в .NET. Хорошая новость - наследовать можно классы.

Источник: http://www.codeproject.com/Tips/877525/Inherit-a-struct-in-Csharp-why-not

Комментарии

Популярные сообщения из этого блога

Использование curl в C#

Для того что бы скачать страницу какого-либо сайта в основном используют код следующего вида: string page ; using ( var client = new WebClient ( ) ) { page = client . DownloadString ( "http://www.example.com/" ) ; } Конечно, это простейший код, но в большинстве случаев он работает. Но бывают случаи, когда при "парсинге" возникает ошибка:   500 Internal Server Error .  В основном, она возникает в случае, если сервер не может получить от нас h t tp-заголовок при запросе данных. (Конечно, это может быть и ошибка на сервере, но мы будет обсуждать случай, когда сайт защищен подобным способом) В данном случае нам необходимо отправлять http-заголовок на сервер, таким образом мы маскируемся под браузер. Для этого и служит библиотека curl. Программа cURL может автоматизировать передачу файлов или последовательность таких операций. Например, это хорошее средство для моделирования действий пользователя в веб-обозревателе. ( wikipedia ) Для исп

Проблемы с SSL при использовании VK API

При реализации одного из проектов, связанных с API VK я столкнулся с одной крайне интересной проблемой. У меня просто перестал работать скрипт, при этом выдавая следующую ошибку: Traceback ( most recent call last ) : File "C:\Users\Oleg\AppData\Local\Programs\Python\Python36-32\lib\site-packages\urllib3\contrib\pyopenssl.py" , line 441 , in wrap_socket cnx.do_handshake ( ) File "C:\Users\Oleg\AppData\Local\Programs\Python\Python36-32\lib\site-packages\OpenSSL\SSL.py" , line 1806 , in do_handshake self._raise_ssl_error ( self._ssl , result ) File "C:\Users\Oleg\AppData\Local\Programs\Python\Python36-32\lib\site-packages\OpenSSL\SSL.py" , line 1546 , in _raise_ssl_error _raise_current_error ( ) File "C:\Users\Oleg\AppData\Local\Programs\Python\Python36-32\lib\site-packages\OpenSSL\_util.py" , line 54 , in exception_from_error_queue raise exception_type ( errors ) OpenSSL.SSL.Error: [ ( 'SSL routines&