- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我知道我可以创建方法指针,而且我知道它们通常不同于函数指针。不允许在它们之间进行转换。方法指针可以包含大量有关如何调整 this
指针等的数据。
我想知道如何获取正在执行给定方法的代码的实际地址。该地址不会以任何方式被引用,但将用作给定方法所属类型的标识符 - 有点像 RTTI,具有对库边界不敏感的优点,因为此代码通常仅在一个单元中可用.
编辑:
为什么我不添加一些返回对象类型的 getType()
方法?
因为我不仅希望能够以这种方式使用我创建的类。我想做的基本上是我对 variant
类的实现,它基本上可以接受所有内容,前提是存在针对给定类型的 void* getVariantId() 的专门化。
然后我可以写:
template <typename T>
class getTypeId{
};
class foo{
public:
void bar();
};
// getTypeId declared but not defined for generic type T earlier...
template <>
class getTypeId<foo>{
static void* get<foo>(){
return &foo::bar;
}
};
void doSomething(myVariant var);
int main(int argc, char* argv[]){
foo f;
myVariant var = myVariant::fromType<foo*>(&f);
doSomething(f);
}
void doSomething(myVariant var){
foo* f = var.toType<foo*>(); // Returns foo object from main
int* i = var.toType<int*>(); // returns null, as var is not int* in my example.
}
fromType
的想法是使用getTypeId
获取类型的表示值,并将其与转换为void*
的对象指针一起strongig。另一方面,toType
比较从 `getTypeId::get 获取的值和存储在对象中的值 - 如果匹配,则将内部保存的对象指针重新解释回原始类型并返回。
这个解决方案的美妙之处在于,即使有一些共享库 X 定义了类型 x,然后是库 Y 和 Z,它们分别使用库 X 也会同意类型 x 是相同的(例如,如果变体创建在 Y 中传递给 Z),因为 X::method 上的地址保持不变。作为 Y 或 Z 库(但不是 X!!!)的创建者,我不需要确保 X 库启用了 RTTI,甚至不需要知道 myVariant 的存在,但类型 x 仍然完全兼容 myVariant。
编辑2:
我现在看到没有独立于实现的方法来完成这项工作。但是我认为这是不常需要的怪癖,因此不存在的功能,而不是无法创建的功能。我想我还没有说清楚我想要实现什么以及我是如何计划这样做的。所以:
这里所说的库是指共享库(.dll、.so 等)。
每个非纯虚拟的方法(意味着每个已定义的方法)都必须有实现。此实现表现为接受一个附加参数 this
的函数 - 虚函数仅在调用方有所不同。让我们以类 X 及其方法 X::foo
为例。这个方法只绑定(bind)到X类型,即使Y类型继承它,这个方法仍然是X::foo
,因此它足以识别类型X
。覆盖子类 Xchild
中的虚方法或覆盖方法实际上是用新地址定义新方法 Xchild::foo。那么您可能会使用 Xchild::foo
来表示 Xchild
,而不是 X
。
如果类型 X(准确地说是它的所有方法)在库 libX 中定义,并且 libY 和 libZ 都在使用 libX(但彼此不认识),并使用第一种方法为它们自己的目的定义 getTypeId在 X 中表示它(请注意,这不会对 libX 开发人员施加任何影响),然后他们就 X 是什么类型达成一致。如果应用程序开发人员使用 libY 获取变体并将其传递给 libZ,两者都会正确识别提到的类型。
我不是在尝试开发类型转换变体 - 保存为 X 的变体只能作为 X 读取,即使传递给 myVariant 的实际指针是 Xchild。我对获取给定实例的最顶层类的类型不感兴趣 - 只是用于创建变体的类。
最佳答案
C++ FAQ Lite 将整个部分用于 Pointers to Member Functions .
有两个答案特别说明了为什么您尝试做的事情无法完成(至少不能以可移植的方式):
C++ introduces a new type of pointer, called a pointer-to-member, which can be invoked only by providing an object. Do not attempt to "cast" a pointer-to-member-function into a pointer-to-function; the result is undefined and probably disastrous. E.g., a pointer-to-member-function is not required to contain the machine address of the appropriate function.
A pointer to a member function might be a data structure rather than a single pointer. Think about it: if it's pointing at a virtual function, it might not actually be pointing at a statically resolvable pile of code, so it might not even be a normal address.
关于c++ - 对象方法的真实地址,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8042256/
我想了解 Ruby 方法 methods() 是如何工作的。 我尝试使用“ruby 方法”在 Google 上搜索,但这不是我需要的。 我也看过 ruby-doc.org,但我没有找到这种方法。
Test 方法 对指定的字符串执行一个正则表达式搜索,并返回一个 Boolean 值指示是否找到匹配的模式。 object.Test(string) 参数 object 必选项。总是一个
Replace 方法 替换在正则表达式查找中找到的文本。 object.Replace(string1, string2) 参数 object 必选项。总是一个 RegExp 对象的名称。
Raise 方法 生成运行时错误 object.Raise(number, source, description, helpfile, helpcontext) 参数 object 应为
Execute 方法 对指定的字符串执行正则表达式搜索。 object.Execute(string) 参数 object 必选项。总是一个 RegExp 对象的名称。 string
Clear 方法 清除 Err 对象的所有属性设置。 object.Clear object 应为 Err 对象的名称。 说明 在错误处理后,使用 Clear 显式地清除 Err 对象。此
CopyFile 方法 将一个或多个文件从某位置复制到另一位置。 object.CopyFile source, destination[, overwrite] 参数 object 必选
Copy 方法 将指定的文件或文件夹从某位置复制到另一位置。 object.Copy destination[, overwrite] 参数 object 必选项。应为 File 或 F
Close 方法 关闭打开的 TextStream 文件。 object.Close object 应为 TextStream 对象的名称。 说明 下面例子举例说明如何使用 Close 方
BuildPath 方法 向现有路径后添加名称。 object.BuildPath(path, name) 参数 object 必选项。应为 FileSystemObject 对象的名称
GetFolder 方法 返回与指定的路径中某文件夹相应的 Folder 对象。 object.GetFolder(folderspec) 参数 object 必选项。应为 FileSy
GetFileName 方法 返回指定路径(不是指定驱动器路径部分)的最后一个文件或文件夹。 object.GetFileName(pathspec) 参数 object 必选项。应为
GetFile 方法 返回与指定路径中某文件相应的 File 对象。 object.GetFile(filespec) 参数 object 必选项。应为 FileSystemObject
GetExtensionName 方法 返回字符串,该字符串包含路径最后一个组成部分的扩展名。 object.GetExtensionName(path) 参数 object 必选项。应
GetDriveName 方法 返回包含指定路径中驱动器名的字符串。 object.GetDriveName(path) 参数 object 必选项。应为 FileSystemObjec
GetDrive 方法 返回与指定的路径中驱动器相对应的 Drive 对象。 object.GetDrive drivespec 参数 object 必选项。应为 FileSystemO
GetBaseName 方法 返回字符串,其中包含文件的基本名 (不带扩展名), 或者提供的路径说明中的文件夹。 object.GetBaseName(path) 参数 object 必
GetAbsolutePathName 方法 从提供的指定路径中返回完整且含义明确的路径。 object.GetAbsolutePathName(pathspec) 参数 object
FolderExists 方法 如果指定的文件夹存在,则返回 True;否则返回 False。 object.FolderExists(folderspec) 参数 object 必选项
FileExists 方法 如果指定的文件存在返回 True;否则返回 False。 object.FileExists(filespec) 参数 object 必选项。应为 FileS
我是一名优秀的程序员,十分优秀!