在64位env中将指针转换为长端口问题

| 我正在将应用程序从32位移植到64位。 尽管是C ++,但它是C样式编码(旧产品)。我有一个问题,其中结合使用联合和结构来存储值。这里使用了一个名为\“ Any \”的自定义数据类型,该数据类型应保存任何基本数据类型的数据。 Any的实现如下:
typedef struct typedvalue
{
long data; // to hold all other types of 4 bytes or less
short id; // this tells what type \"data\" is holding
short sign; // this differentiates the double value from the rest
}typedvalue;

typedef union Any 
{
double any_any;
double any_double; // to hold double value
typedvalue any_typedvalue;
}Any;
联合的大小为8个字节。他们使用并集,以便在给定的时间只有一个值,并且使用结构来区分类型。您可以在任何给定时间存储double,long,string,char,float和int值。就是这个主意。 如果它是double值,则该值存储在any_double中。如果是其他任何类型,则将其存储在\“ data \”中,并将值的类型存储在\“ id \”中。 \“ sign \”将告诉您值\“ Any \”是否为double或其他类型。 代码中广泛使用any_any来复制地址空间中的值,而与类型无关。 (这是我们最大的问题,因为我们在给定的时间不知道会发生什么!) 如果假定要保留其字符串或指针\“ Any \”,则将其存储在\“ data \”(类型为long)中。在64位中,问题出在这里。指针是8个字节。因此,我们需要将“ long”更改为等效的8个字节(long long)。但是,这将使联合的大小增加到16个字节,并且“ any_any \”的随意使用将引起问题。 \“ any_any \”的用法过多,您不确定它可以容纳什么。 我已经尝试了这些步骤,但未成功: 1.在结构中将“ long data”更改为“ long long data”,这将使联合的大小为16个字节。 -这将不允许以\“ any_any \”(8字节)的形式传递数据。 2.声明该结构作为并集内的指针。并将结构中的“长数据”更改为“长数据”。 -这里遇到的问题是,由于它是一个指针,我们需要为该结构分配内存。自由使用\“ any_any \”使我们很难分配内存。有时我们可能会覆盖内存,从而擦除该值。 3.创建一个单独的集合,该集合将保存\“ data \”(一个键值对)的值。 -这将不起作用,因为此实现是应用程序的核心,该集合将运行成百万个数据。 有人可以帮我吗?     
已邀请:
        “有人可以帮我吗?”这听起来像是在绝望中哭泣,我完全理解。 编写此代码的人绝对不尊重将来的验证性或可移植性,现在您要为此付出代价。 (对于任何说“但是我们的平台是32位!我们永远不会使用64位!”的人来说,这都是一个教训。) 我知道您会说“但是代码库太大”,但是最好重写产品。这次正确地做吧!     
        忽略原来的设计是疯狂的事实,您可以使用ѭ1soon(或很快使用
<cstdint>
)来获得一些可预测性:
struct typedvalue
{
  uint16_t id;
  uint16_t sign;
  uint32_t data;
};

union any
{
  char any_raw[8];
  double any_double
  typedvalue any_typedvalue;
};
您仍然不能保证
typedvalue
会被紧紧包装,因为非
char
成员没有对齐保证。您可以制作
struct Foo { char x[8]; };
,然后按自己的方式打孔,例如
*(uint32_t*)(&Foo.x[0])
*(uint16_t*)(&Foo.x[4])
,但这也非常丑陋。 如果您使用的是C ++ 0x,我肯定会在某个地方给
sizeof(typedvalue) == sizeof(double)
抛出一个静态断言。     
        如果您需要同时存储8个字节的指针和\“ type \”字段,那么您别无选择,只能使用至少9个字节,并且在64位系统上对齐时,可能会将其填充到16个字节。 您的数据结构应类似于:
typedef struct {
    union {
        void   *any_pointer;
        double  any_double;
        long    any_long;
        int     any_int;
    } any;
    char        my_type;
} any;
如果使用C ++ 0x,请考虑对
my_type
字段使用强类型枚举。在早期版本中,“ 12”所需的存储空间取决于实现,并且可能超过一个字节。 为了节省内存,您可以使用(特定于编译器的)指令来请求对数据结构进行最佳打包,但是由此导致的未对齐内存访问可能会导致性能问题。     

要回复问题请先登录注册