草庐IT

c++ - 在库中拆分实用程序功能以最大化可重用性的最佳方法是什么?

coder 2024-02-25 原文

我编写的静态链接库经常出现问题(或者在某些情况下,代码是从开放源代码中积累的)。

该库名为,是MFC工具箱库的名称,它具有许多免费函数,类等,它们支持MFC编程,Win32 API编程以及古老的C库和较新的C++标准库。

简而言之,这是一个工作库,其中包含适用于我的日常工作的工具,我已经积累了十多年,对于我们的产品是必不可少的。因此,它具有所有这些各种技术的实用程序和增强功能的丰富组合,并且通常在内部将所有这些技术的使用混合在一起以提供进一步的支持。

例如,我有一个 String Utilities.h String Utilities.cpp ,它们提供了大量与字符串相关的自由功能,甚至一两个类。

通常,我发现我有一对函数,一个函数不需要MFC或其CString,而另一个兄弟函数确实需要这些东西。例如:

////////////////////////////////////////////////////////////////////////
// Line Terminator Manipulation
////////////////////////////////////////////////////////////////////////

// AnsiToUnix() Convert Mac or PC style string to Unix style string (i.e. no CR/LF or CR only, but rather LF only)
// NOTE: in-place conversion!
TCHAR * AnsiToUnix(TCHAR * pszAnsi, size_t size);

template <typename T, size_t size>
T * AnsiToUnix(T (&pszBuffer)[size]) { return AnsiToUnix(pszBuffer, size); }

inline TCHAR * AnsiToUnix(Toolbox::AutoCStringBuffer & buffer) { return AnsiToUnix(buffer, buffer.size()); }

// UnixToAnsi() Converts a Unix style string to a PC style string (i.e. CR or LF alone -> CR/LF pair)
CString UnixToAnsi(const TCHAR * source);

如您所见,AnsiToUnix不需要CString。因为Unix使用单个回车符作为行终止符,并且Windows ANSI字符串使用CR + LF作为行终止符,所以可以保证所得的字符串将适合原始缓冲区空间。但是对于反向转换,该字符串几乎可以保证增长,对于每次出现的CR都增加一个额外的LF,因此,希望使用CString(或者也许是std::string)来提供自动增长。字符串。

这仅是一个示例,其本身并不是一个很可怕的例子,可以考虑从CString转换为std::string来从库的该部分中删除对MFC的依赖。但是,还有其他一些例子,其中的依赖关系更加微妙,改变它的工作也更多。此外,代码按原样经过了很好的测试。如果我去尝试删除所有MFC依赖项,则可能会在代码中引入微妙的错误,这可能会损害我们的产品,并加重执行这项本质上非严格必要的任务所需的时间。

我想了解的重要一点是,这里我们有一组函数,它们彼此非常相关(ANSI-> UNIX,UNIX-> ANSI),但是其中一侧使用MFC,而另一侧仅使用字符数组。 。因此,如果我试图提供一个可重用的库头文件,则希望将所有依赖于MFC的函数分解为一个头文件,将不依赖于MFC的那些文件分解为一个头文件,以便于将所述文件分发到其他不使用MFC的项目(或任何有问题的技术:例如,希望具有不需要Win32头文件的所有功能-只是对C++的增强,具有自己的头文件,并且等等。)。

我对大家的问题是,您如何处理这些问题-技术依赖性与相关功能都在同一个地方?

您如何分解您的库-划分内容?什么跟什么?

也许增加我的动力很重要:我希望能够发表文章并与他人共享代码,但是总的来说,他们倾向于使用MFC Toolbox库的某些部分,而MFC Toolbox库本身也使用其他部分,从而形成了一个深层次的网络。依赖关系,我不想给这些文章和代码项目的读者/程序员/消费者带来如此沉重的负担!

我当然可以只除去给定文章或项目所需的部分,但这似乎是一项耗时且毫无意义的工作。在我看来,以一种我可以更轻松地共享内容而不用拖拽整个库的方式来清理库将是更加明智的。即重新组织一次,而不是每次都要挖掘事物...

这是另一个很好的例子:
UINT GetPlatformGDILimit()
{
    return CSystemInfo::IsWin9xCore() ? 0x7FFF : 0x7FFFFFFF;
}

GetPlatformGDILimit()是一个相当通用的,无功利的函数。除了作为客户端,它实际上与CSystemInfo没有任何关系。因此它不属于“SystemInfo.h”。而且这只是一个自由功能-肯定没有人会尝试将其放在自己的 header 中吗?我将其放在“Win32Misc.h”中,其中包含各种各样的东西-免费功能主要是增强Win32 API。但是,这个看似无害的功能依赖于CSystemInfo,后者本身使用CStrings和大量其他库函数来使其能够高效地完成工作,或者使用更少的代码行,或更可靠地完成其工作,或者以上全部。

但是,如果我有一个演示项目,该项目引用Win32Misc.h header 中的一个或两个函数,那么我将陷入困境,要么只提取项目所需的单个函数(以及这些函数所依赖的一切,以及所有这些实体依赖于此类对象,等等)-或我必须尝试包括Win32Misc.h及其.cpp-这会给它们带来更多不必要的开销(只是为了演示项目的编译)。

因此,人们使用哪种经验法则来指导自己确定划界的位置-随之而来的是什么?如何防止C++库从 hell 变成依赖树? ;)

最佳答案

就个人而言,我将在功能上进行 segmentation 。在一个库中进行字符串操作。另一个中的整数类型(也许char放在字符串lib中除外)

我当然会让与平台无关的东西远离与平台无关的东西。特定于供应商的东西,而不是非特定于供应商的东西。这可能需要两个甚至三个字符串库。

也许您可以使用范例“它需要MFC吗?”任何需要mfc的东西都应该分开。然后继续进行到“是否需要窗口”再次进行拆分。等等...

毫无疑问,某些项目将要求所有库都必须在VC++中编译并且只能在Windows上运行,这就是它的做法。其他项目将仅使用库的一部分即可在Linux上愉快地编译,并且可以使用gcc进行编译。

直流电

关于c++ - 在库中拆分实用程序功能以最大化可重用性的最佳方法是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2231462/

有关c++ - 在库中拆分实用程序功能以最大化可重用性的最佳方法是什么?的更多相关文章

  1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  2. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  3. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  4. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  5. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  6. ruby - 为什么 4.1%2 使用 Ruby 返回 0.0999999999999996?但是 4.2%2==0.2 - 2

    为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返

  7. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  8. ruby - ruby 中的 TOPLEVEL_BINDING 是什么? - 2

    它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput

  9. ruby - Infinity 和 NaN 的类型是什么? - 2

    我可以得到Infinity和NaNn=9.0/0#=>Infinityn.class#=>Floatm=0/0.0#=>NaNm.class#=>Float但是当我想直接访问Infinity或NaN时:Infinity#=>uninitializedconstantInfinity(NameError)NaN#=>uninitializedconstantNaN(NameError)什么是Infinity和NaN?它们是对象、关键字还是其他东西? 最佳答案 您看到打印为Infinity和NaN的只是Float类的两个特殊实例的字符串

  10. ruby-on-rails - 如果 Object::try 被发送到一个 nil 对象,为什么它会起作用? - 2

    如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象

随机推荐