- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
鉴于以下预先存在的框架,我需要找到好的设计模式来创建派生类的不同实例。
我面临的主要挑战如下:
challenge-1> 每个类都有10多个字段,如何有效地将这些字段传递给派生类,再传递给基类。
针对这个问题,我可以想出四个解决方案,但没有一个对我有吸引力。
方法一>以简单格式传递所有参数
classA::classA(int field1, float field2, ..., double field29)
=> 缺点:创建传入参数超过 6~7 个的函数不是一个好主意
方法2>将所有参数作为结构传递
struct DataClassA
{
int field1;
float field2;
...
double field29;
};
struct DataClassBA : DataClassA
{
int m_iField30;
// ...
double m_iField40;
};
所以首先我将 DataClassBA
传递给 classBA
然后 classBA
将 DataClassA
传递给 classA
。=> 缺点:类型 DataClassBA
和 classBA
是相似的类型,除了一个包含操作而另一个不包含操作。此外,当将结构传递给构造函数时,存在复制和重复的惩罚。想象一下,对于每一个不同的类,我们必须定义一个相似的结构来保存所有的初始化数据。类与其对应结构的主要区别在于类包含一些方法,而结构纯粹用于传输数据。
方法三>通过Set函数设置所有字段
classA
{
public:
int Field1() const { return m_iField1; }
classA& Field1(int field1)
{
m_iField1 = field1;
return *this;
}
...
}
classBA : public classA
{
public:
int Field30() const { return m_iField30; }
classBA& Field30(int field30)
{
m_iField30 = field30;
return *this;
}
...
}
=> 缺点:每次创建一个实例都会导致许多函数调用并且非常昂贵。
方法 4> 将映射传递给基类和派生类的所有构造函数。
=> 缺点:我真的认为这是个坏主意,尽管它使数据传递变得容易。
challenge-2> 基类的默认值由其不同的派生类决定。例如,classA::m_iField2
的默认值根据派生类的不同而不同。
针对这个问题,我可以想出两个解决方案,但没有一个对我有吸引力。
方法一> 将默认逻辑添加到派生类本身。
方法二> 将默认逻辑添加到工厂类本身。
我列出了所有我能想到的方法。但是,我仍然在寻找一个干净专业的解决方案来解决这个问题。如果有一个编写良好的 API 库,我可以将其用作解决类似问题的引用,那将是最好的。欢迎任何评论。
谢谢
/////////////////////// framework ////////////////////////////////////////
// Note:
// <1> the class hierarchy has to kept as this
// <2> getter and setter functions in each class have to kept as this
// <3> add new functions(i.e constructors) are allowed
// <4> add new classes or structures are allowed
/////////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include <map>
#include <string>
#include <iostream>
using namespace std;
/************************************************************************/
/* Class Name: classA (an abstract base class)
* default value of m_iField2 is determined by its derived class
/************************************************************************/
class classA
{
public:
virtual ~classA() = 0 {}
// ...
private: //
int m_iField1;
float m_iField2; // one of the potential field that has to get the default value
// ...
double m_iField29;
};
/************************************************************************/
/* Class Name: classBA
* If the pass-in parameters do NOT include value for the field classA::m_iField2
* then assign its value as 200.0f
/************************************************************************/
class classBA : public classA
{
// ...
private:
int m_iField30;
// ...
double m_iField40;
};
/************************************************************************/
/* Class Name: classCA
* If the pass-in parameters do NOT include value for the field classA::m_iField2
* then assign its value as 300.0f
/************************************************************************/
class classCA : public classA
{
// ...
private:
int m_iField50;
// ...
int m_iField60;
};
int main(int argc, char* argv[])
{
map<string, string> mapStrsBA;
mapStrsBA["name"] = "classBA";
mapStrsBA["field1"] = "5";
// ...
mapStrsBA["field40"] = "1.89";
// pass mapStrsBA to a factory class with function to create a new instance of class classBA
map<string, string> mapStrsCA;
mapStrsBA["name"] = "classCA";
mapStrsBA["field1"] = "6";
// ...
mapStrsBA["field60"] = "19";
// pass mapStrsCA to a factory class with function to create a new instance of class classCA
return 0;
}
最佳答案
让我在你的挑战 1 中添加方法 5,它并不总是适用,但经常适用,尤其是如果你碰巧有很多字段:找到严格属于一起的成员字段,并收集它们成逻辑对象。例如,假设您有一个如下所示的类 Shape
:
class Shape
{
public:
Shape(pass initial values for all member variables);
// ...
private:
// bounding box coordinates
int xmin, xmax;
int ymin, ymax;
// color
int red, green, blue;
int alpha;
// center point (for rotations)
int cx, cy;
};
这是 10 个变量。然而,这些并不是真正无关的。其中大部分是 x/y 对,然后有一组指定颜色。因此你可以重写如下:
struct Point
{
int x, y;
Point(int ax, int ay): x(ax), y(ax) {}
};
struct Color
{
int red, green, blue, alpha;
Color(int r, int g, int b, int a): red(r), green(g), blue(b), alpha(a) {}
};
class Shape
{
public:
// ...
private:
// bounding box
Point lower_left, upper_right;
Color color;
Point center;
};
现在您突然只有四个变量可以传递了。您甚至可能会考虑制作一个由两个角点组成的矩形类型,并将其用于边界框,进一步将变量数量减少到 3。
请注意,不仅减少了要传递的参数数量,还增加了清晰度。
既然你没有提供任何关于你的类的细节,我不能说这样的逻辑分组是否也适用于你的类,但考虑到你提到的大量参数,如果不能的话我会感到惊讶。
对于挑战 2,我认为将默认逻辑添加到派生类会更好。
关于c++ - 创建派生类的模式,派生类本身和基类都包含许多字段,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8585161/
对此感到疯狂,真的缺少一些东西。 我有webpack 4.6.0,webpack-cli ^ 2.1.2,所以是最新的。 在文档(https://webpack.js.org/concepts/mod
object Host "os.google.com" { import "windows" address = "linux.google.com" groups = ["linux"] } obj
每当我安装我的应用程序时,我都可以将数据库从 Assets 文件夹复制到 /data/data/packagename/databases/ .到此为止,应用程序工作得很好。 但 10 或 15 秒后
我在 cc 模式缓冲区中使用 hideshow.el 来折叠我不查看的文件部分。 如果能够在 XML 文档中做到这一点就好了。我使用 emacs 22.2.1 和内置的 sgml-mode 进行 xm
已结束。此问题不符合 Stack Overflow guidelines .它目前不接受答案。 我们不允许提出有关书籍、工具、软件库等方面的建议的问题。您可以编辑问题,以便用事实和引用来回答它。 关闭
根据java: public Scanner useDelimiter(String pattern) Sets this scanner's delimiting pattern to a patt
我读过一些关于 PRG 模式以及它如何防止用户重新提交表单的文章。比如this post有一张不错的图: 我能理解为什么在收到 2xx 后用户刷新页面时不会发生表单提交。但我仍然想知道: (1) 如果
看看下面的图片,您可能会清楚地看到这一点。 那么如何在带有其他一些 View 的简单屏幕中实现没有任何弹出/对话框/模式的微调器日期选择器? 我在整个网络上进行了谷歌搜索,但没有找到与之相关的任何合适
我不知道该怎么做,我一直遇到问题。 以下是代码: rows = int(input()) for i in range(1,rows): for j in range(1,i+1):
我想为重写创建一个正则表达式。 将所有请求重写为 index.php(不需要匹配),它不是以/api 开头,或者不是以('.html',或'.js'或'.css'或'.png'结束) 我的例子还是这样
MVC模式代表 Model-View-Controller(模型-视图-控制器) 模式 MVC模式用于应用程序的分层开发 Model(模型) - 模型代表一个存取数据的对象或 JAVA PO
我想为组织模式创建一个 RDF 模式世界。您可能知道,组织模式文档基于层次结构大纲,其中标题是主要的分组实体。 * March auxiliary :PROPERTIES: :HLEVEL: 1 :E
我正在编写一个可以从文件中读取 JSON 数据的软件。该文件包含“person”——一个值为对象数组的对象。我打算使用 JSON 模式验证库来验证内容,而不是自己编写代码。符合代表以下数据的 JSON
假设我有 4 张 table 人 公司 团体 和 账单 现在bills/persons和bills/companys和bills/groups之间是多对多的关系。 我看到了 4 种可能的 sql 模式
假设您有这样的文档: doc1: id:1 text: ... references: Journal1, 2013, pag 123 references: Journal2, 2014,
我有这个架构。它检查评论,目前工作正常。 var schema = { id: '', type: 'object', additionalProperties: false, pro
这可能很简单,但有人可以解释为什么以下模式匹配不明智吗?它说其他规则,例如1, 0, _ 永远不会匹配。 let matchTest(n : int) = let ran = new Rand
我有以下选择序列作为 XML 模式的一部分。理想情况下,我想要一个序列: 来自 my:namespace 的元素必须严格解析。 来自任何其他命名空间的元素,不包括 ##targetNamespace和
我希望编写一个 json 模式来涵盖这个(简化的)示例 { "errorMessage": "", "nbRunningQueries": 0, "isError": Fals
首先,我是 f# 的新手,所以也许答案很明显,但我没有看到。所以我有一些带有 id 和值的元组。我知道我正在寻找的 id,我想从我传入的三个元组中选择正确的元组。我打算用两个 match 语句来做到这
我是一名优秀的程序员,十分优秀!