分类
iOS开发 MacOS 技术

用 Xcode cloud 解决 Xcode-beta 无法提交 App 的问题

前两天苹果的 WWDC 2024结束,每次苹果召开 WWDC 都是苹果平台开发者的节日。听完 Keynote,大家往往就想最先下载测试版本的操作系统和 Xcode 来尝鲜。

但是这里也有一个矛盾,一般来说从测试版到正式版要1个多月的时间。而测试版的 Xcode-beta 是无法提交 App。也就是尝鲜的话,你就有一个多月没办法提交你的 App。

解决的方法,一般是如果自己有多台 mac,其中一台不升级,专门留着用来提交 App。还有一种就是安装多系统。

但是前几年,苹果推出了 Xcode cloud,我就想是不是可以利用 Xcode cloud来解决这个问题呢?果然解决了。

分类
iOS开发 IT行业 技术

WWDC 2024将近,苹果的AI革命会给基于AI的独立开发者带来灭顶之灾么?

6月11日,WWDC 2024 就要召开了。每年 WWDC 召开以后,随着苹果发布一堆新的内嵌功能,就会在推特和各种 blog 网站引发很多讨论。很多独立开发者就会哭着喊着说,苹果在搞垄断,苹果新内嵌的某某功能会影响他们的生意,甚至毁掉他们的营生。

分类
iOS开发

Swift编程中应用@available和deprecated在自己的编程中提供便利

在 Swift 中,@available 是一个属性标注(Attribute),它用来表示一个类、结构体、枚举、函数或者方法等的可用性信息,指示特定的平台和版本。通过这个属性标注,开发者可以对代码中的各个部分设定适用的操作系统平台及最低支持的版本,也可以指定在某个版本中它们已经过期或不推荐使用,甚至是未来某个版本中会被添加的特性。

我们使用苹果的标准 Swift 库的时候,经常遇到 @available ,一个老代码的某个函数调用,系统升级后,可能会发现一个警告,

'oldMethod(text:messages:key:)' is deprecated: Use newMethod() instead

意思就是这个方法或者函数,现在已经被废弃了,最好不要再使用。在这个情况下,往往你其实还可以编译还可以执行。但是这些被废弃的老方法,要么也许有 bug ,或者系统的代码风格,习惯变了,已经被推荐使用了。有的人觉的既然可以编译和执行,何必理会呢?其实不然,有些方法会在某一个操作系统版本废弃,然后在后续的某个大操作系统版本就彻底消失,无法编译和运行。也就是说,废弃声明就是提醒你可以修改代码了,未来这个方法或者类有可能完全不能用了。最好尊重这些声明来写代码。

我们去看系统代码和头文件的时候,会看到很多类似的例子:

1. 指定功能在平台上的最小可用版本:
@available(iOS 10, *)
func newiOS10Method() {
// 这个方法只会在 iOS 10 及以上版本可用
}

2. 标示一个功能在新版本中被弃用:
@available(iOS, deprecated: 11.0, message: "Use newMethod instead")
func oldMethod() {
// 这个方法在 iOS 11 中被标记为弃用
}

3. 标示一个功能在新版本中被废弃或移除:
@available(iOS, obsoleted: 11.0, message: "Use anotherMethod instead")
func obsoleteMethod() {
// 这个方法从 iOS 11 开始就废弃了,调用它会出错
}

4. 标示功能未来新增:
@available(iOS 15, *)
func futureMethod() {
// 这个方法在未来的 iOS 15 版本中会被添加
}

5. 跨多平台设置可用性:
@available(iOS 10, macOS 10.12, *)
func crossPlatformMethod() {
// 这个方法在 iOS 10 和 macOS 10.12 及以上版本都可用
}

6. 为不同的平台指定不同的最小版本:
@available(iOS 14, watchOS 7, *)
func multiPlatformMethod() {
// 对于 iOS 平台,这个方法要求版本 14.0 及以上
// 对于 watchOS 平台,这个方法要求版本 7.0 及以上
}

但是,这是一般情况下我们接触的 @available 和 deprecated。其实我们自己的代码里面也可以用这些用法。比如我最近在给 TinyStudio(最好的Mac字幕软件之一)增加新的功能,代码里面涉及到如何调用 ChatGPT,以及用 ChatGPT 来进行一些复杂的字幕后续处理操作。

在此之前我不知道 Swift 的 await 和 actor 语法,所以涉及到 ChatGPT 的代码非常复杂,比如一个巨长的文本发给 ChatGPT 需要实现分段,然后一段一段的去调用 ChatGPT 来进行处理。代码就非常复杂,我用递归调用和回调函数的方法来做,非常麻烦。虽然我可以轻松的写出这样的代码,但是这样的代码在调试、调优和异常处理方面简直是噩梦。

我知道了 Swift 的 await 和 actor 语法后,发现这个场景用这些并发语法来写,代码可以变得非常简洁,而且在调试、调优和异常处理方面非常方便。于是我就着手写 ChatGPT 和文本处理库的异步版本(await/actor)。写的时候,原有的函数叫 send ,新版本就叫做 asyncSend,原有的函数叫 format,新的就叫做 asyncFormat。

初步实现以后,我发现原有的 send 和 format 在很多地方都有调用,直接用文字搜索也会遇到一些类似的函数名。于是我就想起了 @available 和 deprecated 语法。

