解决iPhone5等32位设备,13位时间戳转换异常(转)

32-bit 64-bit

2015年初,所有提交审核的App需要适配64位,目的是充分利用ARM处理器的高性能支持,让应用能够有更为极致的体验。在iPhone5S以后的设备都是支持64位的。

一、时间显示引发的问题

问题页面

由于32位处理器的NSInteger是int类型的,导致溢出,下面我们来具体看一下。

1、服务端返回JSON数据

服务端返回日期数据为13位时间戳。

{
    "count": 0,
    "data": {
        "nowDate": 1508470153576,       //13位时间戳
        "skuCommission": {
            "actPrice": 89,
            "commissionRate": 0.1,
            "endTime": 1508547599000,   //13位时间戳
            "isDelete": 0,
            "limitCommissionRate": 0,
            "limitInventory": 0,
            "limitUseCoupon": false,
            "skuId": 0,
            "specialSell": true,
            "startTime": 1508461200000, //13位时间戳
            "type": 1
        },
        "saleInfo": {
            "endTime": 1508547599000,   //13位时间戳
            "price": 89,
            "startTime": 1508461200000  //13位时间戳
        },
    },
    "isRedirect": 0,
    "isSuccess": 1,
    "login": 0
}

2、iOS客户端Model数据模型

客户端通过MJExtension(数据转模型开源库),解析为Model模型。
修改前:时间戳使用NSInteger类型
修改后:时间戳使用double类型(或者 long long类型)

@interface ProductDetailModel : NSObject
#@property (nonatomic, assign) NSInteger nowDate;               //修改前
@property (nonatomic, assign) double nowDate;                   //修改后
@end

@interface ProductSkuCommission : NSObject
@property (nonatomic, assign) CGFloat actPrice;
@property (nonatomic, assign) CGFloat commissionRate;
#@property (nonatomic, assign) NSInteger endTime;               //修改前
@property (nonatomic, assign) double endTime;                   //修改后
@property (nonatomic, assign) NSInteger gmtCreate;
@property (nonatomic, assign) NSInteger gmtModified;
@property (nonatomic, assign) NSInteger isDelete;
@property (nonatomic, assign) NSInteger limitCommissionRate;
@property (nonatomic, assign) NSInteger limitInventory;
@property (nonatomic, assign) NSInteger skuId;
@property (nonatomic, assign) BOOL specialSell;
#@property (nonatomic, assign) NSInteger startTime;             //修改前
@property (nonatomic, assign) double startTime;                 //修改后
@property (nonatomic, assign) NSInteger type;
@end

@interface ProductSaleInfo : NSObject
#@property (nonatomic, assign) NSInteger endTime;               //修改前
@property (nonatomic, assign) double endTime;                   //修改后
@property (nonatomic, assign) CGFloat price;
#@property (nonatomic, assign) NSInteger startTime;             //修改前
@property (nonatomic, assign) double startTime;                 //修改后
@end

3、出现溢出问题

❌时间戳转换成时间后,显示异常❌,如下:

// 原始数据
1508547599000

// 64位架构,使用NSInteger
1508461200000 10月20日09点

// 32位架构,使用NSInteger
927679104     01月12日01点

二、排查问题

排查了网络返回数据,用户设备时区,使用设备机型,使用设备系统,是否越狱,转换时间方法等等,后来发现在iPhone5,iPhone5C设备中存在该问题(目前App兼容iOS8以上,暂时不考虑iPhone5以下机型),其他设备是OK的,考虑问题出在机型之间存在的差异。

1、通过XCode,按住「Command」点击「NSInteger」查看NSInteger的定义如下:

32位CPU上,将NSInteger定义为了int类型
64位CPU上,将NSInteger定义为了long类型

2、通过Dash,查看Apple官网API文档,查看NSInteger的定义如下:

构建32位应用,NSInteger32位整型
构建64位应用,NSInteger64位整型

注意:由于long和NSInteger的字节数变了,所以在兼容的时候可能会导致溢出

相同类型在不同处理器中的差异:

类型32位 处理器64位 处理器
NSIntegerintlong
NSUIntegerunsigned intunsigned long
~~~
int32bit(4字节)32bit(4字节)
long32bit(4字节)64bit(8字节)
long long64bit(8字节)64bit(8字节)

变量范围:

整型有符号 范围无符号 范围
4字节-2147483648 ~ 21474836470 ~ 4294967295
8字节-9223372036854775808 ~ 92233720368547758070 ~ 18446744073709551615

