原创:知识点总结性文章
创作不易,请珍惜,之后会持续更新,不断完善
个人比较喜欢做笔记和写总结,毕竟好记性不如烂笔头哈哈,这些文章记录了我的IOS成长历程,希望能与大家一起进步
温馨提示:由于不支持目录跳转,大家可通过command + F 输入目录标题后迅速寻找到你所需要的内容
目录
- 一、Git
- 1、简介
- 2、安装和使用
- 3、常用命令
- 二、使用 CocoaPods 工具管理依赖库
- 1、安装 cocopods
- 2、安装框架
- 3、cocoapods常见问题和解决办法
- 三、制作cocoapod库
- 1、iOS 组件化/模块化的需求
- 2、cocoapod 的原理
- 3、创建框架工程
- 4、配置 .podspec 文件
- 5、更新 GitHub 里面的库
- 6、提交 .podspec 文件到 CocoaPods
- 7、使用创建的库
- 8、创建过程遇到的问题
- 四、python3.7
- 1、安装 python3.7
- 2、使用 python3.7
- 3、Mac上Python安装request库老是失败的问题
- 参考文献
一、Git
1、简介
a、管理模式
- 集中管理模式:SVN
- 分布式管理模式:有一个中心代码库,每位团队人员本地也都有代码库
- 代码库 (repository): 存放项目代码以及历史备份的地方
- 分支 (branch):用于验证和实验一些想法、版本发布、缺陷修改
- 合并分支 (merging branch)
- 签出 (checkout):从代码库获得文件或目录,将其作为副本保存在目录下
- 提交 (commit):修改的文件或目录作为新版本复制回代码库
- 冲突 (conflict)
- 解决 (resolution):需要人为干预解决,以接受一组更改并删除另一组更改
- 索引 (index):快照
b、Git 还有成熟的代码托管服务 GitHub 网站
- 如果项目不需要与其他开发人员协同开发,我们就不需要
Git
服务器。 - 任何能够提供通信协议和安全认证的代码库, 都可以认为是服务器。
-
HTTP(S)
协议和SSH
协议都可以支持远程库读写 、身份认证等操作。 -
Git
常用的协议是SSH
协议,我们需要在本机上生成公钥后,将公钥提供给GitHub
网站 - 只有付费账户才可以创建私有库
- 选中
Initialize this repository with a README
复选框创建一个README
文件,用来说明这个代码库的用途 - 但此时代码库中还是空的,我们需要在本地电脑中推送一个
Xcode
项目到GitHub
代码库 - 通过
SourceControl- Push
菜单项推送到GitHub
服务器 ,其中选择Remote
为origin/master
- 派生与
Git
中的分支很像,可以把它理解为代码库级别的分支,把Fork
代码库派生到当前账户下面了 - 点击
Transfer
按钮,输入要转移的代码库和新的拥有者 - 组织中会有管理员,他可以管理代码库,对代码库重新命名 、 删除和转移
- 普通成员对代码库可以接受推送请求、合并推送
c、注意点
- 每天修改量在10M以内,尽量每天提交修改,防止一次性提交造成大量代码冲突问题
- 每次
push
之前一定需要先进行pull
操作,先将本地更新再提交自己的修改 - 遇到冲突,可以使用如
$ open ucareshop/AppDelegate.m
打开文件的冲突部分将其删除 - 切换到分支再
git push
,之后等代码审核通过了再更新到master
2、安装和使用
a、安装镜像
使用以下命令添加 TUNA 源并移除 gems
默认源。
gem sources --add https://mirrors.tuna.tsinghua.edu.cn/rubygems/ --remove https://rubygems.org/
❷ 列出已有源,应该只有 TUNA 一个
xiejiapei@MacBook-Pro ~ % gem sources -l
*** CURRENT SOURCES ***
https://mirrors.tuna.tsinghua.edu.cn/rubygems/
b、配置 Git 的用户信息
当安装完 Git
应该做的第一件事就是设置你的用户名称与邮件地址
git config --global user.name "Xie Jiapei"
git config --global user.email [email protected]
c、Git 通常使用流程
// 切换到本地想安装的目录
$ cd Desktop/
// 安装Github上的项目
git clone http://gitlab.10101111.com:8888/jiapei.xie/gitlearn.git
// 切换进该项目目录
$ cd gitlearn/
将本地想要添加的文件拖进到下载下来的项目中
// 切换到分支,不影响 mainTree
$ git checkout develop
// 将拖进来的文件添加到项目中
git add .
// 本地提交,写好提交原因
git commit -m "init"
// 先拉下来远程Github上的最新内容
// git pull命令是git fetch和git merge命令的一个组合
git pull
// 再将自己修改后的内容提交到远端
git push
// early EOF 倘若文件太大则需要配置提交的缓存大小
git config --global http.postBuffer 15728640
d、完成 Xcode 中GitHub帐号的设置
点击「+
」按钮添加帐号并选择 GitHub
之后,你会发现此处需要填写 Account
和 Token
。其中 Account
是 GitHub
用户名,Token
则是一个特殊的许可,它允许 Xcode
使用你的 GitHub
帐号管理你的 Remote Repository
远程仓库。
进入 GitHub
设置页面,点击右上角的「Generate new token
」即可生成一串许可数字,将这串数字复制回到 Xcode
登录界面的 Token
文本框中,并点击「Sign In
」登录即可完成 GitHub
帐号的设置。
3、常用命令
查找命令的帮助信息
% git help
目录和文件
// 新建目录
% mkdir learngit
// 用于显示当前目录路径
% pwd
// 回到上一级
% cd ..
// 新建文件
% touch readme.txt
// 查看文件内容
% cat Podfile
// 查看文件状态,用于解决冲突
% git status
// 查看所有文件
% ls -al
分支
// 创建仓库
% git init 这个目录默认是隐藏的
// 创建并切换到分支
% git checkout -b develop
// 切换到master
% git checkout master
// 删除分支
% git branch -d develop
vim 文本编辑器
// 进入vim文本编辑器
% vim readme.txt
// 保存修改并退出
wq (或者shift zz)
// 忽略文件
% vim .gitignore
log 日志
// 历史记录
% git log
// 如果嫌输出信息太多看得眼花缭乱,可以使用以下语法
% git log --pretty=oneline
// 如果只想看最近几条的数据
% git log -n 1
// 例如
1 file changed:1个文件被改动(我们新添加的readme.txt文件)
2 insertions:插入了两行内容(readme.txt有两行内容)
reset 回退
// 寻找版本号
% git reflog
// 文件的索引:文件内容的哈希值
5e9ae2dd
// 回退机制,上上一个版本就是HEAD^^,上100个版本写成HEAD~100,版本号没必要写全,前几位就可以了
% git reset --hard HEAD^(版本号)
% git reset HEAD readme.txt
// 回滚版本
% git revert
// 删除文件
% rm test.txt
二、使用 CocoaPods 工具管理依赖库
1、M1 MacBook安装cocoapods
CocoaPods
搭建了一个中心库, 第三方库被收录到该中心库中,没有收录的第三方库是不能使用CocoaPods
管理的,这就是所谓的"中心化“思想。
CocoaPods
在下载第三方库后,会将其编译成静态链接库或动态框架文件,然后会修改Xcode
项目属性,配置依赖关系,这就是"侵入性"。
mac m1
笔记本都会自带可用的ruby
版本,也就是不需要你去升级ruby
,因为升级它还要装homebrew
,你装这个又会遇到一些坑,所以能用系统自带的就用它。你可以用如下命令查看系统的ruby
版本:
ruby -v
ruby 2.6.10p210 (2022-04-12 revision 67958) [universal.arm64e-darwin22]
查看ruby
镜像源sources
的命令:
gem source -l
*** CURRENT SOURCES ***
https://rubygems.org/
https://rubygems.org/
这个源在国内是访问不到的,所以你需要先移除这个镜像,然后添加国内可访问的镜像 https://gems.ruby-china.com/
(只保留这一个镜像),使用如下命令:
gem sources --remove https://rubygems.org/
gem sources --add https://gems.ruby-china.com/
https://rubygems.org/ removed from sources
https://gems.ruby-china.com/ added to sources
再验证下就会发现变成了https://gems.ruby-china.com/
:
*** CURRENT SOURCES ***
https://gems.ruby-china.com/
你需要更新最新的xcode
命令工具,用下面命令,如果已经更新了他会提示已经是最新的,没有的话会提示你是否安装,你只要确认安装就行:
xcode-select --install
xcode-select: error: command line tools are already installed, use "Software Update" in System Settings to install updates
然后安装最新的cocoapods
,使用如下命令:
sudo gem install cocoapods -n /usr/local/bin
......
34 gems installed
查看当前 pods
版本:
pod --version
1.11.3
查看pods
源:
pod repo list
0 repos
但是你使用pod search AFN
时你就发现它报错了,其实它是m1芯片与ruby
不兼容问题,错误如下(或者其他错误信息都可以使用这种方法试试):
LoadError - dlopen(/Library/Ruby/Gems/2.6.0/gems/ffi-1.13.1/lib/ffi_c.bundle, 0x0009): missing compatible arch in /Library/Ruby/Gems/2.6.0/gems/ffi-1.13.1/lib/ffi_c.bundle - /Library/Ruby/Gems/2.6.0/gems/ffi-1.13.1/lib/ffi_c.bundle
解决它首先找到finder
(访达),然后找到应用程序选项,在找到实用工具文件,点击进去找到终端,然后右键点击它,选中"显示简介",你会发现一个选项使用Rosetta打开
,对,问题的关键就是这个选项,要勾选它,关闭后重启终端后你还要输入命令让它生效,命令如下:
sudo gem install ffi
然后你再去执行pod
命令pod search AFN
就成功了。
2、安装框架
❶ 在项目根目录下建立 Podfile 文件,然后修改其中的内容
# Uncomment the next line to define a global platform for your project
# platform :ios, '9.0'
target 'UseUIControlFramework' do
pod 'DZNEmptyDataSet'
pod 'MGSwipeTableCell'
pod 'FSCalendar'
pod 'Toast'
pod 'MBProgressHUD'
pod 'zhPopupController'
pod 'Masonry'
end
❷ 明确第三方版本号
指定依赖库名一般是指 GitHub
上库的项目名,这个名字如果不能确定, 需要到 GitHub
上查询确定。
pod 'AFNetworking'// 不显式指定依赖库版本,表示每次都获取最新版本
pod 'AFNetworking', '5.0'// 只使用5.0版本
pod 'AFNetworking', '> 5.0'// 使用高于5.0的版本
pod 'AFNetworking', '>= 5.0'// 使用大于或等于5.0的版本
pod 'AFNetworking', '< 5.0'// 使用小于5.0的版本
pod 'AFNetworking', '<= 5.0'// 使用小于或等于5.0的版本
pod 'AFNetworking', '~> 4.5'//使用大于等于4.5但小于5.0的版本
❷ 打开终端安装框架
xiejiapei@xiejiapeis-iMac Example % pod install
Analyzing dependencies
Downloading dependencies
Installing XJPCategoryKit 0.1.0
Generating Pods project
Integrating client project
Pod installation complete! There are 3 dependencies from the Podfile and 3 total pods installed.
3、cocoapods常见问题和解决办法
iOS工程在组件化的过程中难免会碰到cocoapods的一些疑难问题,而网上的资料鱼龙混杂,严重阻碍开发进程。
问题1:podspec文件lint不通过或者无法push
加了—verbose
可以看到Error
提示:
Xcodebuild error: Unable to find a destination matching the provided destination specifier:
{ id:DB7BC0D9-CA24-4CA5-8FE9-AE322FD7B059}
解决办法:在xcode的simulators中找到这个id
的模拟器,删除,重试命令。如果仍然出现该提示,再删再试,直到所有把出错的模拟器都删了就好了。怀疑新版的cocoapods(1.10.1)查找不到Xcode后安装添加的模拟器。
问题2:pod库加了新文件,删了原标签(0.1.1),在新的commit上重新打这个标签(还是0.1.1),主工程pod install/update,都无法download到新文件
原因:本地有缓存
解决办法:
cd ~/Library/Caches/CocoaPods/Pods/Release // pod库缓存的文件
rm -r 删除这个库
重新pod install
即可。
问题3:pod库里的swift代码,依赖OC的库(比如叫NEToolKit),编译不过
解决办法:在swift文件开头添加一行代码:
import NEToolKit
问题4:pod install 提示文件重复
主工程Podfile
同时引入A、B两个pod
,A依赖B,但是这两个B的版本不同,需要在A的pod
工程下执行pod update
,将A使用的B的版本更新到最新,然后主工程再pod install
,这样就不会发生引入了两个版本的B导致的文件重复了。
问题5:pod repo push时提示Found multiple specifications for ''*
原因:pod repo list
发现同时存在cocoapods和trunk
,导致specifications
重复。但是pod repo remove trunk
后再pod repo push
,trunk
会被自动创建,仍然会出现该错误提示
解决办法:pod repo push
时加上-sources='https://github.com/CocoaPods/Specs.git'
,指定specifications
的来源。
问题6:pod repo push时提示错误podspec specification does not validate
原因:低级错误,podspec
指定的version
没打tag
。
问题7:swift的pod(比如叫LFTWidget)使用了oc的pod(比如叫NEToolKit),pod install失败,提示如下错误信息:
[!] The following Swift pods cannot yet be integrated as static libraries
The Swift pod `LFTWidget` depends upon `NEToolKit`, which does not define modules. To opt into those targets generating module maps (which is necessary to import them from Swift when building as static libraries), you may set `use_modular_headers!` globally in your Podfile, or specify `:modular_headers => true` for particular dependencies.
原因:在私有库中不能使用bridging-header.h
桥接文件,swift只能通过swift module
机制来引用其他oc模块。
解决办法:podfile
里将依赖的OC库挨个加上:modular_headers => true
问题8:提示libyary找不到
原因:一般libyary
找不到,都是searchPath
的问题。在project
里,Test
和Release
的配置是一样的;而在Pod
里,hook
的时候会更改配置,将Release
也配置成和Test
一样,就好了。一般pod
有dependency
的时候引用是没有问题的,但如果pod
里面的层级比较多,在hook
的时候就要加inhertied
。
解决办法:podfile
文件里添加如下代码(供参考):
post_install do |installer|
installer.pods_project.targets.each do |target|
target.build_configurations.each do |configuration|
if configuration.name == 'Debug'
configuration.build_settings['GCC_PREPROCESSOR_DEFINITIONS'] ||= ['$(inherited)','DEBUG=1']
elsif configuration.name == 'Test'
configuration.build_settings['GCC_PREPROCESSOR_DEFINITIONS'] ||= ['$(inherited)','TEST=1']
elsif configuration.name == 'Release'
configuration.build_settings['GCC_PREPROCESSOR_DEFINITIONS'] ||= ['$(inherited)']
end
end
end
end
三、制作 cocoapod 库
1、iOS 组件化/模块化的需求
当 App 的功能逐渐庞大,开发人员越来越多的时候,就会出现各种各样的问题。这时合理的组织架构就变得尤为重要,通常的做法就是将 App 内的功能进行模块化。只有项目到达一定规模后,开始出现相对独立的业务,团队人员开始增加,单一项目的开发模式成为了开发效率的掣肘,这个时候才是应该开始做组件化的时候。
问题
- 代码量庞大,相同功能代码可能反复被实现,导致编译速度变慢
- 功能交织,难以抽离,导致问题排查难度加大,提交时容易出现冲突
- 只要一处出现错误就会导致整个项目难以运行
模块化
- 按照功能将项目分成不同的模块,不同的人员维护不同的模块,这样可以使责任清晰
- 不经常改动的代码做成模块之后就不需要再编译整个项目,这样可以增加编译速度
- 模块之间相互独立,减少依赖,不容易出现代码提交冲突
盲目组件化导致开发效率变低
- 两三个人开发的简单项目,组件化的解耦反而可能会导致代码逻辑变复杂
- 组件化粒度过细也会导致一个业务需求需要修改多个组件,容易让代码追踪变得困难
- 与对应组件负责人的沟通会拖慢开发的进度
2、cocoapod 的原理
点开Specs后,在This repository
里输入AFNetworking
进行搜索。
CocoaPods
帮我们管理第三方和自己的库,它本身也是一个.xcodeproject
工程,只不过它是专门管理第三方和自己的库的工程。它的工作原理就是:你把.podspec
文件制作好,然后trunk push
给它,它再把.podspec
文件保存为.podspec.json
文件放到GitHub
,你需要的话,它通过这个.podspec.json
文件把GitHub
上对应的开源库的文件下载下来,放到你的工程里。
3、创建框架工程
a、在 GitHub 上新建一个仓库
b、区分私人框架和公共pods库不同的创建方式
很多教大家制作pods
的教程令人感到可笑又可恨,他们连最基本的如何创建框架工程都没告诉大家,导致很多小白忙活了半天还是在做APP而不是在做框架,还由于造成了一堆意外错误而浪费了大量时间。
不要用上面这个来创建工程。作为小白的我最初就是由于分不清私人框架和公共pods
库不同的创建方式,使用了上面的方式创建了私有框架来进行下面的流程导致了一大堆莫名其妙的诡异错误,浪费了很多时间。
c、创建相应的模板工程
正确的步骤是使用下面的方式来创建公共的pods
库。
xiejiapei@xiejiapeis-iMac Desktop % pod lib create XJPCategoryKit
在终端运行以上命令后会进入后续的选择流程,cocopods
通过这些选择会为你创建相应的模板工程,包括框架、使用的示例和测试。
Cloning `https://github.com/CocoaPods/pod-template.git` into `XJPCategoryKit`.
Configuring XJPCategoryKit template.
What platform do you want to use?? [ iOS / macOS ]
> iOS
What language do you want to use?? [ Swift / ObjC ]
> ObjC
Would you like to include a demo application with your library? [ Yes / No ]
> Yes
// 你应该测试你的库,测试可确保使用您库的人员的稳定性,建议使用测试框架,而不是依赖Apple的XCTest
Which testing frameworks will you use? [ Specta / Kiwi / None ]
> Specta
// 根据您所构建的内容库,你可能会发现快照基于测试是验证结果的聪明的方式
Would you like to do view based testing? [ Yes / No ]
> Yes
// 使CocoaPods生成的所有类都适合您的样式,所有从Xcode内部生成的类都将以您的前缀开头
What is your class prefix?
> XJP
Downloading dependencies
Installing Expecta (1.0.6)
Installing Specta (1.0.7)
Installing XJPCategoryKit (0.1.0)
Generating Pods project
Integrating client project
[!] Please close any current Xcode sessions and use `XJPCategoryKit.xcworkspace` for this project from now on.
Pod installation complete! There are 3 dependencies from the Podfile and 3 total pods installed.
Ace! you're ready to go!
We will start you off by opening your project in Xcode
open 'XJPCategoryKit/Example/XJPCategoryKit.xcworkspace'
创建完成后自动打开的pod
库工程目录如下。
d、拖入项目文件
将自己的工程文件拖入到项目中,需要分成两个部分,一个部分放资源文件,另外一个部分放自己的工程文件,这个架构很重要!!!
再将这些文件导入到工程中。你可能会好奇为什么在Classes
目录下的文件在工程目录中跑到了XJPCategoryKit
文件下,这个无需担心,系统会根据XJPCategoryKit.podspec
文件自动进行关联,不信你在Classes
目录下删除个文件试试,工程目录中XJPCategoryKit
文件下的该文件马上会爆红表示不存在该文件了。
提示下,这里有个小技巧,将项目文件拖入到文件夹后无需再回到工程中重复导入,而是可以直接在终端执行pod install
就可以更新工程目录了。
e、解决被墙屏蔽导致的超时问题
运行pod install
后可能会遇到被墙屏蔽导致的超时问题。
[!] CDN: trunk Repo update failed - 22 error(s):
CDN: trunk URL couldn't be downloaded: https://cdn.jsdelivr.net/cocoa/Specs/5/0/9/FBSnapshotTestCase/1.1/FBSnapshotTestCase.podspec.json Response: Timeout was reached
解决方案是进入到Podfile
文件中,添加清华大学开源软件镜像站。
source 'https://mirrors.tuna.tsinghua.edu.cn/git/CocoaPods/Specs.git'
再执行如下命令清除当前的 trunk
。
pod repo remove trunk
再次尝试 pod install
就可以了。
xiejiapei@xiejiapeis-iMac repos % pod repo remove trunk
Ignoring ffi-1.12.2 because its extensions are not built. Try: gem pristine ffi --version 1.12.2
Removing spec repo `trunk`
xiejiapei@xiejiapeis-iMac repos % cd /Users/xiejiapei/Desktop/XJPCategoryKit/Example
xiejiapei@xiejiapeis-iMac Example % pod install
Ignoring ffi-1.12.2 because its extensions are not built. Try: gem pristine ffi --version 1.12.2
Analyzing dependencies
Downloading dependencies
Installing Expecta (1.0.6)
Installing Expecta+Snapshots (3.1.1)
Installing FBSnapshotTestCase (2.1.4)
Installing Specta (1.0.7)
Installing XJPCategoryKit (0.1.0)
Generating Pods project
Integrating client project
[!] Please close any current Xcode sessions and use `XJPCategoryKit.xcworkspace` for this project from now on.
Pod installation complete! There are 5 dependencies from the Podfile and 5 total pods installed.
更新工程目录后如下。
4、配置 .podspec 文件
podfile
文件想必大家都应该很熟悉才是,它可以方便管理第三方库的使用。podspec
文件其实就是一个pod
的配置文件,里面记录这个pod
的配置信息、编译依赖、仓库地址以及版本信息等等,本地的.cocoapod
文件夹其实就是存储了每个pod
的podspec
文件(不过是JSON
格式)。每个pod install
就是去通过找podspec
的仓库地址,再去下载相应的源码。
a、创建 .podspec 文件,create 后面必须是开源库名
podspec
文件会自动生成,假如不小心删除掉了,则可以通过以下的方式进行创建。
xiejiapei@xiejiapeis-iMac desktop % cd /Users/xiejiapei/Desktop/DriverRecordSoundKit
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % pod spec create DriverRecordSoundKit
Specification created at DriverRecordSoundKit.podspec
使用Sublime Text
打开podspec
文件后阅读注释
Be sure to run `pod spec lint RecordSoundKit.podspec' to ensure this is a valid spec and to remove all comments including this before submitting the spec.
在你提交.podspec文件到CocoaPods前,务必要在命令行运行 pod spec lint RecordSoundKit.podspec 来确保.podspec有效,并且删除所有的注释,包括这句(Be sure ... submitting the spec.)
b、podspec 文件的语法说明
Pod::Spec.new do |spec| #表示用spec作为代称
...
end
开源库描述信息
#开源库的名字
spec.name = "DriverRecordSoundKit"
#开源库当前版本
spec.version = "1.0.0"
#开源库概述(打开GitHub能看到的描述)
spec.summary = "使用场景包括:司机行驶过程、会议、课程等的全程录音"
#开源库描述 (这个描述会被用来生成开源库的标签,必需写在中间一行)
spec.description = <<-DESC
1. 部分城市运管部门要求网约车需提供服务过程中录音功能
2. 与高德等聚合平台对账时,对于部分争议作弊订单,需要行程中录音证据
3. 司乘发生纠纷时,有行程中录音可方便责任判定,并且行程中录音功能的存在,可在一定程度上避免一些恶意投诉或司机不规范服务
4. 便于业务部门抽查司机服务规范情况,例如人工抽查或语音质检
5. 本录音工具类也可用于会议或者上课等场景进行全程录制
DESC
#可以是开源库的GitHub地址
spec.homepage = "https://github.com/xiejiapei-creator/DriverRecordSoundKit"
#协议
spec.license = "MIT"
#开源库作者
spec.author = { "jiapei.xie" => "[email protected]" }
#开源库作者的社交链接(此处我放的是)
spec.social_media_url = "https://www.jianshu.com/u/1ceb4a330607"
#开源库支持的平台(暂时没考虑tvOS、OSX等)
spec.platform = :iOS
#开源库最低支持
spec.ios.deployment_target = "10.0"
#支持的swift版本
spec.swift_version = "4.0"
使用到的依赖库
#是否使用静态库。如果podfile指明了use_frameworks!命令,但是pod仓库需要使用静态库则需要设置
spec.static_framework = true
#开源库GitHub的路径与tag值,GitHub路径后必须有.git,tag实际就是上面的版本
spec.source = { :git => "https://github.com/xiejiapei-creator/DriverRecordSoundKit.git", :tag => "#{spec.version}" }
#开源库头文件
spec.public_header_files = 'DriverRecordSoundKit/DriverRecordSoundKit.h'
#私有库依赖的三方库
spec.dependency 'AFNetworking', '~> 1.0'
#指定私有库文件是否是ARC,默认是true,表示所有的 source_files是arc文件
spec.requires_arc = true
#pod库使用的系统库
spec.ios.framework = 'CFNetwork'
spec.frameworks = 'QuartzCore', 'CoreData'
#libraries使用的静态库 比如libz、sqlite3.0等,如果有多个则用逗号分开
spec.ios.library = "xml2"
spec.libraries = 'xml2', 'z'
项目资源与文件路径
#自定义前缀文件,默认是true cocoapos会生成默认前缀.pch文件
spec.prefix_header_file = false
spec.prefix_header_file = 'iphone/include/prefix.pch'
#pod文件路径
#一个宗旨:库里面每个文件、资源都要有路径指向它
spec.source_files = "Classes/*.{h,m}" #表示匹配所有Classes目录下文件,不包含子目录的
spec.source_files = "Classes/**/*.{h,m}" #表示匹配所有.h、.m结尾的文件(主目录和子目录,其中**相当于省略中间层级)
spec.source_files = 'Classes/**/*.{h,m}', 'More_Classes/**/*.{h,m}'
s.source_files = "UnionPay/**/*.{h, mm}"
#公共头文件,这些头文件将暴露给用户的项目。如果不设置,所有source_files的头文件都将被暴露
#private_header_files.和public_header_files相反,指定不暴露的头文件
spec.public_header_files = 'Headers/Public/*.h'
#依赖的框架
s.dependency 'UCARLogger'
#指定三方库的路径
spec.ios.vendored_frameworks = "MyPod/Frameworks/MyFramework.framework"
spec.vendored_frameworks = 'MyFramework.framework', "TheirFramework.framework"
#三方静态库,指明具体路径
spec.ios.vendored_library = "Libraries/libProj4.a"
spec.vendored_libraries = 'libProj4.a', 'libJavaScriptCore.a'
s.ios.vendored_libraries = "UnionPay/**/*.a"
s.vendored_libraries = 'UCARFaceID/UCarLive/MGBaseKit/libMobileSDKAuth.a', 'UCARFaceID/UCarLive/FacePlusPlus/libliveness.a'
#系统的框架
s.frameworks = "CFNetwork", "SystemConfiguration"
s.framework = 'CoreMotion', 'CoreMedia', 'AVFoundation'
s.frameworks = "Foundation"
#系统的静态库,需要去掉lib前缀
s.library = 'c++', 'z'
#资源文件
s.ios.resource_bundle = { 'MapBox' => 'MapView/Map/Resources/*.png' }
s.resource_bundles = { 'XBPodSDK' => ['XBPodSDK/Assets/**']}
s.resource = "FaceID/MGLiveResource.bundle"
s.resource_bundles = { 'PushEngine' => ['PushEngine/Assets/*.plist'] }
#被排除的文件
spec.ios.exclude_files = "Classes/osx"
spec.exclude_files = 'Classes/**/unused.{h,m}'
私有库模块
#Subspecs 私有库模块
spec.subspec 'ConvertMP3' do |ss| #简单
ss.source_files = 'DriverRecordSoundKit/ConvertMP3.h' #指定子模块路径
end
#Subspecs 私有库模块
spec.subspec 'Encryption' do |ss| #有几个do就有几个end
ss.subspec 'RNCryptor' do |sss| #复杂
sss.source_files = 'DriverRecordSoundKit/Encryption/RNCryptor/*.{h,m}' #指定子模块路径
end
end
s.subspec 'BaiduComponent' do |f|
f.source_files = 'MapFundation/Classes/Baidu/**/*.{h,m}'
f.public_header_files = 'MapFundation/Classes/Baidu/**/*.h'
f.vendored_libraries = 'MapFundation/thirdlibs/*.{a}'
f.vendored_frameworks = 'MapFundation/frameworks/*.{framework}'
f.frameworks ='CoreGraphics','CoreLocation','OpenGLES','QuartzCore','Security','SystemConfiguration'
f.libraries = 'iconv','sqlite3','c++','z'
f.pod_target_xcconfig = { 'OTHER_LDFLAGS' => '-ObjC' }
f.dependency 'MapFundation/Common'
end
部署的架构
s.pod_target_xcconfig = { 'VALID_ARCHS' => 'arm64 armv7', 'OTHER_LDFLAGS' => '-force_load'}
s.user_target_xcconfig = { 'VALID_ARCHS' => 'arm64 armv7', 'OTHER_LDFLAGS' => '-force_load'}
c、模仿默认的 podspec 文件生成自己的 podspec 文件
默认的 podspec 文件
Pod::Spec.new do |s|
s.name = 'XJPCategoryKit'
s.version = '0.1.0'
s.summary = 'A short description of XJPCategoryKit.'
# This description is used to generate tags and improve search results.
# * Think: What does it do? Why did you write it? What is the focus?
# * Try to keep it short, snappy and to the point.
# * Write the description between the DESC delimiters below.
# * Finally, don't worry about the indent, CocoaPods strips it!
s.description = <<-DESC
TODO: Add long description of the pod here.
DESC
s.homepage = 'https://github.com/xiejiapei-creator/XJPCategoryKit'
# s.screenshots = 'www.example.com/screenshots_1', 'www.example.com/screenshots_2'
s.license = { :type => 'MIT', :file => 'LICENSE' }
s.author = { 'xiejiapei-creator' => '[email protected]' }
s.source = { :git => 'https://github.com/xiejiapei-creator/XJPCategoryKit.git', :tag => s.version.to_s }
# s.social_media_url = 'https://twitter.com/'
s.ios.deployment_target = '8.0'
s.source_files = 'XJPCategoryKit/Classes/**/*'
# s.resource_bundles = {
# 'XJPCategoryKit' => ['XJPCategoryKit/Assets/*.png']
# }
# s.public_header_files = 'Pod/Classes/**/*.h'
# s.frameworks = 'UIKit', 'MapKit'
# s.dependency 'AFNetworking', '~> 2.3'
end
DriverRecordSoundKit 最终的 podspec 文件
Pod::Spec.new do |s|
s.name = 'DriverRecordSoundKit'
s.version = '1.0.0'
s.summary = '使用场景包括:司机行驶过程、会议、课程等的全程录音'
s.description = <<-DESC
1. 部分城市运管部门要求网约车需提供服务过程中录音功能
2. 与高德等聚合平台对账时,对于部分争议作弊订单,需要行程中录音证据
3. 司乘发生纠纷时,有行程中录音可方便责任判定,并且行程中录音功能的存在,可在一定程度上避免一些恶意投诉或司机不规范服务
4. 便于业务部门抽查司机服务规范情况,例如人工抽查或语音质检
5. 本录音工具类也可用于会议或者上课等场景进行全程录制
DESC
s.homepage = 'https://github.com/xiejiapei-creator/DriverRecordSoundKit'
s.license = { :type => 'MIT', :file => 'LICENSE' }
s.author = { 'xiejiapei-creator' => '[email protected]' }
s.source = { :git => 'https://github.com/xiejiapei-creator/DriverRecordSoundKit.git', :tag => s.version.to_s }
s.social_media_url = "https://www.jianshu.com/u/1ceb4a330607"
s.ios.deployment_target = '10.0'
s.source_files = 'DriverRecordSoundKit/Classes/**/*'
s.public_header_files = 'DriverRecordSoundKit/Classes/UCARRecordSoundTool.h'
s.vendored_libraries = 'DriverRecordSoundKit/Classes/**/*.a'
s.frameworks = "AVFoundation", "Foundation", "Security"
s.dependency 'RNCryptor-objc'
end
XJPCategoryKit 最终的 podspec 文件
Pod::Spec.new do |s|
s.name = 'XJPCategoryKit'
s.version = '0.1.0'
s.summary = '提供便利的分类'
s.description = <<-DESC
提供OC常见的各种分类方法
DESC
s.homepage = 'https://github.com/xiejiapei-creator/XJPCategoryKit'
s.license = { :type => 'MIT', :file => 'LICENSE' }
s.author = { 'xiejiapei-creator' => '[email protected]' }
s.source = { :git => 'https://github.com/xiejiapei-creator/XJPCategoryKit.git', :tag => s.version.to_s }
s.social_media_url = "https://www.jianshu.com/u/1ceb4a330607"
s.ios.deployment_target = '8.0'
s.source_files = 'XJPCategoryKit/Classes/**/*'
s.public_header_files = 'Pod/Classes/XJPCategoryKit.h'
end
d、解决系统库文件不存在报错的问题
首先逐个查找文件,看看依赖了哪些系统的库
#import
#import
#import "UCARSingle.h"
#include //获取CPU信息所需要引入的头文件
然后将其添加到DriverRecordSoundKit.podspec
文件中
s.frameworks = "AVFoundation", "Foundation", "Security","mach"
接着就报错了说是该系统库不存在
ld: framework not found mach
clang: error: linker command failed with exit code 1 (use -v to see invocation)
所以添加到DriverRecordSoundKit.podspec
文件中之前需要先去查找下看看该系统库是否存在,如果不存在说明其已经包含在其他系统库文件中了,这里mach
库包含在了Foundation
库中。
5、更新 GitHub 里面的库
a、依次执行下面命令以关联远程分支
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git add .
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git commit -m "IOS司机端APP行驶全程录音框架"
// 添加tag标记
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git tag 1.0.0
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git push --tag
// 关联远程分支
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git remote add origin 'https://github.com/xiejiapei-creator/XJPCategoryKit'
// 得知分支名称为master
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git branch -a
* master
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git push origin master
b、验证 podspec 文件是否有效
pod spec lint DriverRecordSoundKit.podspec
如果验证不通过,可以执行pod spec lint DriverRecordSoundKit.podspec --verbose
查看详细的ERROR
和WARING
提示,根据提示依次解决相关问题。
pod spec lint DriverRecordSoundKit.podspec --verbose
修改后总是报错,清除缓存或直接删除APP重新执行就OK了,因为使用pod repo push
的使用,pod
会从缓存中复制工程,然后再编译验证。
pod cache list //查看缓存
pod cache clean --all //清除缓存
假如出现了无关痛痒的警告,可以直接忽略掉
- NOTE | [iOS] xcodebuild: warning: Skipping code signing because the target does not have an Info.plist file and one is not being generated automatically. (in target 'App' from project 'App')
- WARN | [iOS] xcodebuild: XJPCategoryKit/XJPCategoryKit/Classes/NSDate+Custom.m:70:67: warning: values of type 'NSInteger' should not be used as format arguments; add an explicit cast to 'long' instead [-Wformat]
xiejiapei@xiejiapeis-iMac XJPCategoryKit % pod spec lint XJPCategoryKit.podspec --allow-warnings
最后,终于出现了梦寐以求的成功符号!
Analyzed 1 podspec.
XJPCategoryKit.podspec passed validation.
c、更新版本之前要先删除远程库的 tag 和本地的 tag
// 删除本地tag
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git tag -d 1.0.0
Deleted tag '1.0.0' (was 9a60dca)
// 删除远程库tag
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % git push origin :refs/tags/1.0.0
否则会被拒绝更新
[!] Unable to accept duplicate entry for: XJPCategoryKit (0.1.0)
删除后更新tag
再提交就ok了
Downloading dependencies
Installing XJPCategoryKit 1.0.0 (was 0.1.0)
6、提交 .podspec 文件到 CocoaPods
如果是第一次提交,需要先执行下面的命令去CocoaPods
注册(填写邮箱,收取验证链接,点击链接,就OK)。
xiejiapei@xiejiapeis-iMac XJPCategoryKit % pod trunk register [email protected] '谢佳培' --description='分类方法的库'
Ignoring ffi-1.12.2 because its extensions are not built. Try: gem pristine ffi --version 1.12.2
[!] Please verify the session by clicking the link in the verification email that has been sent to [email protected]
然后再执行以下命令。
pod trunk push DriverRecordSoundKit.podspec
[!] The spec did not pass validation, due to 3 warnings (but you can use `--allow-warnings` to ignore them).
如果出现警告的话则使用忽略警告的语法,执行然后大功告成!!!
xiejiapei@xiejiapeis-iMac XJPCategoryKit % pod trunk push XJPCategoryKit.podspec --allow-warnings
Ignoring ffi-1.12.2 because its extensions are not built. Try: gem pristine ffi --version 1.12.2
Updating spec repo `trunk`
CocoaPods 1.10.0 is available.
To update use: `gem install cocoapods`
For more information, see https://blog.cocoapods.org and the CHANGELOG for this version at https://github.com/CocoaPods/CocoaPods/releases/tag/1.10.0
Validating podspec
Congrats
XJPCategoryKit (0.1.0) successfully published
November 18th, 21:18
https://cocoapods.org/pods/XJPCategoryKit
Tell your friends!
这时进入到远程仓库,发现没有显示我们提交的文件,而是有个警告框让我们审查一下提交的内容,究其原因是我们在远程仓库中出现了两个分支,一个main
分支,另外一个是master
分支,而我们目前正处于main
分支上,所以显示的内容还是之前的。
为了方便地解决上面提到的问题,我们需要使用Sourcetree
工具来管理远程仓库,下面是将远程仓库的url
克隆到Sourcetree
工具里。
在Sourcetree
工具上将项目从main
分支切换到master
分支。在远程仓库中不好进行这个操作,而在Sourcetree
工具中切换和合并分支的操作却很容易进行。
接下来在Sourcetree
工具上删除掉远程仓库中无用的main
分支,结果发现出错了。
这是因为我们的远程仓库默认的分支仍然是main
分支,拒绝删除,所以需要到远程仓库中修改默认分支,修改完毕之后,不要忘记点击旁边的 update
按钮,因为update
一下才能生效。
之后就成功删除掉main
分支了,再回到github
的仓库首页看下,发现我们提交的库出来了,感到到哭,我做的第一个库完成了,一路坎坷都是泪。
在Github
上搜索我们的框架发现也能够成功查找到了哈哈。
7、使用创建的库
a、安装框架
xiejiapei@xiejiapeis-iMac Example % pod install
Integrating client project
Pod installation complete! There are 3 dependencies from the Podfile and 3 total pods installed.
b、导入框架
#import
XJPCategoryKit.h
文件为提供给外界的公共入口文件,其包括了以下头文件。
#import
#import
//! Project version number for XJPCategoryKit.
FOUNDATION_EXPORT double XJPCategoryKitVersionNumber;
//! Project version string for XJPCategoryKit.
FOUNDATION_EXPORT const unsigned char XJPCategoryKitVersionString[];
#import "NSDate+Custom.h"
#import "NSObject+Custom.h"
#import "NSString+Custom.h"
#import "UIColor+Custom.h"
#import "UIImage+Custom.h"
#import "UILabel+Custom.h"
#import "UIScreen+Custom.h"
#import "UITextField+Custom.h"
#import "UIView+Custom.h"
#import "UIViewController+Custom.h"
c、使用框架
- (void)viewDidLoad
{
[super viewDidLoad];
// 判断一个字符串是否都是纯数字
NSString *testString = @"123456";
if ([testString judgeIsPureInt])
{
NSLog(@"该字符串为纯数字,说明生成的库可用");
}
}
输出结果为:
2020-11-19 15:08:54.798484+0800 XJPCategoryKit_Example[26855:668614] 该字符串为纯数字,说明生成的库可用
搞定!!!
8、创建过程遇到的问题
a、pattern did not match any file
- NOTE | [iOS] xcodebuild: DriverRecordSoundKit/DriverRecordSoundKit/RecordSoundKit.h:18:9: fatal error: 'lame.h' file not found
- NOTE | [iOS] xcodebuild: DriverRecordSoundKit/DriverRecordSoundKit/AudioTool.m:11:9: fatal error: 'RNCryptor iOS.h' file not found
- ERROR | [iOS] [DriverRecordSoundKit/SupportingFiles] file patterns: The `source_files` pattern did not match any file.
- ERROR | [iOS] file patterns: The `resources` pattern did not match any file.
- ERROR | [iOS] file patterns: The `vendored_libraries` pattern did not match any file.
因为路径设置有误所以找不到文件。经我试验后发现最有效的解决方案就是设置模糊匹配:spec.source_files = 'DriverRecordSoundKit/**/*.{h,m}'
。你真要一个个去匹配,你会发现不仅麻烦而且有时候即使你认为路径设置正确了,它也找不到文件.....
b、Prefix.pch
- ERROR | [iOS] unknown: Encountered an unknown error (No such file or directory @ rb_sysopen - /private/var/folders/2j/53b9tdz551l0x8mjtg3zhv4r0000gn/T/CocoaPods-Lint-20201117-15842-kud7vb-DriverRecordSoundKit/Pods/DriverRecordSoundKit/DriverRecordSoundKit/RNCryptor-Prefix.pch) during validation.
目前得出的结论,pch
在pod
库中,完全就是绊脚石般的存在。个人亲测,无论你怎么玩,只要pch
存在,就会报这个错误。查看了很多第三方库的pod
文件,发现别人压根不使用pch
这玩意。
c、missing required architecture i386
- NOTE | [iOS] xcodebuild: ld: warning: ignoring file DriverRecordSoundKit/DriverRecordSoundKit/ConvertMP3/libmp3lame.a, missing required architecture i386 in file DriverRecordSoundKit/DriverRecordSoundKit/ConvertMP3/libmp3lame.a (4 slices)
这个就太坑了。眼看离胜利只剩最后一步,你给我报个啥玩意?遍查无解。分析这个错误,缺少了i386
架构的东西。查看一下这个框架支持哪些架构,发现确实没有 i386。
xiejiapei@xiejiapeis-iMac DriverRecordSoundKit % lipo -info /Users/xiejiapei/Desktop/DriverRecordSoundKit/DriverRecordSoundKit/SupportingFiles/libmp3lame.a
Architectures in the fat file: /Users/xiejiapei/Desktop/DriverRecordSoundKit/DriverRecordSoundKit/SupportingFiles/libmp3lame.a are: armv7 armv7s x86_64 arm64
查看这个框架的源文件,发现做了个版本号的判断,低版本生成i386,高版本生成x86_64,所以这个答案是无解的。
PLATFORM="iPhoneSimulator"
if [ "$ARCH" = "x86_64" ]
then
SIMULATOR="-mios-simulator-version-min=7.0"
HOST=x86_64-apple-darwin
else
SIMULATOR="-mios-simulator-version-min=5.0"
HOST=i386-apple-darwin
fi
在网上找到一份支持i386的框架尝试了一下,发现的确出现了 i386 ,但是带入到项目后发现也是无效的,仍然报了之前的错误。不过它只是个警告,就不去管它了,之前以为是它影响到了linker command failed
才煞费苦心。
Architectures in the fat file: /Users/xiejiapei/Downloads/libmp3lame/libmp3lame.a are: armv6 armv7 i386 armv7s arm64
d、Info.plist
- NOTE | [iOS] xcodebuild: warning: Skipping code signing because the target does not have an Info.plist file and one is not being generated automatically. (in target 'App' from project 'App')
plist
可以作为资源文件保存下来。
spec.resources = "DriverRecordSoundKit/**/*.{plist,modulemap}"
使用方式
NSString *filePath = [[NSBundle bundleForClass:[DriverRecord class]] pathForResource:@"Themes" ofType:@"plist"];
NSDictionary *dictionary = [NSDictionary dictionaryWithContentsOfFile:filePath];
e、linker command failed
- ERROR | [iOS] xcodebuild: Returned an unsuccessful exit code. You can use `--verbose` for more information.
- NOTE | [iOS] xcodebuild: clang: error: linker command failed with exit code 1 (use -v to see invocation)
[!] The spec did not pass validation, due to 1 error.
我遇到的玩命的错误,真的解答不出来。不过有几个心得。
1、.a
文件的存放位置需要放到Classes
文件夹下,放到其他地方不生效。
2、.a
文件放入后需要通过pod install
安装到项目中,然后再添加到此处
3、需要添加到DriverRecordSoundKit.podspec
文件中,语法比较诡异,你按照目录添加,它却说找不到,还是用模糊匹配吧。添加单个文件的话用的是s.ios.vendored_library
,而不是复数那个语法。
s.ios.vendored_library = 'DriverRecordSoundKit/Classes/**/*.a'
需要把项目中遇到的各种系统库和第三方库全部分离出来进行添加,否则就会出现链接出错的问题。
s.vendored_libraries = 'DriverRecordSoundKit/Classes/**/*.a'
s.frameworks = "AVFoundation", "Foundation", "Security"
s.dependency 'RNCryptor-objc'
四、python3.7
1、安装 python3.7
- 网上下载
Python 3.7
Mac版的安装包,下载后一步步点击下一步进行安装。 - 在终端输入
open ~/.bash_profile
,在文件中添加以下语句配置环境。
export PATH=$PATH:/Library/Frameworks/Python.framework/Versions/3.7/bin/pip3.7
alias pip="/Library/Frameworks/Python.framework/Versions/3.7/bin/pip3.7"
alias python="/Library/Frameworks/Python.framework/Versions/3.7/bin/python3.7"
- 点击
save
进行保存,然后关闭弹出的文件。 - 在终端继续输入
source .bash_profile
加载刚才修改过的环境文件。 - 输入
python
查看是否安装成功以及版本号。
Python 3.7.9 (v3.7.9:13c94747c7, Aug 15 2020, 01:31:08)
Type "help", "copyright", "credits" or "license" for more information.
>>>
- 输入
exit()
退出版本号查询。 - 输入
help()
查询帮助命令,然后在其中输入module
查询已经安装好的模块,通过这个方式可以帮我们排除模块尚未安装的错误。当安装成功后还显示未安装,进入module
查询后发现已经存在了,则重启终端即可。
2、使用 python3.7
//运行文件
python3 mainServer.py
//安装缺失模块
pip3 install flask
3、Mac上Python安装request库老是失败的问题
记录折腾安装request库问题,最终原因是mac上python
可以安装多个版本的python
,vs code
上使用的默认是最新的,输入python
用的是系统默认安装的。
在学习python
时,import requests
时报错:
ModuleNotFoundError: No module named 'requests’
那我们去安装requests
,网上的文档都是pip3 install requests
。输入命令后,一直提示:
Defaulting to user installation because normal site-packages is not writeable
Requirement already satisfied: requests in ./Library/Python/3.7/lib/python/site-packages (2.27.1)
Requirement already satisfied: urllib3<1.27,>=1.21.1 in /Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages (from requests) (1.26.9)
Requirement already satisfied: idna<4,>=2.5 in /Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages (from requests) (3.3)
Requirement already satisfied: certifi>=2017.4.17 in /Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages (from requests) (2021.10.8)
Requirement already satisfied: charset-normalizer~=2.0.0 in /Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages (from requests) (2.0.12)
提示已经安装了,可是已经安装了为什么import
的时候一直提示失败呢。
检查是否已经安装成功request
输入pip3 list
,确认也有request
。那这到底什么问题呢?是不是和系统的默认安装python
冲突了呢,又想着把系统默认的python2.7
改为我们用brew
新装的python
。
将homeBrew安装的python3.9设置为默认版本
brew 安装
默认安装了最新版本,最后会在summary
中给出安装的地址,比如我的3.9安装在了:/usr/local/xiejiapei/[email protected]/3.9.10
。
brew install python
打开配置文件并写入python的外部环境变量
open ~/.bash_profile
配置文件中添加
export PATH=${PATH}:/usr/local/xiejiapei/[email protected]/3.9.10/bin
在终端将python重命名
alias python="/usr/local/xiejiapei/[email protected]/3.9.10/bin/3.9”
最后让配置文件生效
source ~/.bash_profile
现在直接用python
命令,可以看到已经使用我们安装的3.9了。可是忙活一通发现request
还是没有成功。
最后想想是不是pip3和python的路径不一致
输入下面命令,原来有这么多默默的有这么多python
版本。其实最终的解决方法是:先看看我们python
的位置以及pip3
的位置,终端输入命令:which python
。如果你安装的3.9,没有设为默认应该使用which python3.9
。
这才发现这些版本的路径问题
➜ ~ which python
python: aliased to /usr/local/xiejiapei/[email protected]/3.9.10/bin/python3.9
➜ ~ which python3
/Library/Frameworks/Python.framework/Versions/3.7/bin/python3
➜ ~ which pip3
/Library/Frameworks/Python.framework/Versions/3.7/bin/pip3
➜ ~ which pip3.9
/usr/local/bin/pip3.9
我们brew
安装的是3.9,但默认的python3
实际对应的是python3.7
,我们用pip3
安装一直提示已安装,但一直import
失败,是因为我们安装的request
和使用import
不是同一个版本,不在一个路径。所以我们在使用brew
安装python
之后,查brew list
查看我们安装的python
的版本,然后使用对应的pip
版本来安装request
,才会安装在同一路径里。这样在默认python
版本中import request
库才不会提示错误。比如:默认使用的版本是3.9,即python3.9
,那就使用pip3.9 install requests
来安装。