前言

多年以前一位老程序员告诉笔者代码片段(code snippets)是程序员的财富,他有一个U盘,里面装着他的财富。每当他需要切换电脑写代码的时候,他就会把把精心配置的字体、主题、代码片段等部署到新电脑上,然后开始高速编码。每次看他写代码都是一种享受,不过这是另一个故事了。

需求

多年之后,笔者也终于凑够了钱买了自己的Mac,闲暇无事的时候也会写写代码祭奠下逝去的青春。但是某些时候总会觉得很别扭,例如感觉字体和单位的有细小的差距,或者一个代码片段怎么也按不出来——最后发现是没有在这台电脑配置这段代码片段。这种事发生的事情多了之后,就会感觉厌烦,同样的操作为什么得重复两次、三次?或者拿出吃了几年灰的U盘抽插在各地的电脑上人工同步?就不能有什么办法可以一次更改多次应用?笔者稍微一拍脑门,想到了今天的主角——iCloud Drive

1、为什么使用iCloud Drive?

因为这是苹果本家的网盘,嵌入系统中,只要开启我们就无需关心上传下载,正如OneDrive在Windows一样,我们只需要把文件放进去,他就会自动开始上传,并在你的每一台苹果设备上同步。利用这点我们就能方便的做到在不同的设备上同步Xcode配置文件,无需手动同步或者上传下载。

2、其他的替代方案

GitHub之类的大型同性交友网站

目前想来用git应该更好更方便,不过实现起来有点复杂,有能力的朋友可以自己动手

OneDrive/坚果云等网盘

我觉得能有自带的还是用自带的吧

思路

总所周知Xcode的代码片段是保存在~/Library/Developer/Xcode/UserData/CodeSnippets路径下的,附近位置还有主题等配置信息。基于笔者的经验我们只需要备份同级目录下的CodeSnippets、FontAndColorThemes和KeyBindings三个子目录就行了。每当我们修改了代码片段、主题或者快捷键,把对应的文件放在iCloud Drive同步,当在其他电脑上时就使用最新的覆盖到对应目录即可。

脚本

虽说思路如此,但是笔者肯定不敢把这种三岁小孩子就能分析出来的东西发出来糊弄人。所以为了简化这个繁琐而又机械的操作,笔者编写了这样一个脚本:

#!/usr/bin/env bash

set -euo pipefail

################# variable define ##########
now=`date " %Y%m%d%H%M%S"`

red=`tput setaf 1`
green=`tput setaf 2`
yellow=`tput setaf 3`
reset=`tput sgr0`

xcode_dir="${HOME}/Library/Developer/Xcode/UserData"
cloud_backup_dir="${HOME}/Library/Mobile Documents/com~apple~CloudDocs/XcodeBackup"
local_backup_dir="${HOME}/资源/归档/XcodeBackup"

code_snippets="CodeSnippets"
font_and_color_themes="FontAndColorThemes"
key_bindings="KeyBindings"

########### MAIN ##################
# check directory exist
if [ ! -d "${cloud_backup_dir}" ]; then
 echo "${red}iCloud Drive备份路径不存在!${reset}"
 mkdir -p "${cloud_backup_dir}"
 echo "${green}自动创建iCloud Drive备份路径:${reset}${cloud_backup_dir}"
 else
 echo "${green}iCloud Drive备份路径:${reset}${cloud_backup_dir}"
fi

if [ ! -d "${local_backup_dir}" ]; then
 echo "${red}本地备份路径不存在!${reset}"
 mkdir -p "${local_backup_dir}"
 echo "${green}自动创建本地备份路径:${reset}${local_backup_dir}"
 else
 echo "${green}本地备份路径:${reset}${cloud_backup_dir}"
fi

# zip files
cd "${xcode_dir}"
zip -r "${cloud_backup_dir}/XcodeBackup ${now}.zip" "${code_snippets}" "${font_and_color_themes}" "${key_bindings}" &
zip -r "${local_backup_dir}/XcodeBackup ${now}.zip" "${code_snippets}" "${font_and_color_themes}" "${key_bindings}" &

wait

# delete unnecessary backup files
num=`ls -l "${cloud_backup_dir}" |grep "^-"|wc -l`
if [ ${num} -gt 5 ]; then
 num=`expr ${num} - 5`
 cd "${cloud_backup_dir}"
 ls -tr "${cloud_backup_dir}" | head -${num} | xargs rm
fi

num=`ls -l "${local_backup_dir}" |grep "^-"|wc -l`
if [ ${num} -gt 5 ]; then
 num=`expr ${num} - 5`
 cd "${local_backup_dir}"
 ls -tr "${local_backup_dir}" | head -${num} | xargs rm
