我的 Android 应用程序使用了两个 API:Google Maps API v2(由包标识com.google.android.gms.maps
)和 Facebook API(com.facebook
) (感谢@RINK,最后一个已解决)
这两项服务都是按照他们的官方教程安装在我的项目中的,直到昨天它们都运行良好,我的代码中的某些内容发生了变化,我显然无法弄清楚是什么。我很确定原因与在我的测试手机上安装之前签署我的应用程序的调试/发布密钥有关。
特别是,我正在使用它默认存储在我的个人文件夹(我在 Windows + Netbeans 上)中的调试密钥,并且每次我删除它时都会重新生成它:C:\Users\Alessandro\.android\debug.keystore
.
Maps 和 Facebook 都需要知道此类密钥的 SHA-1 指纹,我已经使用keytool
. 有了它,我把它放在专用的 API 服务页面上:
- 表单中的 Google 控制台 API
BB:0D:AC:74:D3:21:E1:43:67:71:9B:62:91:AF:A1:66:6E:44:5D:75;com.UpMap
- 和 Facebook 的 App Dev 页面,Key Hashes 字段。
正如我所说,一切正常,但我的应用程序中的地图突然变为空白,Facebook 登录引发错误:
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue com.facebook.http.protocol.ApiException: remote_app_id does not match stored id
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.http.protocol.ApiResponseChecker.b(ApiResponseChecker.java:74)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.http.protocol.ApiResponseChecker.a(ApiResponseChecker.java:103)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.http.protocol.ApiResponse.h(ApiResponse.java:208)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.katana.server.protocol.AuthorizeAppMethod.a(AuthorizeAppMethod.java:266)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.katana.server.protocol.AuthorizeAppMethod.a(AuthorizeAppMethod.java:27)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.http.protocol.SingleMethodRunnerImpl.a(SingleMethodRunnerImpl.java:141)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.http.protocol.AbstractSingleMethodRunner.a(AbstractSingleMethodRunner.java:16)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.katana.server.handler.PlatformOperationHandler.c(PlatformOperationHandler.java:284)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.katana.server.handler.PlatformOperationHandler.a(PlatformOperationHandler.java:185)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.fbservice.service.BlueServiceQueue.d(BlueServiceQueue.java:245)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.fbservice.service.BlueServiceQueue.d(BlueServiceQueue.java:51)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.fbservice.service.BlueServiceQueue$3.run(BlueServiceQueue.java:191)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:441)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at java.util.concurrent.FutureTask.run(FutureTask.java:137)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at com.facebook.common.executors.HandlerExecutorServiceImpl$ListenableScheduledFuture.run(HandlerExecutorServiceImpl.java:268)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at android.os.Handler.handleCallback(Handler.java:587)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at android.os.Handler.dispatchMessage(Handler.java:92)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at android.os.Looper.loop(Looper.java:123)
11:24:51.064 1708 #1708 WARN fb4a(:<default>):BlueServiceQueue at android.os.HandlerThread.run(HandlerThread.java:60)
我在登录时阅读了 Android Facebook SDK 3.0 给出的“remote_app_id 与存储的 id 不匹配”的问题,我应该遇到同样的问题,但事实是我使用 keytool 生成的“密钥哈希”很好。
我试图:
- 删除存储的调试密钥并重新生成它,更新它的 SHA 并将其保存在 Google 和 Facebook 上
- 检查 Google 和 Facebook 提供的并写在我的 AndroidManifest.xml 中的 API Keys 是否正确
- 检查我的应用程序的包名是否正确
- 清理并重建整个项目,包括 Google 和 Facebook 源库
这是我的清单:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.UpMap"
android:versionCode="1"
android:versionName="1.0">
<uses-sdk android:minSdkVersion="8" android:targetSdkVersion="18" />
<permission android:name="com.UpMap.permission.MAPS_RECEIVE" android:protectionLevel="signature"/>
<uses-permission android:name="com.UpMap.permission.MAPS_RECEIVE"/>
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="com.google.android.providers.gsf.permission.READ_GSERVICES"/>
<uses-feature android:glEsVersion="0x00020000" android:required="true"/>
<application android:label="@string/app_name" android:allowBackup="true">
<uses-library android:name="com.google.android.maps" />
<activity
android:name="MainActivity"
android:label="@string/app_name"
android:theme="@style/Theme.AppCompat"
android:screenOrientation="portrait">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name="LevelActivity"
android:label="@string/app_name"
android:theme="@style/Theme.AppCompat"
android:windowSoftInputMode="adjustPan|adjustResize"
android:screenOrientation="portrait"
uiOptions="splitActionBarWhenNarrow">
<meta-data
android:name="android.support.UI_OPTIONS"
android:value="splitActionBarWhenNarrow" />
</activity>
<activity
android:name="FacebookActivity"
android:label="@string/app_name"
android:theme="@style/Theme.AppCompat"
android:windowSoftInputMode="adjustPan|adjustResize"
android:screenOrientation="portrait">
</activity>
<activity android:name="com.facebook.LoginActivity"/>
<meta-data
android:name="com.google.android.maps.v2.API_KEY"
android:value="<HIDDEN_GOOGLE_KEY>"
/>
<meta-data
android:name="com.facebook.sdk.ApplicationId"
android:value="@string/fb_app_id"/>
</application>
</manifest>
这是显示 Google 地图的 Activity 代码:
<?xml version="1.0" encoding="UTF-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/LevelLayout">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/where_is_it"
android:textSize="28sp"
android:layout_margin="5dp"
android:layout_marginBottom="0dp"
/>
<LinearLayout
android:id="@+id/SolutionLayout"
android:orientation="horizontal"
android:layout_width="match_parent"
android:layout_height="40dp"
android:layout_margin="5dp"
android:background="#000000"
android:gravity="center"
>
</LinearLayout>
<FrameLayout
android:layout_width="match_parent"
android:layout_height="300dp"
>
<fragment
class="com.google.android.gms.maps.SupportMapFragment"
xmlns:map="http://schemas.android.com/apk/res-auto"
android:id="@+id/TheMap"
android:layout_width="match_parent"
android:layout_height="match_parent"
map:mapType="satellite"
map:uiRotateGestures="false"
map:uiScrollGestures="false"
map:uiTiltGestures="false"
map:uiZoomControls="false"
map:uiZoomGestures="false"
/>
<LinearLayout
android:id="@+id/MapOverlay"
android:clickable = "true"
android:onClick="mapOverlayClick"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingTop="80dp"
android:paddingBottom="80dp"
android:paddingLeft="20dp"
android:paddingRight="20dp"
android:background="#00000000"
>
<TextView
android:id="@+id/HintView"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:textSize="15sp"
android:background="#AA000000"
android:padding="10dp"
android:gravity="center"
android:visibility="invisible"
/>
</LinearLayout>
</FrameLayout>
</LinearLayout>
我使用这一行来检索代码中的 GoogleMap 对象:
GoogleMap theMap = ((SupportMapFragment) getSupportFragmentManager().findFragmentById(R.id.TheMap)).getMap();
我无法弄清楚问题是什么,也许我认为我正在使用上面指定的调试密钥对应用程序进行签名......但实际上真正使用的密钥在另一条路径中?