我正在尝试通过Android手机安装可穿戴应用程序,使用'使用Android Studio方法'找到here,但它无效。 apk永远不会安装到可穿戴设备上。这是logcat输出:
07-28 15:11:54.107 766-820/? W/PackageManager﹕ Unknown permission com.google.android.wearable.READ_SETTINGS in package com.google.android.gms
07-28 15:11:54.117 766-820/? W/PackageManager﹕ Not granting permission com.google.android.gm.permission.AUTO_SEND to package com.google.android.wearable.app (protectionLevel=2 flags=0x88be44)
07-28 15:11:54.117 766-820/? W/PackageManager﹕ Not granting permission android.permission.MEDIA_CONTENT_CONTROL to package com.google.android.wearable.app (protectionLevel=18 flags=0x88be44)
07-28 15:11:55.047 632-632/? D/WearablePkgInstaller﹕ Got PackageUpdateReceiver message Intent { act=android.intent.action.PACKAGE_REMOVED dat=package:my.package.name flg=0x4000010 cmp=com.google.android.wearable.app/com.google.android.clockwork.companion.packagemanager.PackageUpdateReceiver (has extras) }
07-28 15:11:55.177 632-632/? D/WearablePkgInstaller﹕ Got PackageUpdateReceiver message Intent { act=android.intent.action.PACKAGE_ADDED dat=package:my.package.name flg=0x4000010 cmp=com.google.android.wearable.app/com.google.android.clockwork.companion.packagemanager.PackageUpdateReceiver (has extras) }
作为旁注,我可以手动打包(也在上面的链接中描述),当我在手机上运行时,apk会安装在可穿戴设备上。我正在使用buildToolsVersion 20.0.0,我正在运行Android Studio 0.8.2并且在我的手机模块的build.gradle中有这一行:
wearApp project(':wearable')
我已经没有关于如何调试它的想法,日志似乎没用。有什么想法吗?
编辑:发布手机和可穿戴模块的Manifest和build.gradle的相关部分。
手机清单
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="my.package.name"
android:installLocation="auto"
android:versionCode="259"
android:versionName="4.6.1">
<!-- =========== -->
<!-- PERMISSIONS -->
<!-- =========== -->
<permission
android:name="my.app.name.permission.C2D_MESSAGE"
android:protectionLevel="signature"/>
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.WAKE_LOCK"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<!-- ====================== -->
<!-- APPLICATION PROPERTIES -->
<!-- ====================== -->
<uses-sdk
android:minSdkVersion="9"
android:targetSdkVersion="19" />
<supports-screens
android:anyDensity="true"
android:largeScreens="true"
android:normalScreens="true"
android:smallScreens="true"
android:xlargeScreens="true"/>
<application
android:name=".AppState"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/Theme.MyApp"
android:hardwareAccelerated="true">
<!-- ==================== -->
<!-- GOOGLE PLAY SERVICES -->
<!-- ==================== -->
<meta-data
android:name="com.google.android.gms.version"
android:value="@integer/google_play_services_version"/>
<!-- ================== -->
<!-- Android Wear -->
<!-- ================== -->
<service
android:name=".wear.DataLayerListenerService" >
<intent-filter>
<action android:name="com.google.android.gms.wearable.BIND_LISTENER" />
</intent-filter>
</service>
<!-- This is used for manual packaging I have this commented out when -->
<!-- packaging with Android Studio-->
<!--<meta-data android:name="com.google.android.wearable.beta.app"-->
<!--android:resource="@xml/wearable_app_desc"/>-->
...The rest of the activities
</application>
</manifest>
手机build.gradle
buildscript {
repositories {
maven { url 'http://download.crashlytics.com/maven' }
}
dependencies {
}
}
apply plugin: 'com.android.application'
apply plugin: 'crashlytics'
apply plugin: 'newrelic'
repositories {
mavenCentral()
maven { url 'http://download.crashlytics.com/maven' }
}
dependencies {
compile fileTree(dir: 'libs', include: '*.jar')
compile 'com.android.support:support-v4:20.0.+'
compile 'com.android.support:appcompat-v7:20.0.+'
compile project(':facebook')
compile project(':mopub-sdk')
compile project(':GooglePlay')
compile 'com.newrelic.agent.android:android-agent:3.+'
compile 'com.crashlytics.android:crashlytics:1.+'
androidTestCompile fileTree(dir: 'tests/libs', include: '*.jar')
wearApp project(':wearable')
}
android {
compileSdkVersion 19
buildToolsVersion '20.0.0'
//Build type is debug to avoid conflict with Proguard
testBuildType = "debug"
defaultConfig {
testApplicationId "my.package.name.test"
testInstrumentationRunner "com.zutubi.android.junitreport.JUnitReportTestRunner"
}
lintOptions {
// We do not want to abort the build due to lint errors
abortOnError false
}
sourceSets {
// Main is the default unless stated otherwise
main {
manifest.srcFile 'AndroidManifest.xml'
java.srcDirs = ['src']
resources.srcDirs = ['src']
aidl.srcDirs = ['src']
renderscript.srcDirs = ['src']
res.srcDirs = ['res']
assets.srcDirs = ['assets']
}
// Testing
androidTest.setRoot('tests')
androidTest {
java.srcDirs = ['tests/src']
res.srcDirs = ['tests/res']
}
// Cannot add beta icons in here because custom flavour source sets are created
// during compilation and name duplication will result in a crash
}
signingConfigs {
debug {
storeFile file("**")
storePassword "***"
keyAlias "***"
keyPassword "***"
}
release {
storeFile file("***")
storePassword "***"
keyAlias "***"
keyPassword "***"
}
}
buildTypes {
// Development configuration
debug {
debuggable true
jniDebugBuild true
signingConfig signingConfigs.debug
runProguard false
}
// Release configuration
release {
debuggable false
jniDebugBuild false
signingConfig signingConfigs.release
// COMMENTED PROGUARD OUT FOR NOW TO SEE IF IT WILL HELP
// // Configure ProGuard
// runProguard true
// // General configuration
// proguardFile 'proguard/proguard.cfg'
// // Add all of our component-specific configurations (excluding the Android generic, as we want it to be last)
// FileTree tree = fileTree(dir: 'proguard', include: '*.txt', exclude: 'Android.txt')
// tree.each {File file ->
// proguardFile file.getCanonicalPath()
// }
// // Add a fallback configuration for all Android apps
// proguardFile 'proguard/Android.txt'
}
// Release configuration, but debuggable and without ProGuard
// Used for testing features like G+ and in-app billing where a release config is required
staging {
debuggable true
jniDebugBuild true
signingConfig signingConfigs.release
runProguard false
}
}
productFlavors {
production {
applicationId "my.package.name"
}
internalBeta {
applicationId "my.internalbetapackage.name"
// Beta icons
sourceSets.internalBeta.res.srcDirs = ['res-beta/internal']
}
externalBeta {
applicationId "my.externalbetapackage.name"
// Beta icons
sourceSets.externalBeta.res.srcDirs = ['res-beta/external']
}
testing{
applicationId "my.package.name"
}
}
// Without this, gradle will complain that duplicate files were added to the APK, see:
// http://stackoverflow.com/questions/20673888/duplicate-files-copied-android-studio-0-4-0
packagingOptions {
exclude 'META-INF/LICENSE.txt' // twitter4j
exclude 'META-INF/ASL2.0' // jackson
exclude 'META-INF/LICENSE' // jackson
exclude 'META-INF/NOTICE' // jackson
}
}
task makeTestApks {
dependsOn "assembleProductionRelease"
dependsOn "assembleProductionTest"
}
可穿戴式舱单
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="my.package.name">
<uses-feature android:name="android.hardware.type.watch" android:required="false"/>
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@android:style/Theme.DeviceDefault" >
<meta-data
android:name="com.google.android.gms.version"
android:value="@integer/google_play_services_version" />
<activity
android:name=".WearReaderActivity"
android:label="MyApp" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
<intent-filter>
<action android:name="my.package.name.READ"/>
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<service
android:name=".wear.DataLayerListenerService" >
<intent-filter>
<action android:name="com.google.android.gms.wearable.BIND_LISTENER" />
</intent-filter>
</service>
</application>
</manifest>
可穿戴build.gradle
repositories {
mavenCentral()
}
apply plugin: 'com.android.application'
apply plugin: 'newrelic'
android {
compileSdkVersion 19
buildToolsVersion '20.0.0'
defaultConfig {
minSdkVersion 19
targetSdkVersion 19
versionCode 259
versionName "4.6.1"
}
signingConfigs {
debug {
keyAlias '***'
keyPassword '***'
storeFile file('sameAsHandset/debug.keystore')
storePassword '***'
}
release {
storeFile file("sameAsHandset/android.keystore")
storePassword "***"
keyAlias "***"
keyPassword "***"
}
}
buildTypes {
release {
debuggable true
jniDebugBuild false
signingConfig signingConfigs.release
}
}
productFlavors {
production {
applicationId "my.package.name"
}
internalBeta {
applicationId "my.internalBetaPackage.name"
// Beta icons
sourceSets.internalBeta.res.srcDirs = ['res-beta/internal']
}
externalBeta {
applicationId "my.externalBetaPackage.name"
// Beta icons
sourceSets.externalBeta.res.srcDirs = ['res-beta/external']
}
testing{
applicationId "my.package.name"
}
}
}
dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
compile project(':GooglePlay')
compile 'com.newrelic.agent.android:android-agent:3.+'
}
答案 0 :(得分:24)
要特别回答@odiggity,在手机应用的 build.gradle 文件中,您应该提及磨损应用文件夹的确切名称。如果您在 Android Studio 中创建了项目,那么您的build.gradle应如下所示:
wearApp project(':wear')
由于以下原因,可能会发生这种情况:
"permissions"
不相同(Wear App权限应该是移动应用权限的子集)。开发者网站上的任何地方都没有记录,但我发现这是我个人的经验。我可以想到这个限制背后的原因是,谷歌希望防止偷偷摸摸的开发者利用用户&amp;他们的隐私。
Package name
穿着&amp;移动应用程序不匹配。VersionNumber
&amp; Wear&amp ;; VersionName
移动应用程序不匹配Application ID
(build.gradle文件)穿app不匹配。release
版本signed
使用相同的密钥test only builds
不允许在佩戴时安装(returnCode -15,请参阅this information)如果您使用Eclipse进行开发,请确保关闭&#34;资产压缩&#34; ,否则关闭&#34; .apk&#34;原始文件夹中的文件将获得double compressed
&amp; phone won't be able to recognize
如果移动应用程序是否包装磨损应用程序。
最佳解决方案:
使用Android Studio。
<强>调试强>
如果您没有在可穿戴设备上找到磨损应用程序,您可以随时查看两个设备的日志以查看正在发生的事情。您可以在WearablePkgInstaller
上进行过滤,以查找与可穿戴软件包安装相关的所有日志记录。
从您设备上的Wear OS应用程序中触发&#34;重新同步应用程序&#34;选项来自&#34;高级设置&#34;并检查日志。仅为您的应用同步可穿戴设备的替代方法是重新安装您的应用。此时,可穿戴设备也会同步到您的包裹。
设备日志记录应列出如下内容:
11-07 14:58:53.127 3330-8739/? I/WearablePkgInstaller: Setting DataItem to install wearable apps for com.spotify.music
将com.spotify.music作为您的应用ID。这只是Spotify的一个例子。
在手表上(通过蓝牙或USB进行调试),您可以使用相同的过滤器查找记录问题或成功的日志记录:
11-07 15:00:02.533 1032-1048/? I/WearablePkgInstaller: Package com.spotify.music was installed.
大多数错误消息都是不言自明的。您可以在the source code of WearPackageInstallerService class中找到许多这些错误的示例。然而,有些只是一个returnCode。对于这些返回代码,请检查this PackageManager source code。
中的值答案 1 :(得分:5)
我有穿着产品口味。我在手持和磨损应用程序中都有口味。为了解决我的问题,我在build.gradle手持设备应用程序文件中使用了下一个代码:
freeWearApp project(path: ':wear', configuration: 'wearfreeRelease')
fullWearApp project(path: ':wear', configuration: 'wearfullRelease')
其中,免费和完整是来自手持模块的口味, wearfree 和 wearfull 是来自磨损模型的风味,发布是磨损模块构建类型的名称。
并且不要忘记穿着gradle的android块中的 publishNonDefault true
。
答案 2 :(得分:4)
即使检查了包裹名称,应用程序ID,并确保磨损模块没有掌上电脑模块没有的任何权限,我也遇到了这个问题。
我的问题是我在AndroidManifest.xml中为可穿戴模块请求了权限android.permission.BIND_NOTIFICATION_LISTENER_SERVICE
。似乎只应该为手持模块请求。
答案 3 :(得分:2)
据我所知,目前不支持产品口味,要么使用手动构建(资产或原始),要么删除产品口味。
答案 4 :(得分:2)
根据我的经验,它仅在构建签名版本构建时有效。在部署调试版本时,它从未为我安装过磨损应用程序。使用签名版本,它与buildToolsVersion“19.1.0”一起使用。
在这种特定情况下,我可以看到您正在使用“编译项目(':GooglePlay')”。那是最新版本吗? (编译'com.google.android.gms:play-services-wearable:+')
答案 5 :(得分:1)
我最近遇到过这个问题,最后修好了它;
然后它突然出现在手表上。在此之前,我尝试了一切,并确信构建不正常
答案 6 :(得分:1)
只是想在许多答案中添加一些我没见过的东西。
您必须匹配掌上电脑应用中的public static void <T> foo(List<X<T>> arg) {
...
}
并佩戴应用。
我以前认为您需要在掌上电脑和观看清单中都有applicationId
,并为可穿戴式清单添加<uses-feature android:name="android.hardware.type.watch" />
,为设备添加android:required="true"
。
这是假的。您在手持设备清单中不需要上述内容。事实上,随着Wear 2.0即将推出,谷歌已经做了一些更改,如果android:required="false"
在您的掌上电脑应用程序中,则不允许您上传任何minSdk低于23的apk。抱歉有任何困惑。
答案 7 :(得分:0)
遇到同样的问题,结果证明newrelic正在检测Android Wear App。尝试评论该行: // apply plugin:&#39; newrelic&#39;
我无法找到告诉newrelic远离磨损应用程序的方法...