fi

简化了这个繁琐的操作,仅需在开机的时候跑一下,就能达到自动备份的效果。功能也是十分的简单:

  • 首先创建了两个备份Xcode配置文件的路径,一个在云端,一个在本地(本地路径大家可以自行配置,一般也不会用上)。
  • 然后把Xcode归档到这两处各一份,笔者这里选用zip包而不是更高压缩比的7zip等是因为想做通用一点便于大家开箱即用,不需要额外安装其他软件。
  • 最后将多次运行后生成的老包删除,只保留最新的5个,以便节约宝贵的空间(毕竟笔者比较穷只舍得用免费的5g版)

有了这个脚本之后,大家只需要坚持开机的时候跑一跑就行了。笔者喜欢每天开机就更新下cocoapods、brew、brew cask这类的,所以就写了个脚本,刚好顺便也就备份一下。脚本思路大致如下,因为和主题无关就不细说了。

#!/usr/bin/env bash

open 自用魔法丝袜之影

wait

pod repo update --verbose &
更新Homebrew cask &
备份各种币钱包 &

备份Xcode等IDE配置文件 &

wait

killall 自用魔法丝袜之影

不过这样其实也不是很方便,毕竟打开terminal输入指令都很烦了,难道还要手动计算这台电脑的配置是否是最新的?然后再考虑是不是需要把云盘里面的配置解压到指定的位置覆盖?而且很有可能在做这些前已经把这台电脑的配置当最新版上传到云盘里了。

让所有的电脑用同一个版本的配置

笔者再次进行了思考。如果可以根据这些文件的最后修改日期和备份的文件进行比较,谁新就用哪个版本,那么不就实现了吗?只要我们确保每次修改都跑一次脚本,每次开机都跑一次,就能达到我们想要的效果了。至于如何判断文件的最后修改时间,笔者认为只需要一个根据文件名生成的key和一个对应的文件的最后修改时间做value的数据结构就行了(虽说也可以把备份的文件展开比较,但是因为笔者才疏学浅,尚不知如何操作,就只能通过键值对来判断了)

不过实际操作起来,再次彰显了笔者的才疏学浅,笔者也不知道如何在bash中创建一个高效并能持久化的键值对,如果哪位大佬知道请务必告诉笔者。

最后笔者想到Mac自带的SQLite3,虽说这样一个小小的功能上数据库是有一点高射炮打蚊子,但是能跑就行吧。脚本如下;

#!/usr/bin/env bash

set -euo pipefail

################# variable define ##########
now=`date " %Y%m%d%H%M%S"`

red=`tput setaf 1`
green=`tput setaf 2`
yellow=`tput setaf 3`
reset=`tput sgr0`

xcode_dir="${HOME}/Library/Developer/Xcode/UserData"
cloud_backup_dir="${HOME}/Library/Mobile Documents/com~apple~CloudDocs/XcodeBackup"
local_backup_dir="${HOME}/资源/归档/XcodeBackup"

xcode_backup_database="${HOME}/Library/Mobile Documents/com~apple~CloudDocs/.BackupDatabase"

code_snippets="CodeSnippets"
font_and_color_themes="FontAndColorThemes"
key_bindings="KeyBindings"

temp="DoNotModify"
database="${xcode_backup_database}/${temp}"

########### MAIN ##################
# check directory exist
if [ ! -d "${cloud_backup_dir}" ]; then
 echo "${red}iCloud Drive备份路径不存在!${reset}"
 mkdir -p "${cloud_backup_dir}"
 echo "${green}自动创建iCloud Drive备份路径:${reset}${cloud_backup_dir}"
else
 echo "${green}iCloud Drive备份路径:${reset}${cloud_backup_dir}"
fi

if [ ! -d "${local_backup_dir}" ]; then
 echo "${red}本地备份路径不存在!${reset}"
 mkdir -p "${local_backup_dir}"
 echo "${green}自动创建本地备份路径:${reset}${local_backup_dir}"
else
 echo "${green}本地备份路径:${reset}${cloud_backup_dir}"
fi

if [ ! -d "${xcode_backup_database}" ]; then
 echo "${red}同步数据库路径不存在!${reset}"
 mkdir -p "${xcode_backup_database}"
 echo "${green}自动创建数据库路径:${reset}${local_backup_dir}"
else
 echo "${green}数据库路径:${reset}${cloud_backup_dir}"
fi

sqlite3 "${database}" 'create table if not exists backupXcode(id integer primary key not NULL,key integer unique not NULL,value integer not NULL);'

