Анимация
JavaScript


Главная  Библионтека 

0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 [ 28 ] 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85

Задача 15. Потребление и злоупотребление

правами доступа Сложность: 6

Кто в действительности имеет право доступа к внутренней организации вашего класса? Эта задача - о фальсификаторах, мошенниках, карманниках и ворах, а также о том, как их распознать и спастись от них.

Вопрос для новичка

1. Какой код может обращаться к следующим частям класса:

а) public

б) protected

в) private

Вопрос для профессионала

2. Рассмотрим следующий заголовочный файл.

Файл x.h

class X { public:

х() : private (l) { /*...*/ } tempiate<class т> void fС const T& t ) { /*...*/ } int ValueO { return private ; }

...

private:

int private ;

Покажите, как произвольный вызывающий код может получить непосредственный доступ к закрытому члену класса private..:

а) при помощи нестандартного и непереносимого "хака";

б) при помощи переносимой и соответствующей стандарту методики.

3. Подумайте, не является ли это "дырой" в механизме управления правами доступа в С++ (а следовательно, дырой в инкапсуляции С++)?

Решение

Эта задача - о фальсификаторах, обманщиках, мошенниках и ворах...

1. Какой код имеет право доступа к следующим частям класса:

а) public

Воз.можность обращения к открытым членам имеет любой код.

б) protected

Возможность обращения к защищенным членам имеют только функции-члены того же класса и его друзья, а также функции-члены и друзья производных классов.

в) private

Возможность обращения к закрытым членам имеют только функции-члены того же класса и его друзья.



Это обычный ответ, и обычно это правильно. И о в этой задаче мы рассмотрим специальный случай, когда этот ответ не полон, поскольку иногда С++ предоставляет возможность законного (пусть и аморального) нарушения прав доступа к закрытым членам.

2. Рассмотрим следующий заголовочный файл ... Покажите, как произвольный вызьшающий код может получить непосредственный доступ к закрытому члену класса pri vate :

а) при помощи нестандартного и непереносимого "хака";

б) при помощи переносимой и соответствующей стандарту методики.

Есть странная притягательность в самых трагических сценах, когда мы смотрим не отрываясь на автомобильные аварии, падающие небоскребы и... взлом кода.

При попытке ответить на вопрос о том, как обратиться к закрытому члену нестандартными и непереносимыми методами, в голову приходит целый ряд идей. Вот три из них.

Преступник №1: фальсификатор

Метод фальсификатора состоит в том, чтобы продублировать определение фальсифицируемого класса, в которое внесены все необходимые нам изменения, например:

Пример 15-1: ложь и подделка

class X {

вместо включения файла x.h, вручную (и незаконно) дублируем определение X и добавляем строку наподобие этой:

friend :: Hi jack С х& ) ;

void HijackC х& х ) {

X.private. =2; Злобный смех

Этот человек - фальсификатор. Запомните его хорошенько, ему нельзя доверять...

Конечно, то, что сделано, - не законно. Это не законно хотя бы потому, то нарушает правило одного определения, которое гласит, что если тип (в нашем случае -- х) определен более одного раза, то все его определения должны быть идентичны. Используемый же в приведенном примере объект хотя и называется X и выглядит похожим на X, но это НС тот же х. который используется остальным кодом программы. Тем не менее, такой "хак" будет работать на большинстве компиляторов, поскольку расположение данных объекта останется неизменным.

Преступник №2: карманник

Карманник сработает тихо - подменив смысл определения класса, например, следующим образом.

пример 15-2: использование макромагии

#define private public не законно!

#include "x.h"

void HijackC x& x ) {

x.private - 2; Злобный смех

Этот человек - карманник. Запомните сго умелые пальцы!

Конечно, то, что делает карманник, - не законно. Код из примера 15-2 непереносим по двум причинам.

• Не законно переопределять при помощи директивы #define зарезервированные слова.



• При этом происходит такое же нарушение правила одного определения, как и у фальсификатора. Однако если расположение данных объекта остается неизменным, этот способ сработает.

Преступник №3: мошенник

Принцип работы мошенника - в подмене понятий.

Пример 15-3: попытка имитации размещения данных объекта

class BaitAndswitch { в надежде, что размещение данных в public: этом классе будет тем же, что и в х

int notSoPrivate;

void fC х& X ) {

Creinterpret cast <BaitAndSwitch&>(x)).notSoPrivate = 2;

} Злобный смех

Этот человек - мошенник. Хорошенько его запомните! Его реклама рассчитана только на то, чтоб затащить вас в магазин, а уж там он обязательно ухитриться продать вам совершенно не ту вещь, о которой шла речь в рекламе, да еще и в несколько раз дороже, чем в любом другом месте.

Конечно, то, что делает мошенник, - не законно. Код из примера 15-3 не законен по двум причинам.

• Нет гарантии, что размещение объектов х и BaitAndswitch в памяти будет одинаковым - хотя на практике это обычно так и есть.

• Результат reinterpret cast не определен, хотя большинство компиляторов сделают именно то, чего от них хочет мошенник. В конце концов, говоря волшебное слово reinterpret cast, вы заставляете компилятор поверить вам и закрыть глаза на подготавливаемое вами мошенничество.

Но вопрос не исчерпывался только обманными способами. Если помните, нас спрашивали и о том, как добиться интересующего результата совершенно корректно и переносимо с точки зрения стандарта. Увы, крупные преступники имеют весьма респектабельный вид и большие счета в банках, так что их очень трудно схватить за руку.

Персона грата №4: адвокат

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

Пример 15-4: проныра-законник

namespace {

struct Y {};

templateo

void x::f( const Y& ) {

private = 2; Злобный смех

void TestO { X x;

cout « X.ValueO « endl; выводит 1 x.fC YO );

cout « x.valueO « endl ; выводит 2

Этот человек - адвокат, который знает все лазейки. Его невозможно поймать, поскольку он слишком осторожен, чтобы нарушить букву закона, при этом нарушая его дух. Запомните и избегайте таких неджентльменов.



0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 [ 28 ] 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85