现在我们的手机一般都内置有方向感应器,手机屏幕会根据所处位置自动进行横竖屏切换(前提是未锁定屏幕方向)。但有时我们的应用程序仅限在横屏或者竖屏状态下才可以运行,此时我们需要锁定该程序Activity运行时的屏幕方向。还有就是在我们用手机观看视频时,随意的进行横竖屏切换,但播放进度不会随着屏幕的转换而从头开始播放,为了实现这个功能,我们就需要在Activity转换时对当前数据进行保存。
现在根据以上两种需求,个人提出以下解决方案:
一、锁定Activity运行时屏幕方向,如下图(演示锁定横屏):
我们可以通过以下两种方式实现锁定Activity运行时屏幕的方向:
(1)通过修改AndroidMainfest.xml配置文件
修改Android/app/mainfests/AndroidMainfest.xml下的配置文件如下:
<activity>节点的android:screenOrientation属性可以完成该任务(portrait为保持竖屏,landscape为保持横屏)
1 <manifest xmlns:android="http://schemas.android.com/apk/res/android" 2 package="com.example.administrator.day18"> 3 <application 4 android:allowBackup="true" 5 android:icon="@mipmap/ic_launcher" 6 android:label="@string/app_name" 7 android:supportsRtl="true" 8 android:theme="@style/AppTheme"> 9 <activity android:name=".Main2Activity" 10 //添加screenOrientation属性(portrait为保持竖屏,landscape为保持横屏) 11 android:screenOrientation="landscape"> 12 <intent-filter> 13 <action android:name="android.intent.action.MAIN" /> 14 15 <category android:name="android.intent.category.LAUNCHER" /> 16 </intent-filter> 17 </activity> 18 </application> 19 </manifest>
(2)通过java代码实现:
1 import android.content.pm.ActivityInfo; 2 import android.os.Bundle; 3 import android.support.annotation.Nullable; 4 import android.support.v7.app.AppCompatActivity; 5 /** 6 * Created by panchengjia on 2016/12/9. 7 */ 8 public class Test extends AppCompatActivity { 9 @Override 10 protected void onCreate(@Nullable Bundle savedInstanceState) { 11 super.onCreate(savedInstanceState); 12 setContentView(R.layout.activity_main2); 13 //添加setRequestedOrientation方法实现锁定横屏(portrait为保持竖屏,landscape为保持横屏) 14 setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE); 15 } 16 }
现在第一种需求通过修改配置文件以及java代码的方式实现,但这里要注意的是当同时在配置文件以及java文件中设置了方向锁定时,系统会以配置文件设置为准(可以理解为配置文件优先级高于java代码)
二、Activity转换方向时数据的保存
下面我们展示Activity转换方向时未保存数据的情况:
演示中,我们通过记录按钮的点击次数(土司)来判断数据是否被保存
我们发现当屏幕(Acticity)由横屏转为竖屏时,数据未实现保存,继续从0开始计数点击次数。
当我们设置保存数据后,演示情况如下:
保存数据后,点击数土司会继续竖屏状态的数据更新,而不是从零重新开始。
针对数据保存我们同样有两种方法实现,分别为在java代码中使用onSaveInstanceState()方法以及在属性。
下面我们详细分析这两种方法的实现:
(1)onSaveInstanceState详解
实现代码如下(这里写了Activity的全部生命周期,并打印log用于分析性能):
1 import android.support.v7.app.AppCompatActivity; 2 import android.os.Bundle; 3 import android.util.Log; 4 import android.view.View; 5 import android.widget.Toast; 6 /** 7 * Created by panchengjia on 2016/12/9. 8 */ 9 public class Main2Activity extends AppCompatActivity { 10 int num=1;//初始点击次数为1 11 @Override 12 protected void onCreate(Bundle savedInstanceState) { 13 super.onCreate(savedInstanceState); 14 setContentView(R.layout.activity_main2); 15 if(savedInstanceState!=null){ 16 num=savedInstanceState.getInt("data"); 17 } 18 Log.i("Tag","onCreate"); 19 } 20 //设置点击事件土司记录数据 21 public void show(View v){ 22 Toast.makeText(this, num++ +"", Toast.LENGTH_SHORT).show(); 23 } 24 @Override 25 protected void onSaveInstanceState(Bundle outState) { 26 super.onSaveInstanceState(outState); 27 outState.putInt("data",num); 28 Log.i("Tag","onSaveInstanceState"); 29 } 30 @Override 31 protected void onStart() { 32 super.onStart(); 33 Log.i("Tag","onStart"); 34 } 35 @Override 36 protected void onResume() { 37 super.onResume(); 38 Log.i("Tag","onResume"); 39 } 40 @Override 41 protected void onRestart() { 42 super.onRestart(); 43 Log.i("Tag","onRestart"); 44 } 45 @Override 46 protected void onPause() { 47 super.onPause(); 48 Log.i("Tag","onPause"); 49 } 50 @Override 51 protected void onStop() { 52 super.onStop(); 53 Log.i("Tag","onStop"); 54 } 55 @Override 56 protected void onDestroy() { 57 super.onDestroy(); 58 Log.i("Tag","onDestroy"); 59 } 60 }
功能实现的核心代码为protected void onSaveInstanceState(Bundle outState)方法。Activity旋转前后的Log如下:
从Log中可以得知屏幕(Acticity)旋转时,会销毁之前的Activity,然后重建旋转后的Activity。
(2)修改AndroidManifest.xml配置文件
在AndroidManifest.xml中添加configChanges属性,如下:
1 <?xml version="1.0" encoding="utf-8"?> 2 <manifest xmlns:android="http://schemas.android.com/apk/res/android" 3 package="com.example.administrator.day18"> 4 <application 5 android:allowBackup="true" 6 android:icon="@mipmap/ic_launcher" 7 android:label="@string/app_name" 8 android:supportsRtl="true" 9 android:theme="@style/AppTheme"> 10 <activity android:name=".Main2Activity" 11 //添加configChanges属性 12 android:configChanges="orientation|keyboard|screenSize"> 13 <intent-filter> 14 <action android:name="android.intent.action.MAIN" /> 15 16 <category android:name="android.intent.category.LAUNCHER" /> 17 </intent-filter> 18 </activity> 19 <!--<activity android:name=".Main2Activity"></activity>--> 20 </application> 21 </manifest>
在java中添加对应的onConfigurationChanged()方法用于Log打印分析,方法本身在功能实现上无实际意义:
1 @Override 2 public void onConfigurationChanged(Configuration newConfig) { 3 super.onConfigurationChanged(newConfig); 4 Log.i("Tag","onConfigurationChanged"); 5 }
屏幕旋转前后Log如下(演示两次旋转):
从Log中可以得知屏幕(Acticity)旋转时,使用第二种方法保存数据不会销毁Activity,不用重复的销毁建立Activity从而达到优化内存性能的目的。建议使用这种方法