三、为什么用NSInteger,不直接用int或long呢?

现在,Xcode可以支持开发32位和64位的应用,为了让开发者写出在其他系统上也能够运行的代码,苹果公司引入了NSInteger类型以及NSUInteger类型。这两种类型无论在32位系统还是64位系统上都能够通用。

NSInteger以及NSUInteger类型在苹果的类库中被广泛使用,所以写Objective-C的时候,会经常使用它们。

用NSInteger 声明的整型变量,你不用关心是32位还是64位不用关心是声明int还是longiOS编译器会自动分配完成

四、总结:

一个10位以上的整数。
64位系统中,使用NSInteger或者long类型,是可以正常存储的 。
32位系统中,由于字节数变了,导致溢出了。
兼容32位64位系统,使用intlong long(或者int32_tint64_t)这样的数据类型,比使用NSInteger要更加可靠,具体问题具体分析。

最后,使用了double代替NSInteger来解决该问题

### 回答1: 在Linux中,时间戳是指从1970年1月1日00:00:00 UTC到某一特定时间的秒数。而13时间戳是指从1970年1月1日00:00:00 UTC起至某一特定时间的毫秒数。 要将一个13时间戳转换为可读的日期和时间,我们可以使用一些Linux命令和脚本。首先,我们需要使用`date`命令来格式化时间戳。例如,假设我们有一个13时间戳`1609459200000`,表示2021年1月1日00:00:00 UTC。 我们可以使用以下命令将其转换为可读的日期和时间: ```shell date -d @$(echo "1609459200000 / 1000" | bc) ``` 在这个命令中,我们使用了`bc`命令来执行除法运算,将13时间戳转换为10的秒数。然后,我们将转换后的秒数作为参数传递给`date`命令,并使用`-d`选项将其格式化为人类可读的日期和时间。 运行以上命令后,将输出`Fri Jan 1 00:00:00 UTC 2021`,即时间戳`1609459200000`对应的日期和时间。 这是将13时间戳转换为可读的日期和时间的一个简单方法。但是,请注意,该方法只适用于运行Linux的系统。在其他操作系统中,可能需要使用其他工具或脚本来进行转换。 ### 回答2: 在Linux中,时间戳是指从1970年1月1日起的秒数。通常情况下,时间戳是10位数,但有时也可能是13位数。 要将13时间戳转换为一个可读的日期和时间格式,可以使用以下方法: 1. 将13时间戳除以1000,将其转换为10时间戳。这是因为10时间戳是以秒为单,而13时间戳是以毫秒为单。 2. 使用date命令来将10时间戳转换为可读的日期和时间格式。例如,可以使用以下命令: date -d @<10时间戳> "+%Y-%m-%d %H:%M:%S" 这会将时间戳转换为"年-月-日 时:分:秒"的格式。其中,%Y表示年份,%m表示月份,%d表示日期,%H表示小时,%M表示分钟,%S表示秒。 例如,假设有一个13时间戳为1551324589000,我们可以将其转换为10时间戳,然后使用date命令将其格式化为"年-月-日 时:分:秒"的格式。具体的步骤如下: 1. 将13时间戳除以1000,得到10时间戳:1551324589 2. 运行以下命令:date -d @1551324589 "+%Y-%m-%d %H:%M:%S" 输出结果为:2019-02-28 12:23:09 通过上述方法,我们可以轻松地将13时间戳转换为可读的日期和时间格式。 ### 回答3: 在Linux中,13时间戳指的是以毫秒为单时间戳。要将其转换为可读的日期和时间格式,可以使用date命令。 首先,将13时间戳除以1000,以将其转换为以秒为单时间戳。然后,可以使用date命令来将时间戳转换为可读的格式。通过在date命令后附加"+格式"参数,可以指定所需的日期和时间格式。 以下是一个示例命令: timestamp=1609459200000 date -d @$(($timestamp/1000)) +"%Y-%m-%d %H:%M:%S" 在上面的命令中,我们首先将13时间戳除以1000,然后使用date命令和-d参数指定将时间戳转换为日期和时间。最后,我们使用+"%Y-%m-%d %H:%M:%S"参数来指定希望获得的日期和时间格式。 以上命令将13时间戳1609459200000转换为2021-01-02 00:00:00的可读格式。 使用这种方式,你可以将任何13时间戳转换为所需的日期和时间格式。只需替换代码中的timestamp值(以毫秒为单13时间戳),然后运行命令即可。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值