什么是封装?

Him*_*dav 11 java oop encapsulation

我有两个封装定义,不能适合一个定义.

  1. 封装是数据隐藏.通过使用private,protectedpublic,将数据打包到单个组件中.
  2. 无论改变什么都包含它.保护任何易于改变的东西.

这两个定义如何讨论相同的概念?

Fab*_*ica 14

封装可能是最容易被误解的OOP概念.

封装不是数据隐藏!

"封装"来自"胶囊".这意味着将事物放在一起,将它们关在一个包中,我们在这里讨论的"事物"是数据和功能.没有封装的编程意味着处理数据的函数在代码中的某处"浮动",虽然它们处理数据甚至将特定类型作为输入,但它们与数据分离.

让我举一个例子而不是关注"公共"等等:如果你有一个处理复数的类,它有一个真实和想象的部分,你可以简单地定义它:

class complex {
    double real;
    double imaginary;
};
Run Code Online (Sandbox Code Playgroud)

使用在C中使用的旧的预封装样式来获取此数字的绝对值,您将定义如下函数:

double absolute(double real, double imaginary);
Run Code Online (Sandbox Code Playgroud)

这根本不会与班级联系起来!当然,您也可以定义一个将类复杂作为输入的函数,但它仍然是一个外部函数.因此,要使用它,您必须这样做:

complex A;
A.real = 1;
A.imaginary = -3;
Run Code Online (Sandbox Code Playgroud)

并获得你必须调用的绝对值

absolute(A.real, A.imaginary);
Run Code Online (Sandbox Code Playgroud)

相反,您可以使用封装并将数据和函数放在一起:

class complex {
    double real;
    double imaginary;
    double absolute();  // inside the class, encapsulated into it!
};
Run Code Online (Sandbox Code Playgroud)

然后获得绝对值,你只需要调用方法就好

A.absolute();
Run Code Online (Sandbox Code Playgroud)

这根本不需要数据隐藏.优点是代码更易于管理,因为您可以清楚地看到所有相关的"事物"(即数据和功能)组合在一起,因此您可以一眼就知道您拥有的内容(数据)以及您可以使用的内容它(方法).

如果没有这个,信息隐藏是不可能的,因为这意味着你从外部限制对某些成员(私有成员)的访问,因此你必须在里面有一些方法,否则你将无法对你的数据做任何事情!

同时,信息隐藏有助于充分利用封装:如果人们可以从外部访问数据,那么让其他编码人员编写自己的(未封装的)代码来处理您的数据会有很大的危险,如果实现不完全兼容,至少会导致代码重复(即无用的努力)和不一致.相反,数据隐藏意味着每个人都必须使用所提供的公共方法访问私有数据,以便它们对每个人都是相同的.

因此,数据隐藏需要封装才有意义,同时数据隐藏有助于它.他们一起工作得很好,但他们不是一回事!

回到你的问题:鉴于此,定义1是错误的.而且,正如CommuSoft所指出的那样,并不是一个真正的定义,这是一个经验法则.我将补充一点,关于何时使用数据隐藏而不是封装是一个经验法则.

另一方面,电子仪表表明这可能是这个问题的重复.我认为值得注意的是,大多数答案都是错误的,包括最佳答案,它提供了封装的一个例子,实际上与封装相反.

如果你想要外部引用,这里有两篇关于这个的文章:

封装不是信息隐藏

抽象,封装和信息隐藏(请注意,当他开始一个名为"ENCAPSULATION"的段落并引用了很多定义时,他只是试图展示围绕这个主题的混淆;这些定义是错误的,正如他后面解释的那样!)

  • 写得很好,这是OOP中最容易混淆的部分. (3认同)

mk.*_*mk. 6

隐藏数据与控制有关.当您隐藏数据时,将其隐藏在其他程序员(您将来自己包含在内)中,因为他们可能会对数据执行不正确的操作,因为他们并不像您那样了解它.因此,您需要仔细控制他们如何使用数据 - 在这种情况下,通过隐藏它来限制访问和可见性.

封装并不是"保护易于改变的一切"的做法.声称这一点的文章过于笼统,因为大多数软件开发技术都是关于"保护"我们以后可能会改变的事情.你应该知道的是,检查容易发生变化的事情通常是一个好主意,并决定你希望对其他人如何看待和使用这些东西的约束 - 你如何封装这些东西.


......同样,封装也不是"将功能与数据放在一起"的做法,正如在另一个答案声明中链接的一些文章.适当的术语可能是模块化,或简单的旧的面向对象编程 - 有许多相关的概念.这些技术可以带来封装,但称其封装是令人困惑的,特别是对于初学者.封装真的是"隐藏"数据.(举一个稍微高级的例子,在JavaScript中,你可以将一个值与一个函数一起作为函数对象的一个​​属性,或者通过一个闭包.这些都不使用字段,它们不涉及使用类的方法/ object/prototype,都将数据和函数"放在一起",但只有闭包才会通过强制访问约束来封装数据.)