最近在做公司项目,需要在cordova 打包的项目里加入微信支付功能。而不是使用通过新建single application的项目里开发。
使用从微信开放平台下载来的开发包,按照普通的新建single application项目的方式加了在targets下的Link binary with libraries 里加上libsqlite3.0.dylib, libz.dylib , SystemConfiguration.framework CoreGraphics.framework Foundation.framework UIKit.framework,还有对于 c++需要的 libc++.dylib。这种方法在普通新建single application项目里是完全通过的。可是就是在cordova 打包的项目在仍然还有14个错。。。。
报错如下:
Undefined symbols for architecture x86_64:
"_SecItemAdd", referenced from:
+[MTAWXOKeyChain save:data:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_SecItemCopyMatching", referenced from:
+[MTAWXOKeyChain load:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_SecItemDelete", referenced from:
+[MTAWXOKeyChain save:data:] in libWeChatSDK.a(MTAWXOKeyChain.o)
+[MTAWXOKeyChain delete:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecAttrAccessible", referenced from:
+[MTAWXOKeyChain getKeychainQuery:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecAttrAccessibleAfterFirstUnlock", referenced from:
+[MTAWXOKeyChain getKeychainQuery:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecAttrAccount", referenced from:
+[MTAWXOKeyChain getKeychainQuery:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecAttrService", referenced from:
+[MTAWXOKeyChain getKeychainQuery:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecClass", referenced from:
+[MTAWXOKeyChain getKeychainQuery:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecClassGenericPassword", referenced from:
+[MTAWXOKeyChain getKeychainQuery:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecMatchLimit", referenced from:
+[MTAWXOKeyChain load:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecMatchLimitOne", referenced from:
+[MTAWXOKeyChain load:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecReturnData", referenced from:
+[MTAWXOKeyChain load:] in libWeChatSDK.a(MTAWXOKeyChain.o)
"_kSecValueData", referenced from:
+[MTAWXOKeyChain save:data:] in libWeChatSDK.a(MTAWXOKeyChain.o)
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)
这里普及下编译链接出错:
每次遇到这种错误就头痛,害死N多脑细胞了,就在这里做个总结吧
可能会遇到这几种错误:
Undefined symbols for architecture armv7
Undefined symbols for architecture armv7s
Undefined symbols for architecture arm64
Undefined symbols for architecture i386
Undefined symbols for architecture x86_64
错误原因分析
1、大部分情况下是忘记添加了某个系统framework或dylib吧,比如你在项目中使用了sqlite3,但是没有添加libsqlite3.dylib,就会出现这个问题。解决办法是增加对应的framework或dylib。
2、如果是在C++里调用C函数,检查是否有添加extern "C",这可以通过观察错误提示中的函数名形式来决定,如果是C函数而以C++的方式调用就需要添加extern "C"。
3、
如果是把其它工程的xcodeproj文件加入到当前项目中,检查Build Phases中的Target
Dependencies有没有添加依赖,以及General中的Linked Frameworks and
Libraries有没有添加相关的.a文件。
4、如果添加.a文件编译无错而添加xcodeproj文件编译出错可参考3
5、如果添
加.a文件编译出错,首先检查其对应的头文件是否添加正确,或者在Build Setting中有没有添加对应的Header Search
Path路径;其次检查.a文件的c++编译选项与当前项目的c++编译选项是否一致;最后检查.a文件与当前项目的CPU架构信息是否一致
6、如果是extern变量报这个错误,要检查extern变量有没有在其它地方声明,如果没有则加上;如果外部变量在静态库中,可根据5检查引用头文件或头文件搜索路径是否正确;如果头文件无问题,就需要检查静态库与与当前项目的CPU架构信息是否一致
7、
如果是使用了静态库,真机Debug测试时正常,而在执行for iOS
Device测试时报这个错误,很可能是因为静态库支持的架构不全。出现这种情况是Build Setting中的Build Active
Architecture Only在Debug下设为Yes,从而使得真机Debug测试正常。
检查静态库的CPU架构支持命令:
lipo -info xxxxx.a
找出不支持arm64的静态库
find . -name *.a -exec lipo -info "{}" \;
在@end处提示expected }错误解决
问
题很明显是}匹配出现问题了。如果代码少很好找,如果代码很多怎么缩小查找范围呢?一般出现这个问题时伴随另一个错误:在某个函数定义处提示"Use
of undeclared identifire
‘someMethod‘”,那么就是在此函数定义之前的地方少了一个},导致编译器误把函数定义当成函数调用了。
查看微信提供的开发包,以上各方面对应可能出错地方都排查,包也确实支持64位的。捉摸了一天,把网上所有资料基本翻了个遍,无果。。。。。。
最后,在无望的情况下各种乱试,看到这个:http://stackoverflow.com/questions/11013997/strange-errors-in-apples-reachability-files
在里面看到如下有人写到这么一条:
Add these frameworks: SystemConfiguration.framework Security.framework CFNetwork.framework
然后我抱着试一试,也没啥指望的情况下,弄了下,结果,神奇的编译过来。。。
总结 :在 cordova项目里还需要额外添加Security.framework这个库!