#获取最后修改时间
cd "${xcode_dir}"
find "./${code_snippets}" "./${font_and_color_themes}" "./${key_bindings}" -type f >> ${temp}

while read path; do
 key=`md5 -q -s "${path}"`
 value=`stat -f "%m" "${path}"`
 isModify=`sqlite3 "${database}" "select value from backupXcode where key == '${key}';"`
 if [ -z ${isModify} ]; then
  echo "${yellow}本地Xcode配置尚未同步${reset}!"
  num=`ls -l "${cloud_backup_dir}" |grep "^-"|wc -l`
  if [ ${num} -ge 1 ]; then
   echo "${green}找到最新的Xcode配置,开始自动替换${reset}!"
   
   cd "${xcode_dir}"
   ## backup before
   zip -r "XcodeBackup.zip" "${code_snippets}" "${font_and_color_themes}" "${key_bindings}" &
   wait
   
   cd "${cloud_backup_dir}"
   newBackup=`ls -t | head -1`
   
   unzip -u "${newBackup}" -d "${xcode_dir}" &
   wait
   
   cd "${xcode_dir}"
   rm ${temp}
   find "./${code_snippets}" "./${font_and_color_themes}" "./${key_bindings}" -type f >> ${temp}
   echo 更新数据库...
   while read path; do
    key=`md5 -q -s "${path}"`
    value=`stat -f "%m" "${path}"`
    sqlite3 "${database}" "insert or replace into backupXcode values(NULL,'${key}',${value});" &
   done < ${temp}
  fi
  break
 fi
 if [ ${isModify} != ${value} ]; then
  if [ ${isModify} -lt ${value} ]; then
   echo "${yellow}本地Xcode配置超前${reset}!"
  else
   echo "${yellow}本地Xcode配置已经过期${reset}!"
   
   num=`ls -l "${cloud_backup_dir}" |grep "^-"|wc -l`
   if [ ${num} -ge 1 ]; then
    echo "${green}找到最新的Xcode配置,开始自动替换${reset}!"
    
    cd "${xcode_dir}"
    ## backup before
    zip -r "XcodeBackup.zip" "${code_snippets}" "${font_and_color_themes}" "${key_bindings}" &
    wait
    
    cd "${cloud_backup_dir}"
    newBackup=`ls -t | head -1`
    
    unzip -o "${newBackup}" -d "${xcode_dir}" &
    wait
   fi
  fi
  cd "${xcode_dir}"
  rm ${temp}
  find "./${code_snippets}" "./${font_and_color_themes}" "./${key_bindings}" -type f >> ${temp}
  echo 更新数据库...
  while read path; do
   key=`md5 -q -s "${path}"`
   value=`stat -f "%m" "${path}"`
   sqlite3 "${database}" "insert or replace into backupXcode values(NULL,'${key}',${value});"
  done < ${temp}
  break
 fi
done < ${temp}

wait
rm ${temp}

# zip files
cd "${xcode_dir}"
zip -r "${cloud_backup_dir}/XcodeBackup ${now}.zip" "${code_snippets}" "${font_and_color_themes}" "${key_bindings}" &
zip -r "${local_backup_dir}/XcodeBackup ${now}.zip" "${code_snippets}" "${font_and_color_themes}" "${key_bindings}" &

wait

# delete unnecessary backup files
num=`ls -l "${cloud_backup_dir}" |grep "^-"|wc -l`
if [ ${num} -gt 5 ]; then
 num=`expr ${num} - 5`
 cd "${cloud_backup_dir}"
 ls -tr "${cloud_backup_dir}" | head -${num} | xargs rm
fi

num=`ls -l "${local_backup_dir}" |grep "^-"|wc -l`
if [ ${num} -gt 5 ]; then
 num=`expr ${num} - 5`
 cd "${local_backup_dir}"
 ls -tr "${local_backup_dir}" | head -${num} | xargs rm
fi

后记

笔者简单测试了一下,基本上能用。以此思路,应该也可用在Alfred、vimrc等配置文件。不过依旧不是很方便,不过笔者才疏学浅,目前也就这个水平了,希望能对大家有所帮助,不知道大家有没有什么好的建议?笔者认为可以在Xcode关闭时自动运行本脚本,但是尚未找到好的胡克点(:」∠)_,如果大家有什么好的建议,欢迎PR

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对Devmax的支持。