首先,我先把 ChatGPTKit 类和 TextKit 类的新老版本函数,放在不同的 extension 里(不影响执行,但是从代码来看,就可以把不同类型的方法条理清晰的分明白了):

// 遗留版本函数
extension TextKit {

func format(textString) {
...

// 遗留版本函数
extension TextKit {

func asyncFormat(textString) {
...

然后,在遗留版本函数的 extension 声明前,标注 @available 即可:

// 遗留版本函数
@available(*, deprecated, message: "老版本已废弃,请使用异步版本")
extension TextKit {

func format(textString) {
...

这时候编译程序,就会发现还在用老版本函数的代码就会被编译器发出警告,然后我们根据警告信息,一一替换为新版本就好了。如下:

分类
iOS开发

使用StoreKit2不再需要恢复购买按钮,但是为了审核你还需要放一个【AppStore审核手记】

昨天我提交了一个新App,2023年我的计划就是开发一系列的小App试图在独立开发领域有一份收入。但是今天一早就发现被拒绝了。

理由是:

分类
iOS开发 技术

从零开始,使用SwiftUI和PDFKit快速构建完全可定制的PDF阅读器

SwiftUI快速创建UI的能力非常强大,苹果还提供了PDFKit框架,所以,用SwiftUI和PDFKit可以用非常少的代码,非常快速的创建一个完全可定制的PDF阅读器。

我们来从零开始,做一个PDF阅读器。

分类
iOS开发 鸡汤

接受现实、持续改进,心怀梦想、永不放弃

最近我的App《英语轻松读》iOS版本连续更新了两个版本。其实上一个版本 1.17 到今天已经有一年多了。必须要升级的原因是,iOS 16带来的一些API和限制的改变,造成 1.17 版有频繁崩溃的问题。

然而从接到用户的抱怨到解决也花了很久,最后其实就改了几行而已。但是这个代码确实好久没碰了,我碰起来有点心慌。但是真的塌下心去研究,其实也没有花多久就解决了问题。然而马上发现还有一个 UI Bug,我也很快解决了,提交了。

分类
iOS开发 IT行业

苹果iPhone12旗舰机上的黑科技激光雷达到底是什么东西?有什么用处?

今年苹果的 iPhone12旗舰机型上面搭载了激光雷达。很多人其实都听说过激光雷达,苹果的发布会上也提了这个东西,但是其实我发现很多人都不了解这个东西到底是怎么回事。

所以,我今天就想从头从原理里,给大家讲清楚激光雷达是什么东西。以及它到底有什么用处。给大家讲明白,为什么苹果会在旗舰机型上放这么一个东西,以及这个东西对我们的未来、对这个世界有什么改变。

分类
iOS开发 技术

Xcode 12 比 Xcode 11 有很多改进, 我遇到的一个具体案例

编译差异

最近同时使用 Xcode 11 和 Xcode beta (也就是12),发现,后者性能提升超级明显,而编译器的理解能力也提升很多。

比如,我的一个SwiftUI代码里面,有一个稍微复杂一点的判断逻辑,有三个 if/else 分支,如下:

Group {
    if(index>0 && item.rating>4.0) {
        PoiLargeImageCell(item:item)
    }else if(index==0){
        InvisibleCell(largeHeadView:self.$largeHeadView)
        PoiCell(item: item)
    }else {
        PoiCell(item: item)
    }
}

这个代码在 Xcode 12 可以正常编译,没有任何问题。但是,在 Xcode 11 下就会直接报错,而且是那种完全文不对题的错误信息。错误信息不是在 Swift 5 发布后有改善了么?但是在这个问题下仍旧文不对题,所以,我只能把这个稍微复杂一点的 if/else 语句改为两个 Group 嵌套的做法,如下,这样在 Xcode 11 就可以正常编译了。

Group {
    if(index==0){
        ZStack{
            InvisibleCell(largeHeadView:self.$largeHeadView)
            PoiCell(item: item)
        }
    }else {
        Group {
            if(index>0 && item.rating>4.0){
                PoiLargeImageCell(item:item)
            }else {
                PoiCell(item: item)
            }
        }
    }
}

为了迁就编译器,代码变啰嗦多了。不过暂时需要这么做,期待 Xcode 12 早日正式发布。这么多年以来,第一次感觉苹果的 Beta 版本这么靠谱,感觉苹果今年果然有点进步。

提交上出现的错误信息

今天我用 Xcode 11 提交一个 app ,遇到了一个以前没见过的错误信息,“Profile doesn’t include the com.apple.application-identifier entitlement.”。查了下,原来跟 MacOS Big Sur 有关系,跟 Xcode 11 和 Xcode 12 beta 也有关系。解决方法很搞笑,那就是用 Xcode 11来编译打包,用 Xcode 12 beta 来 upload app。详情可以参考苹果开发者论坛的相关帖子

分类
iOS开发 IT行业

iPad革命:开发者的机遇和挑战

iPad采用了和iPhone一样的底层系统,开放方式基本相同。同时性能大幅提升,屏幕尺寸大幅度提升,以前在iPhone上无法实现,或者难以实现的一些应用的机会来了。所以,现在对每个有志于iPad开发的人来说,唯一的挑战是你的想象力和你的执行能力。