- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我正在尝试学习单一职责原则 (SRP),但这非常困难,因为我很难弄清楚我应该在什么时候从一个类(class)中删除什么以及应该删除/组织它的位置。
我在谷歌上搜索了一些 Material 和代码示例,但我发现的大多数 Material 并没有让它更容易理解,而是让它变得难以理解。
For example if I have a list of Users and from that List I have a class Called Control that does lots of things like Send a greeting and goodbye message when a user comes in/out, verify weather the user should be able to enter or not and kick him, receive user commands and messages, etc.
从示例中您不需要太多了解我已经对一个类做了太多的工作,但我还不清楚之后如何拆分和重组它。
如果我理解 SRP,我会有一个加入 channel 的类,一个用于问候和再见的类,一个用于用户验证的类,一个用于读取命令的类,对吧?
但是我应该在哪里以及如何使用踢球呢?
我有验证类,所以我确信我会在那里进行各种用户验证,包括天气是否应该踢用户。
所以 kick 函数将在 channel join 类中,并在验证失败时调用?
例如:
public void UserJoin(User user)
{
if (verify.CanJoin(user))
{
messages.Greeting(user);
}
else
{
this.kick(user);
}
}
如果你们能帮助我提供易于理解的在线免费 C# Material ,或者向我展示如何拆分引用的示例,并在可能的情况下提供一些示例代码、建议等,我将不胜感激。
最佳答案
让我们从 Single Responsibility Principle 的作用开始(SRP) 实际上意味着:
这实际上意味着每个对象(类)都应该有单一的职责,如果一个类有多个职责,这些职责就会耦合在一起并且不能独立执行,即在特定的实现中,一个的变化会影响甚至破坏另一个.
绝对必须阅读的是源代码本身(来自 "Agile Software Development, Principles, Patterns, and Practices" 的 pdf 章节):The Single Responsibility Principle
话虽如此,您应该设计您的类,以便它们在理想情况下只做一件事并且做好一件事。
首先想想你有什么“实体”,在你的例子中我可以看到 User
和 Channel
以及它们之间进行通信的媒介(“消息”) . 这些实体之间有一定的关系:
这也自然导致执行以下功能列表:
SRP 是一个重要的概念,但很难独立存在——对您的设计同样重要的是 Dependency Inversion Principle (蘸)。要将其纳入设计,请记住您对 User
、Message
和 Channel
实体的特定实现应依赖于抽象 或接口(interface)而不是特定的具体实现。出于这个原因,我们从设计接口(interface)而不是具体类开始:
public interface ICredentials {}
public interface IMessage
{
//properties
string Text {get;set;}
DateTime TimeStamp { get; set; }
IChannel Channel { get; set; }
}
public interface IChannel
{
//properties
ReadOnlyCollection<IUser> Users {get;}
ReadOnlyCollection<IMessage> MessageHistory { get; }
//abilities
bool Add(IUser user);
void Remove(IUser user);
void BroadcastMessage(IMessage message);
void UnicastMessage(IMessage message);
}
public interface IUser
{
string Name {get;}
ICredentials Credentials { get; }
bool Add(IChannel channel);
void Remove(IChannel channel);
void ReceiveMessage(IMessage message);
void SendMessage(IMessage message);
}
此列表没有告诉我们执行这些功能的原因。我们最好将“为什么”(用户管理和控制)的责任放在一个单独的实体中——这样 User
和 Channel
实体不必更改“为什么”改变。我们可以在这里利用策略模式和 DI,并且可以让 IChannel
的任何具体实现都依赖于给我们“为什么”的 IUserControl
实体。
public interface IUserControl
{
bool ShouldUserBeKicked(IUser user, IChannel channel);
bool MayUserJoin(IUser user, IChannel channel);
}
public class Channel : IChannel
{
private IUserControl _userControl;
public Channel(IUserControl userControl)
{
_userControl = userControl;
}
public bool Add(IUser user)
{
if (!_userControl.MayUserJoin(user, this))
return false;
//..
}
//..
}
你看到在上面的设计中 SRP 甚至还不够完美,即 IChannel
仍然依赖于抽象 IUser
和 IMessage
.
最终,人们应该努力实现一种灵活、松散耦合的设计,但总是需要权衡取舍,灰色区域也取决于您期望应用程序更改的位置。
在我看来,采用极端的 SRP 会导致非常灵活但也零散和复杂的代码,这些代码可能不像更简单但耦合更紧密的代码那样容易理解。
事实上,如果两项职责总是预计会同时发生变化,您可以说不应该将它们分成不同的类别,因为引用 Martin 的话,这会导致“不必要的复杂性” .永不改变的职责也是如此——行为是不变的,不需要拆分。
这里的主要思想是,你应该在你看到责任/行为在未来可能独立改变的地方做出判断,哪些行为是相互依赖的,并且总是会同时改变(“绑定(bind)在hip”)以及哪些行为从一开始就永远不会改变。
关于c# - 用C#学习单一职责原则,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7542051/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!