如何利用iCloud Drive同步Xcode配置详解的更多相关文章

  1. ios – NSUbiquityIdentityDidChangeNotification和SIGKILL

    当应用程序被发送到后台时,我们会删除观察者吗?我遇到的问题是,当UbiquityToken发生变化时,应用程序终止,因为用户已经更改了iCloud设置.你们如何设法订阅这个通知,如果你不这样做,你会做什么来跟踪当前登录的iCloud用户?

  2. ios – 如何检查CNContact swift中的联系人来源?

    在联系人应用程序中,有像“iCloud”,“yahoo”,“gmail”这样的组.在swift中,是否可以仅从gmail源获取联系人?

  3. ios – 从icloud备份中限制sqlite-wal和sqlite-shm

    我是第一次使用coredata,我必须从文档目录中的iCloud备份限制sqlitedb文件,我使用下面的代码完成了它//阻止iCloud备份文档目录文件夹现在我不明白的是,我们还需要从icloud备份中限制sqlite-wal和sqlite-shm文件,如果是,那么如何从icloud备份限制sqlite-wal和sqlite-shm文件我想要一个解决方案,而无需从文档目录文件夹中更改sqlitedb位置…

  4. ios – iCloud键值存储同步问题(未调用NSUbiquitousKeyValueStoreDidChangeExternallyNotification)

    我正在为iOS编写一个小型通用游戏.高分将通过iCloudKey/Value商店在设备上同步.获得最新分数:存储新的高分但由于某种原因,分数不同步.>没有错误>没有崩溃>没有空值我总是得到当前设备的最高分,而不是其他设备.原因可能在于itunesconnect还是我的代码有问题?

  5. ios – 如何在仅使用NSUbiquitousKeyValueStore时在用户设备上编程检测iCloud是否启用?

    我正在使用NSUbiquitouskeyvalueStore将一些偏好数据同步到iCloud.我发现如果用户在“设置应用程序”中禁用iCloud的“文档和数据”项,则NSUbiquitouskeyvalueStore无法将其数据同步到iCloud.所以,我想先检查一下这个设置是否被打开.在谷歌搜索之后,我发现这个代码段:我想知道的是,这是否是唯一的检测方式,即使我只是使用NSUbiquitousk

  6. 是否可以检查iOS 7中是否启用iCloud KeyChain?

    我想检查一个用户是否在iOS设备上启用了iCloudKeychain,以便提示他们是否没有可以切换它以允许数据同步.有人可以建议我如何以编程方式实现?有没有办法阅读设置?谢谢.解决方法我向Apple开发者技术支持部门提出了这个问题,并得出答案是没有API.

  7. ios – 包含不支持的权利值

    一切顺利.但是当我上传我的二进制文件被拒绝与该错误:无效的代码签名权利代码签名授权无效:您的应用程序包的签名包含不支持的权利值.对于com.apple.developer.ubiquity-container-identifiers权限,数组中的第一个值必须由Apple在配置配置文件中提供的前缀,后跟bundle标识符后缀.捆绑标识符必须与您的某个应用或您被允许用作iCloud容器标识符的另一个应用的捆绑标识符相匹配.真的不知道我做错了什么,因为我已经做了这么多次,从来没有这个问题任何人指出我明显的错误可

  8. 什么原因导致iOS iCloud错误:错误域= BRCloudDocsErrorDomain代码= 12“操作无法完成.”延迟后重试:60

    iPhone/iOS应用程序正在使用CoreDatasqliteiCloudiOS.当应用程序第一次安装开始,并且iCloud中的先前的应用程序数据从先前安装或同一帐户中的其他设备发生,则会出现以下错误,然后是60秒的重试延迟,然后成功迁移到iCloud.结果是应用程序升级时,应用程序的用户认为他们的数据丢失.然而,在60秒的延迟之后,数据被恢复.错误和一些代码如下.什么原因导致这个错误在哪里可以进一步了解这个错误?

  9. ios – 使用Swift访问非默认的Cloudkit容器

    我有一个IOS8应用程序,它成功地将记录写入其默认的Cloudkit容器.现在我希望在OSX下的不同应用程序中读取和处理这些记录.我已经设置了具有iCloud访问权限的新应用程序,并选中了“指定自定义容器”选项.这已成功找到原始应用程序和我的容器没有错误要在Capabilities界面上修复.我的所有搜索都告诉我“同一个开发人员签署的两个应用程序可能共享同一个容器”但我找不到的任何教程文档都告诉我

  10. ios – 如何检测设备上的iCloud帐户是否更改?

    如何检测设备上使用的iCloud帐号是否已更改?用户从“设置”>iCloud和另一个用户登录他/她的帐户.打开应用程序时如何检测到此更改?

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部