Bitmap加载

创建BitmapOptions属性对象

BitmapFactory.Options options = new BitmapFactory.Options();
//options.outWidth = 800;   //图片宽度
//options.outHeight = 600   //图片高度
options.inJustDecodeBounds = false;     //设为true,那么BitmapFactory并不会真的返回一个Bitmap,它仅仅会把它的宽,高取回来给你
options.inSampleSize = 2;   //大于1时,图像高、宽分别以2的inSampleSize次方分之一缩小
options.inPreferredConfig = Bitmap.Config.ARGB_4444;    //设置图片的采样率
options.inScaled = true;    //是否需要放缩位图

采样率属性

  • ALPHA_8: 每个像素用占8位,存储的是图像的透明值,占1个字节;
  • RGB_565:每个像素用占16位,分别为5-R,6-G,5-B通道,占2个字节;
  • ARGB-4444:每个像素占16位,即每个通道用4位表示,占2个字节;
  • ARGB_8888:每个像素占32位,每个通道用8位表示,占4个字节;

应用到位图

Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length, options);

图片的不同来源

  1. BitmapFactory.decodeBetyArray();//字节数组
  2. BitmapFactory.decodeFile();//文件路径
  3. BitmapFactory.decodeResource();//资源ID
  4. BitmapFactory.decodeStream();//流

LruCache缓存机制

使用的缓存策略是LruCache和DiskLruCache。LruCache作为内存缓存DiskLruCache作为磁盘缓存
lru是least recently used,最少使用策略,缓存快满时删除最少使用的缓存目标

LruCache的实现机制是维护一张LinkedHashMap,是由数组和双向链表的数据结构实现的。新的缓存对象从队头插入,有对象被访问则把该对象移动到队头。长时间不被访问的对象就会被移动到队尾,缓存空间满了之后就删除队尾的对象。

使用方法

int maxMemory = (int) Runtime.getRuntime().maxMemory(); //获取应用的可用内存大小
int cacheSize = maxMemory / 8;  //定义缓存的总容量大小,此处为内存容量的 1/8
//创建LruCache,定义key和value的类型。重写计算缓存单位大小的sizeOf方法
LruCache<String, Bitmap> imgCache = new LruCache<String, Bitmap>(cacheSize) {
        @Override
        protected int sizeOf(String key, Bitmap value) {
            return value.getRowBytes() * value.getHeight();
        }
    };

imgCache.put("TEST_IMG", bitmap);   //存储到缓存
bitmap = imgCache.get("TEST_IMG");  //从缓存中取出

LruCache只是内存缓存,缓存内容保持在LruCache作用域期间,对象销毁之后缓存即被删除
持久化缓存使用DiskLruCache

DiskLruCache

导入包

compile 'com.jakewharton:disklrucache:2.0.2'

import class

import com.jakewharton.disklrucache.DiskLruCache;

使用

创建缓存对象
DiskLruCache使用open方法创建自身,参数分别为

  • 缓存文件的存储目录
  • 应用的版本号,版本号更新的时候旧的缓存会被清空
  • 单个节点所对应的数据个数
  • 缓存的总大小

    DiskLruCache diskLruCache = null; try { diskLruCache = DiskLruCache.open(getExternalCacheDir(), 1, 1, cacheSize); } catch (IOException e) { e.printStackTrace(); }

DiskLruCache

数据缓存到存储
使用editor.newOutputStream(INDEX);方法来新建存储数据的输入流,INDEX参数是单个节点的编号

//
try {
    DiskLruCache.Editor editor = diskLruCache.edit("testimg");
    if(editor != null) {
        OutputStream outputStream = editor.newOutputStream(0);
        outputStream.write(data);
        //写入之后必须进行commit操作
        outputStream.flush();
        editor.commit();
    }

} catch (IOException e) {
    e.printStackTrace();
}

读取磁盘缓存

try {
    //获取缓存的snapshot,参数为key
    DiskLruCache.Snapshot snapshot = diskLruCache.get("testimg");
    if(snapshot!=null){
        //snapshot中获取缓存的数据,参数为单个节点的编号
        Bitmap bitmap = BitmapFactory.decodeStream(snapshot.getInputStream(0));
        return bitmap;
    }
} catch (IOException e) {
    e.printStackTrace();
}

使用LocalBroadcastManager完全退出App的方法

定义广播Action名称

public final static String BROADCAST_EXIT = "BROADCAST_EXIT";

Application类中定义LocalBroadcastManager

public class AppContext extends Application {
    public LocalBroadcastManager localBroadcastManager;

    @Override
    public void onCreate() {
        localBroadcastManager = LocalBroadcastManager.getInstance(this);
    }
}

每一个Activity中定义BroadcastReceiver,收到广播的时候退出。Activity需要手动注册和注销接收器

public class MainpageActivity extends AppCompatActivity {
    。。。
    private BroadcastReceiver broadcastReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            finish();
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        。。。
        appContext.localBroadcastManager.registerReceiver(broadcastReceiver, new IntentFilter(StaticValues.BROADCAST_EXIT));
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        appContext.localBroadcastManager.unregisterReceiver(broadcastReceiver);
    }

}

在需要退出App的地方发送广播

//退出App,关闭所有activity
appContext.localBroadcastManager.sendBroadcast(new Intent(StaticValues.BROADCAST_EXIT));

点9图

图片放大时控制缩放区域
文件名以”.9.png“结尾
保证图片在不模糊变形的前提下做到自适应

4个边界的含义

四周分别命名为Left、Top、Right、Bottom

  • 绘制在L的区域:可以被拉伸的纵向区域
  • 绘制在T的区域:可以被拉伸的横向区域
  • 绘制在R的区域:用于显示内容区域的纵向范围
  • 绘制在B的区域:用于显示内容区域的横向范围

Application

App启动的时候最先启动的组件,早于MainActivity.整个App的一个单例对象.生命周期相当于App的生命周期


资源存储 /raw /asset 的区别

相同点是两者目录下的文件在打包后会原封不动的保存在apk包中,不会被编译成二进制。

不同点

  • res/raw中的文件会被映射到R.java文件中,访问的时候直接使用资源ID即R.id.filename;assets文件夹下的文件不会被映射到R.java中,访问的时候需要AssetManager类。
  • res/raw不可以有目录结构,而assets则可以有目录结构,也就是assets目录下可以再建立文件夹

读取文件资源

读取res/raw下的文件资源,通过以下方式获取输入流来进行写操作

InputStream is = getResources().openRawResource(R.id.filename);  

读取assets下的文件资源,通过以下方式获取输入流来进行写操作

AssetManager am = null;  
am = getAssets();  
InputStream is = am.open("filename");

屏幕显示dp、sp、px的区别

dp (dip)设备独立像素,不依赖于实际的分辨率,根据屏幕密度自动进行变换(推荐使用这种单位)
sp (scaled pixels)(放大像素)主要用于字体显示(best for textsize),跟dp不同的是可以根据用户的字体大小首选项进行缩放
px 实际像素数,在不同分辨率设备上显示效果不同,不推荐使用


图片加载相关

图片加载时使用BitmapFactory.Option来调整图片的尺寸和采样率
inSampleSize 等于1时采样后的图片等同于原尺寸,2的时候宽高为原尺寸的1/2

图片加载库 Glide使用

创建文件GlideApp.java

package com.viator42.ugo.utils;

import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.Context;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.VisibleForTesting;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentActivity;
import android.view.View;
import com.bumptech.glide.Glide;
import com.bumptech.glide.GlideBuilder;
import java.io.File;
import java.lang.Deprecated;
import java.lang.String;

/**
* The entry point for interacting with Glide for Applications
*
* <p>Includes all generated APIs from all
* {@link com.bumptech.glide.annotation.GlideExtension}s in source and dependent libraries.
*
* <p>This class is generated and should not be modified
* @see Glide
*/
public final class GlideApp {
    private GlideApp() {
    }

    /**
    * @see Glide#getPhotoCacheDir(Context)
    */
    @Nullable
    public static File getPhotoCacheDir(@NonNull Context context) {
        return Glide.getPhotoCacheDir(context);
    }

    /**
    * @see Glide#getPhotoCacheDir(Context, String)
    */
    @Nullable
    public static File getPhotoCacheDir(@NonNull Context context, @NonNull String string) {
        return Glide.getPhotoCacheDir(context, string);
    }

    /**
    * @see Glide#get(Context)
    */
    @NonNull
    public static Glide get(@NonNull Context context) {
        return Glide.get(context);
    }

    /**
    * @see Glide#init(Glide)
    */
    @Deprecated
    @VisibleForTesting
    @SuppressLint("VisibleForTests")
    public static void init(Glide glide) {
        Glide.init(glide);
    }

    /**
    * @see Glide#init(Context, GlideBuilder)
    */
    @VisibleForTesting
    @SuppressLint("VisibleForTests")
    public static void init(@NonNull Context context, @NonNull GlideBuilder builder) {
        Glide.init(context, builder);
    }

    /**
    * @see Glide#tearDown()
    */
    @VisibleForTesting
    @SuppressLint("VisibleForTests")
    public static void tearDown() {
        Glide.tearDown();
    }

    /**
    * @see Glide#with(Context)
    */
    @NonNull
    public static GlideRequests with(@NonNull Context context) {
        return (GlideRequests) Glide.with(context);
    }

    /**
    * @see Glide#with(Activity)
    */
    @NonNull
    public static GlideRequests with(@NonNull Activity activity) {
        return (GlideRequests) Glide.with(activity);
    }

    /**
    * @see Glide#with(FragmentActivity)
    */
    @NonNull
    public static GlideRequests with(@NonNull FragmentActivity activity) {
        return (GlideRequests) Glide.with(activity);
    }

    /**
    * @see Glide#with(Fragment)
    */
    @NonNull
    public static GlideRequests with(@NonNull Fragment fragment) {
        return (GlideRequests) Glide.with(fragment);
    }

    /**
    * @see Glide#with(Fragment)
    */
    @Deprecated
    @NonNull
    public static GlideRequests with(@NonNull android.app.Fragment fragment) {
        return (GlideRequests) Glide.with(fragment);
    }

    /**
    * @see Glide#with(View)
    */
    @NonNull
    public static GlideRequests with(@NonNull View view) {
        return (GlideRequests) Glide.with(view);
    }
}

加载图片

GlideApp.with(context)
    .load(url)
    .centerCrop()
    .into(imgView);

图片显示fitScale的定义

  • center 居中,无缩放。
  • centerCrop 保持宽高比缩小或放大,使得两边都大于或等于显示边界,且宽或高契合显示边界。居中显示。
  • focusCrop 同centerCrop, 但居中点不是中点,而是指定的某个点。
  • centerInside 缩放图片使两边都在显示边界内,居中显示。和 fitCenter 不同,不会对图片进行放大。如果图尺寸大于显示边界,则保持长宽比缩小图片。
  • fitCenter 保持宽高比,缩小或者放大,使得图片完全显示在显示边界内,且宽或高契合显示边界。居中显示。
  • fitStart 同上。但不居中,和显示边界左上对齐。
  • fitEnd 同fitCenter, 但不居中,和显示边界右下对齐。
  • fitXY 不保存宽高比,填充满显示边界。
  • none 如要使用tile mode显示, 需要设置为none

获取界面组件

<Button
android:id="@+id/btn1" />
Button btn1 = findViewById(R.id.btn1);


输出log日志

import android.util.Log;
Log.v(“log title”, “value");

2个参数,一个是tag,另一个是日志内容.

输出日志类型

  • Log.v(); 日志信息,对应级别verbose
  • Log.i(); 调试信息,对应级别info
  • Log.w(); 警告信息,对应级别warning
  • Log.e(); 错误信息,对应级别error

RelativeLayout属性

第一类:属性值为true或false

Android:layout_centerHrizontal 水平居中
android:layout_centerVertical 垂直居中
android:layout_centerInparent 相对于父元素完全居中
android:layout_alignParentBottom 贴紧父元素的下边缘
android:layout_alignParentLeft 贴紧父元素的左边缘
android:layout_alignParentRight 贴紧父元素的右边缘
android:layout_alignParentTop 贴紧父元素的上边缘
android:layout_alignWithParentIfMissing 如果对应的兄弟元素找不到的话就以父元素做参照物

第二类:属性值必须为id的引用名“@id/id-name”

android:layout_below 在某元素的下方
android:layout_above 在某元素的的上方
android:layout_toLeftOf 在某元素的左边
android:layout_toRightOf 在某元素的右边
android:layout_alignTop 本元素的上边缘和某元素的的上边缘对齐
android:layout_alignLeft 本元素的左边缘和某元素的的左边缘对齐
android:layout_alignBottom 本元素的下边缘和某元素的的下边缘对齐
android:layout_alignRight 本元素的右边缘和某元素的的右边缘对齐

第三类:属性值为具体的像素值,如30dip,40px

android:layout_marginBottom 离某元素底边缘的距离
android:layout_marginLeft 离某元素左边缘的距离
android:layout_marginRight 离某元素右边缘的距离
android:layout_marginTop 离某元素上边缘的距离


SharedReference

//获取SharedPreferences实例     第一个参数是名称, 第二个是作用域
SharedPreferences ref = getSharedPreferences("user", Context.MODE_PRIVATE);
//获取数据,参数分别为key值和默认值(未取到值时的返回值)
int id = ref.getInt("id", 0);
ref.getString("username", "")

//向SharedPreferences存储数据
SharedPreferences ref = activity.getSharedPreferences("user", Context.MODE_PRIVATE);
//获取editor对象
SharedPreferences.Editor editor = ref.edit();
//赋值
editor.putInt("id", 1);
editor.putString("tel", “1234567890");
//完成后提交修改
editor.commit();

Fragment (Android4 以上)

Fragment生命周期 Fragment Lifescycle

Fragment的生命周期和Activity的生命周期关系

onAttach()
在Fragment和Activity关联时调用

onActivityCreated()
在 Activity 的 onCreate() 方法已返回时调用

onDestroyView()
Fragment从屏幕上移除的时候调用

onDetach()
在取消Fragment与Activity的关联时调用

实际演示

场景演示 : 切换到该Fragment

11-29 14:26:35.095: D/AppListFragment(7649): onAttach
11-29 14:26:35.095: D/AppListFragment(7649): onCreate
11-29 14:26:35.095: D/AppListFragment(7649): onCreateView
11-29 14:26:35.100: D/AppListFragment(7649): onActivityCreated
11-29 14:26:35.120: D/AppListFragment(7649): onStart
11-29 14:26:35.120: D/AppListFragment(7649): onResume

屏幕灭掉:

11-29 14:27:35.185: D/AppListFragment(7649): onPause
11-29 14:27:35.205: D/AppListFragment(7649): onSaveInstanceState
11-29 14:27:35.205: D/AppListFragment(7649): onStop

屏幕解锁

11-29 14:33:13.240: D/AppListFragment(7649): onStart
11-29 14:33:13.275: D/AppListFragment(7649): onResume

切换到其他Fragment: 11-29 14:33:33.655: D/AppListFragment(7649): onPause
11-29 14:33:33.655: D/AppListFragment(7649): onStop
11-29 14:33:33.660: D/AppListFragment(7649): onDestroyView

切换回本身的Fragment:

11-29 14:33:55.820: D/AppListFragment(7649): onCreateView
11-29 14:33:55.825: D/AppListFragment(7649): onActivityCreated
11-29 14:33:55.825: D/AppListFragment(7649): onStart
11-29 14:33:55.825: D/AppListFragment(7649): onResume

回到桌面

11-29 14:34:26.590: D/AppListFragment(7649): onPause
11-29 14:34:26.880: D/AppListFragment(7649): onSaveInstanceState
11-29 14:34:26.880: D/AppListFragment(7649): onStop

回到应用

11-29 14:36:51.940: D/AppListFragment(7649): onStart
11-29 14:36:51.940: D/AppListFragment(7649): onResume

退出应用

11-29 14:37:03.020: D/AppListFragment(7649): onPause
11-29 14:37:03.155: D/AppListFragment(7649): onStop
11-29 14:37:03.155: D/AppListFragment(7649): onDestroyView
11-29 14:37:03.165: D/AppListFragment(7649): onDestroy
11-29 14:37:03.165: D/AppListFragment(7649): onDetach

Fragment类定义 HelloFragment.java

public class HelloFragment extends Fragment

自定义fragment类继承Fragment

包含以下方法

public class StatisticFragment extends Fragment {
    private OnFragmentInteractionListener mListener;
    private AppContext context;
    private Store store;

    private TextView productCountTextView;

    /**
     * Use this factory method to create a new instance of
     * this fragment using the provided parameters.
     *
     * @return A new instance of fragment StatisticFragment.
     */
    // TODO: Rename and change types and number of parameters
    public static StatisticFragment newInstance(String param1, String param2) {
        StatisticFragment fragment = new StatisticFragment();
        Bundle args = new Bundle();

        fragment.setArguments(args);
        return fragment;
    }

    public StatisticFragment() {
        // Required empty public constructor
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        context = (AppContext) getActivity().getApplicationContext();
        store = context.getStore();

    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
        View view = inflater.inflate(R.layout.fragment_statistic, container, false);

        productCountTextView = (TextView) view.findViewById(R.id.product_count);
        
        return view;
    }
}

Activity.java 使用 Fragment

先定义FragmentManager

private FragmentManager manager;
private FragmentTransaction transaction;

设置Fragment

manager = getFragmentManager();
transaction = manager.beginTransaction();
transaction.replace(R.id.content, fragment );
transaction.commit();

fragment传递参数

创建对象时使用bundle传递参数

fragment1 = new Fragment1();
Bundle bundle = new Bundle();
bundle.putParcelable("data", model01);

fragment1.setArguments(bundle);

transaction = manager.beginTransaction();
transaction.replace(R.id.content, fragment1);
transaction.commit();

然后fragment获取参数

public static Fragment1 newInstance(Model01 model01) {
    Fragment1 fragment = new Fragment1();
    Bundle args = new Bundle();
    args.putParcelable("data", model01);
    fragment.setArguments(args);
    return fragment;
}

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    if (getArguments() != null) {
        model01 = getArguments().getParcelable("data");
    }
}

可以传递基本类型和parcelble对象

通过id获取fragment对象

FragmentManager manager = getFragmentManager();
Fragment1 fragment1 = (Fragment1) manager.findFragmentById(R.id.content);

获取到fragment实例之后就可以执行重新赋值的操作

Fragment切换

FragmentTransaction transaction;
transaction = manager.beginTransaction();
transaction.setCustomAnimations(android.R.anim.fade_in, R.anim.slide_out);  //可以为fragment切换设置转场动画
transaction.replace(R.id.content_frame, fragment)
transaction.commit();

使用replace方法实现fragment的切换,但这种方法的缺点是每次切换都会导致fragment重新加载
解决方法是hide上一个Fragment,如果新的fragment不存在就先创建再add,如果已存在直接调用show

transaction.hide(currentFragment);
if(mainpageFragment == null) {
    mainpageFragment = new MainpageFragment();
    transaction.add(R.id.content, mainpageFragment);
}
else {
    transaction.show(mainpageFragment);
}
currentFragment = mainpageFragment;
transaction.commit();

ListView

简单列表,只显示一个textview,使用ArrayAdapter实现.

private ArrayList<String> paymethodList;
private ListView payMethodListView;

payMethodListView = (ListView) view.findViewById(R.id.pay_method_list);

paymethodList = new ArrayList<String>();
        paymethodList.add("支付宝");
        paymethodList.add("信用卡1");
        paymethodList.add("信用卡2");

ArrayAdapter adapter = new ArrayAdapter(context, android.R.layout.simple_list_item_1, paymethodList);
        payMethodListView.setAdapter(adapter);
        payMethodListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                payMethod = paymethodList.get(position);
                setPayConfirmMode();
            }
        });

带单选框的listview

listview=new ListView(this);

//android.R.layout.simple_list_item_single_choice设置ListView项目条带单选按钮

listview.setAdapter(new ArrayAdapter<String>(this,android.R.layout.simple_list_item_single_choice,getData()));

listview.setChoiceMode(ListView.CHOICE_MODE_SINGLE);//如果不使用这个设置,选项中的radiobutton无法响应选中事件

listview.setItemChecked(2, true);

payMethodListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                listview.setItemChecked(position, true);
            }
        });

动态添加数据

adapter.add() 或者addAll()添加元素
adapter.notifyDataSetChanged();      //通知listview数据改变

ListView自定义布局

先创建一个layout resource file: layout.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent" android:layout_height="match_parent">

<TextView
android:layout_width="100dp"
android:layout_height="50dp"
android:id="@+id/name" />

<TextView
android:layout_width="100dp"
android:layout_height="50dp"
android:id="@+id/price" />

<TextView
android:layout_width="100dp"
android:layout_height="50dp"
android:id="@+id/address" />
</LinearLayout>

创建一个Adaper类继承自BaseAdapter

public class ContentAdaper extends BaseAdapter {
    //填充数据的List
    List<Map<String, Object>> list = new ArrayList<Map<String, Object>>();

    //上下文
    private Context context;

    //用来导入布局
    private LayoutInflater inflater = null;

    //构造器
    public ContentAdaper(List<Map<String, Object>> list, Context context) {
        this.context = context;
        this.list = list;
        inflater = LayoutInflater.from(context);
    }

    @Override
    public int getCount() {
        // TODO Auto-generated method stub
        return list.size();
    }

    @Override
    public Object getItem(int arg0) {
        // TODO Auto-generated method stub
        return list.get(arg0);
    }

    @Override
    public long getItemId(int position) {
        // TODO Auto-generated method stub
        return position;
    }

    //核心部分,返回Listview视图
    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        // TODO Auto-generated method stub
        ViewHolder holder = null;

        if (convertView == null) {
            holder = new ViewHolder();
            convertView = inflater.inflate(R.layout.layout, null);
            holder.name = (TextView) convertView.findViewById(R.id.name);
            holder.price = (TextView) convertView.findViewById(R.id.price);
            holder.address = (TextView) convertView.findViewById(R.id.address);
            //为view设置标签
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }

        holder.name.setText(list.get(position).get("name").toString());
        holder.price.setText(list.get(position).get("price").toString());
        holder.address.setText(list.get(position).get("address").toString());

        return convertView;
    }

    static class ViewHolder {
        TextView name;
        TextView price;
        TextView address;
    }
}

最后在Activity中调用Adaper.

List<Map<String,Object>> data = new ArrayList<Map<String,Object>>();
Map item1 = new HashMap();
item1.put("name", "Name1");
item1.put("price", "450")
item1.put("address", "address 101");
data.add(item1);
ContentAdaper adaper = new ContentAdaper(data, this);
listView.setAdapter(adaper);

listView onClick事件设置

position

orderlistView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
        @Override
        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {

        }
    });

参数id用来返回某一行的标识id, 默认返回的是行号.需要重写Adapter的getItemId方法来获取id属性.

@Override
public long getItemId(int position) {
    // TODO Auto-generated method stub
    return (long)list.get(position).get("id");

}

ScrollTo方法操作

mListView.scrollTo(x, y)

把显示区域移动到x, y坐标处

scrollTo和scrollBy的区别

scrollTo是距离起始位置的偏移量,scrollBy是根据当前位置的移动

android scrollview 自动滚动到顶部或者底部

//设置默认滚动到顶部
scrollView.post(new Runnable() {

@Override
public void run() {
    // TODO Auto-generated method stub
    scrollView.fullScroll(ScrollView.FOCUS_UP);
}
});

//设置默认滚动到底部
scrollView.post(new Runnable() {

@Override
public void run() {
    // TODO Auto-generated method stub
    scrollView.fullScroll(ScrollView.FOCUS_DOWN);
}
});

View中有EditText的时候页面加载焦点会自动放到第一个EditText上,取消方法是在跟节点添加以下属性

android:focusable="true"
android:focusableInTouchMode="true" 

使用SimpleAddpter

List listData = new ArrayList<Map<String,Object>>();

Map line = new HashMap();
line.put("id", 1);
line.put("name", "beijing");
listData.add(line);

// 参数含义, context, 列表数据, 列表项布局, 显示的value名称, 名称对应的xml布局id
SimpleAdapter adapter = new SimpleAdapter(this, listData, R.layout.spinner_item, new String[] {"name"}, new int[] {R.id.name});
provinceSpinner.setAdapter(adapter);

SimpleAdapter构造函数的参数分别为

  1. 上下文,一般为Activity类
  2. 列表数据, 类型是List,每个元素是Map, 每个Map对象包含key,value键值对表示属性名和属性值.
  3. 布局xml文件.
  4. 字符串数组, 表示数据中key的属性名.
  5. int数组, 数据中每个key属性名所对应的样式xml中的组件id.

最后两个参数数组项是一一对应的,把数据赋值给对应的控件显示

设置列表项选中时的事件

provinceSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
        @Override
        public void onItemSelected(AdapterView<?> adapterView, View view, int i, long l) {
            Toast.makeText(RegisterActivity.this, ((Map)provinceSpinner.getItemAtPosition(i)).get("id").toString(), Toast.LENGTH_SHORT).show();
        }

        @Override
        public void onNothingSelected(AdapterView<?> adapterView) {

        }
    });

Application 全局类配置

定义一个类可以继承Application, onCreate方法在应用启动时运行.

public class AppContext extends Application
{
    private User user;
    private Store store;

    @Override
    public void onCreate() {
        super.onCreate();

    }
}

必须在manifest中指定类名.

<application
    android:name=".AppContext" >

在Activity中可以直接调用Application类的方法和变量

AppContext context =  (AppContext)activity.getApplication();

示例

从网络上下载图片的示例

AndroidManifest.xml添加网络访问权限

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

导入Hanler和Message类.

import android.os.Handler;
import android.os.Message;

Activity类和OnCreate方法.

private String imgUrl = "http://image_url";
private final int IS_END = 1;
private ProgressDialog dialog;

protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    img = (ImageView)findViewById(R.id.img);
    loadImgBtn = (Button)findViewById(R.id.load_image);
    loadImgBtn.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
        	//新建一个网络访问的线程.
            new Thread(new MyThread()).start();
            dialog.show();

        }
    });

    dialog = new ProgressDialog(this);
    dialog.setTitle("提示");
    dialog.setMessage("正在下载,请稍后....");
    dialog.setCancelable(false);

}

Activity类中新建一个Handler类并重写handleMessage方法.

public Handler handler = new Handler() {
    @Override
    public void handleMessage(Message msg)
    {
    	//获取Message消息
        byte[] data = (byte[])msg.obj;
        Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
        img.setImageBitmap(bitmap);

        if(msg.what == IS_END)
        {
            dialog.dismiss();
        }

    }
};


public class MyThread implements Runnable
{

    @Override
    public void run() {
        HttpClient httpClient = new DefaultHttpClient();
        HttpGet httpGet = new HttpGet(imgUrl);
        HttpResponse response = null;
        try{
            response = httpClient.execute(httpGet);
            if(response.getStatusLine().getStatusCode() == 200)
            {
                byte[] data = EntityUtils.toByteArray(response.getEntity());
                //Message.obtain()方法是从消息池中获取消息对象.
                Message message = Message.obtain();
                message.obj = data;
                message.what = IS_END;
                //发送Message消息
                handler.sendMessage(message);

            }

        } catch (Exception e)
        {
            e.printStackTrace();
        }


    }
}

查看SHA1签名

(Mac OS)

keytool -list -v -keystore ~/.android/debug.keystore

口令没有就直接enter


Spinner选择器

有dialog和dropdown两种显示方式. 需要使用适配器来完成填充数据.

获取数据, 设置显示内容name和值id.

ArrayList<Map<String, Object>> result= new ArrayList<Map<String, Object>>();
Map province = new HashMap();
province.put("id", jsonObject.getString("code"));
province.put("name", jsonObject.getString("name"));
result.add(province);

设置spinner

provinceSpinner = (Spinner)findViewById(R.id.province);

SimpleAdapter adapter = new SimpleAdapter
    NewStoreActivity.this, result, R.layout.spinner_item, new String[] {"name"}, new int[] {R.id.alias});
provinceSpinner.setAdapter(adapter);
provinceSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
    @Override
    public void onItemSelected(AdapterView<?> adapterView, View view, int i, long l) {
        String code = ((Map) provinceSpinner.getItemAtPosition(i)).get("id").toString();
        new GetCityTask().execute(new Integer(code));

    }
    @Override
    public void onNothingSelected(AdapterView<?> adapterView) {

    }
});

Spinner_item.xml布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent" android:layout_height="match_parent"
    android:orientation="horizontal">

    <TextView
        android:layout_width="match_parent"
        android:layout_height="25dp"
        android:id="@+id/alias"
        android:gravity="center" />
</LinearLayout>

自定义选项布局

/res/value下添加spinner_item.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent" android:layout_height="match_parent"
    android:orientation="horizontal">

    <TextView
        android:layout_width="match_parent"
        android:layout_height="25dp"
        android:id="@+id/name"
        android:gravity="center" />
</LinearLayout>

AVD 模拟器添加sd卡

使用mksdcard程序,位置如下(MacOS系统)

/Users/viator42/Library/Android/sdk/tools/mksdcard

命令格式 label:标签 size:大小 file:文件位置

mksdcard -l <label> <size> <file>

获取相册图片

方式1, 直接返回图片数据.

点击按钮触发打开相册的intent,此处putExtra(“return-data”, true)是直接返回图片数据,图片较小而且不需要剪裁的时候使用.否则使用临时文件存储.

iconImageBtn.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        //打开相册
        Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
        intent.setType("image/*");
        intent.putExtra("crop", "true");
        intent.putExtra("return-data", true);

        startActivityForResult(intent, 1000);
    }
});

定义onActivityResult方法,获取Bitmap图片.

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (requestCode == 1000 && resultCode == RESULT_OK) {
        if(data != null)
        {
            ContentResolver resolver = getContentResolver();
            Uri originalUri = data.getData();
            try {
                MediaStore.Images.Media.getBitmap(resolver, originalUri);

            } catch (IOException e) {
                e.printStackTrace();
            }
            if (logoBitmap != null)
            {
                //获取Bitmap
                iconImageBtn.setImageBitmap(logoBitmap);

            }

        }
        else
        {
            Toast.makeText(activity.getApplicationContext(), "图片读取失败", Toast.LENGTH_SHORT).show();
        }
    }

    super.onActivityResult(requestCode, resultCode, data);

}

方式2, 使用临时文件. 如果图片需要剪裁则必须使用临时文件,否则会出异常.

定义文件

private File sdcardTempFile = new File("/mnt/sdcard/", "tmp_pic_" + SystemClock.currentThreadTimeMillis() + ".jpg");

点击按钮触发打开相册的intent

iconImageBtn.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        //打开相册
        Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
        intent.setType("image/*");
        intent.putExtra("crop", "true");
        intent.putExtra("aspectX", 1);// 裁剪框比例
        intent.putExtra("aspectY", 1);
        intent.putExtra("outputX", StaticValues.ICON_WIDTH);// 输出图片大小
        intent.putExtra("outputY", StaticValues.ICON_HEIGHT);
        intent.putExtra("return-data", false);// 不返回图片数据(必须)
        intent.putExtra("output", Uri.fromFile(sdcardTempFile));

        startActivityForResult(intent, 1000);
        }
    });

定义onActivityResult方法,获取Bitmap图片.

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (requestCode == 1000 && resultCode == RESULT_OK) {
        if(data != null)
        {
            logoBitmap = BitmapFactory.decodeFile(sdcardTempFile.getAbsolutePath());
            if (logoBitmap != null)
            {
                iconImageBtn.setImageBitmap(logoBitmap);
            }s
        }
        else
        {
            Toast.makeText(activity.getApplicationContext(), "图片读取失败", Toast.LENGTH_SHORT).show();
        }
    }
    super.onActivityResult(requestCode, resultCode, data);
}

图片尺寸压缩

/**
 * 图片进行压缩处理,宽度固定,高度自动适配
 * @param bm
 *
 * @return
 */
public static Bitmap zoomImg(Bitmap bm){
    // 获得图片的宽高
    int width = bm.getWidth();
    int height = bm.getHeight();

    if(width > 0 && height > 0)
    {
        if(width <= StaticValues.WIDTH_LIMIT)
        {
            //原图小于规定尺寸的不用缩放
            return bm;
        }
        else
        {
            int newWidth = StaticValues.WIDTH_LIMIT;
            int newHeight = (int)((newWidth * height) / width);
            // 计算缩放比例
            float scaleWidth = ((float) newWidth) / width;
            float scaleHeight = ((float) newHeight) / height;

            // 取得想要缩放的matrix参数
            Matrix matrix = new Matrix();
            matrix.postScale(scaleWidth, scaleHeight);
            // 得到新的图片
            Bitmap newbm = Bitmap.createBitmap(bm, 0, 0, width, height, matrix, true);
            return newbm;
        }
    }
    else
    {
        return bm;

    }
}

图片裁剪

Intent intent = new Intent("com.android.camera.action.CROP");
intent.setDataAndType(imageUri, "image/*");
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", StaticValues.ICON_WIDTH);
intent.putExtra("outputY", StaticValues.ICON_HEIGHT);
intent.putExtra("scale", true);
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
intent.putExtra("return-data", false);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); // no face detection
startActivityForResult(intent, StaticValues.IMAGE_CROP);

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);

    if(requestCode == StaticValues.IMAGE_CROP && resultCode == RESULT_OK)
    {
        //裁剪
        if(imageUri != null){
            logoBitmap = decodeUriAsBitmap(imageUri);//decode bitmap
            iconImageBtn.setImageBitmap(logoBitmap);
        }
        else
        {
            Toast.makeText(NewStoreActivity.this, "图片裁剪失败", Toast.LENGTH_SHORT).show();
        }
    }
}

GPS获取当前位置

AndroidManifest.xml文件添加以下权限.

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="im.supai.supaimarketing" >
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
...
</manifest>

类必须继承LocationListener接口并重写相关函数.

public class NewStoreActivity extends BaseAccivity implements LocationListener {
    
    //位置相关
    @Override
    public void onLocationChanged(Location location) {
        latitude = location.getLatitude();
        longitude = location.getLongitude();
    }

    @Override
    public void onStatusChanged(String s, int i, Bundle bundle) {
    }

    @Override
    public void onProviderEnabled(String s) {
    }

    @Override
    public void onProviderDisabled(String s) {
    }
}

在Activity类中需要获取位置的地方新建一个LocationManager并发送请求即可.

LocationManager locationManager = (LocationManager)getSystemService(Context.LOCATION_SERVICE);
locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, (android.location.LocationListener) this);

ImageView/ImageButton 设置缩放

设置自适应调整大小和最大宽度.

itemBtn.setAdjustViewBounds(true);
itemBtn.setMaxWidth(100);

根据权重设置控件宽度

把view的宽度设置为0dp同时设置weight属性.

android:layout_width="0dp"
android:layout_weight="1"

AutoCompleteTextView的使用方法

AutoCompleteTextView可以为输入框提供补全提示功能

在layout中定义widget

<AutoCompleteTextView
android:id="@+id/search_bar" />

activity中实现

private AutoCompleteTextView searchEditText;
searchEditText = (AutoCompleteTextView) findViewById(R.id.search_bar);

String[] ary = new String[]{"aaa", "bbb"};  //定义候选词列表,可以使用字符串数组或者ArrayList

//定义adaper
ArrayAdapter<String> searchEditAdapter=new ArrayAdapter<String>(this, 
    android.R.layout.simple_dropdown_item_1line, context.searchList);
searchEditText.setAdapter(searchEditAdapter);


Android ToolBar使用

ToolBar是Android 5.0(API Level 21)之后用来取代ActionBar的

1.AndroidMenifest.xml 中修改theme为NoActionBar

android:theme="@style/AppTheme.NoActionBar"

2.activity.xml中添加ToolBar

<android.support.design.widget.AppBarLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:theme="@style/AppTheme.AppBarOverlay">

    <android.support.v7.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_width="match_parent"
        android:layout_height="?attr/actionBarSize"
        android:background="?attr/colorPrimary"
        app:popupTheme="@style/AppTheme.PopupOverlay" >

        <!--自定义控件-->
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="match_parent"
            android:text="自定义控件" />

    </android.support.v7.widget.Toolbar>
</android.support.design.widget.AppBarLayout>

Toolbar中可以添加自定义控件

3.activity.java

public class MainActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //定义和初始化ToolBar
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        toolbar.setNavigationIcon(R.mipmap.ic_launcher);//设置导航栏图标
        toolbar.setLogo(R.mipmap.ic_launcher);//设置app logo
        toolbar.setTitle("Title");//设置主标题
        toolbar.setSubtitle("Subtitle");//设置子标题

    }
}

Android4.0以上的系统Menu默认在ToolBar的右上

Menu的定义方法

1.在res/menu下创建一个menu的样式文件.

<menu xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:android="http://schemas.android.com/apk/res/android">
    <item
        android:id="@+id/item_1"
        android:showAsAction="never"
        android:icon="@drawable/ic_plus_white_24dp"
        android:title="ITEM_1"
        />
    <item
        android:id="@+id/item_2"
        android:title="ITEM_2"
        />
</menu>

2.Activity中加载这个menu

public boolean onCreateOptionsMenu(Menu menu) {
    super.onCreateOptionsMenu(menu);
    getMenuInflater().inflate(R.menu.mainpage_menu, menu);
    return true;
}

Menu item的点击事件定义

public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId())
    {
        case R.id.item_1:
            Toast.makeText(MainActivity.this, "Menu item 1", Toast.LENGTH_SHORT).show();
            break;

        case R.id.item_2:
            Toast.makeText(MainActivity.this, "Menu item 2", Toast.LENGTH_SHORT).show();
            break;

        case R.id.item_3:
            Toast.makeText(MainActivity.this, "Menu item 3", Toast.LENGTH_SHORT).show();
            break;

    }
    return super.onOptionsItemSelected(item);
}

//动态调整菜单,比如说控制菜单项是否显示
public boolean onPrepareOptionsMenu(Menu menu) {
    MenuItem itemAddressSetDefault = menu.findItem(R.id.action_address_set_default);
    itemAddressSetDefault.setVisible(false);

    return super.onPrepareOptionsMenu(menu);
}

定时器Timer和TimerTask

在开发中我们有时会有这样的需求,即在固定的每隔一段时间执行某一个任务。比如UI上的控件需要随着时间改变,我们可以使用Java为我们提供的计时器的工具类,即Timer和TimerTask。

Timer是一个普通的类,其中有几个重要的方法;而TimerTask则是一个抽象类,其中有一个抽象方法run(),类似线程中的run()方法,我们使用Timer创建一个他的对象,然后使用这对象的schedule方法来完成这种间隔的操作。

schedule方法有三个参数

  • 第一个参数就是TimerTask类型的对象,我们实现TimerTask的run()方法就是要周期执行的一个任务;
  • 第二个参数有两种类型,第一种是long类型,表示多长时间后开始执行,另一种是Date类型,表示从那个时间后开始执行;
  • 第三个参数就是执行的周期,为long类型。

schedule方法还有一种两个参数的执行重载,第一个参数仍然是TimerTask,第二个表示为long的形式表示多长时间后执行一次,为Date就表示某个时间后执行一次。

Timer就是一个线程,使用schedule方法完成对TimerTask的调度,多个TimerTask可以共用一个Timer,也就是说Timer对象调用一次schedule方法就是创建了一个线程,并且调用一次schedule后TimerTask是无限制的循环下去的,使用Timer的cancel()停止操作。当然同一个Timer执行一次cancel()方法后,所有Timer线程都被终止。

用法:

//true 说明这个timer以daemon方式运行(优先级低,程序结束timer也自动结束)   
java.util.Timer timer = new java.util.Timer(true);  
  
TimerTask task = new TimerTask() {  
   public void run() {  
   //每次需要执行的代码放到这里面。     
   }     
};  
  
//以下是几种调度task的方法:  
  
//time为Date类型:在指定时间执行一次。  
timer.schedule(task, time);  
  
//firstTime为Date类型,period为long,表示从firstTime时刻开始,每隔period毫秒执行一次。  
timer.schedule(task, firstTime, period);     
  
//delay 为long类型:从现在起过delay毫秒执行一次。  
timer.schedule(task, delay);  
  
//delay为long,period为long:从现在起过delay毫秒以后,每隔period毫秒执行一次。  
timer.schedule(task, delay, period);  

示例:

import android.app.Activity;  
import android.os.Bundle;  
import android.os.Handler;  
import android.os.Message;  
  
import java.util.Timer;  
import java.util.TimerTask;  
  
public class TimerTaskActivity extends Activity {  
  
    private Timer mTimer;  
  
    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        // init timer  
        mTimer = new Timer();  
        // start timer task  
        setTimerTask();  
    }  
  
    @Override  
    protected void onDestroy() {  
        super.onDestroy();  
        // cancel timer  
        mTimer.cancel();  
    }  
  
    private void setTimerTask() {  
        mTimer.schedule(new TimerTask() {  
            @Override  
            public void run() {  
                Message message = new Message();  
                message.what = 1;  
                doActionHandler.sendMessage(message);  
            }  
        }, 1000, 1000/* 表示1000毫秒之後,每隔1000毫秒執行一次 */);  
    }  
  
    /** 
     * do some action 
     */  
    private Handler doActionHandler = new Handler() {  
        @Override  
        public void handleMessage(Message msg) {  
            super.handleMessage(msg);  
            int msgId = msg.what;  
            switch (msgId) {  
                case 1:  
                    // do some action  
                    break;  
                default:  
                    break;  
            }  
        }  
    };  
}  

Android平台从其它线程访问主线程主要有以下几种方式:

  • Activity.runOnUiThread(Runnable)
  • View.post(Runnable)
  • View.postDelayed(Runnable, long)
  • Handler

View.post

View.post方法的主要作用是保证指定的任务顺序执行,post方法的执行时间是在view的layout过程结束之后. post方法会创建一个Runnable,但还是在主线程/UI线程中执行的.
所以可以直接调用UI的元素
主要用途是获取view的宽高尺寸,如果直接获取的话值为0,因为measure过程还没有完成

msgTextView.setText(String.valueOf(costomWidget.getMeasuredWidth()) + "  " + String.valueOf(costomWidget.getMeasuredHeight()));

ImageView

android:scaleType是控制图片如何resized/moved来匹对ImageView的size。

ImageView.ScaleType / android:scaleType值的意义区别:

  • CENTER 按图片的原来size居中显示,当图片长/宽超过View的长/宽,则截取图片的居中部分显示
  • CENTER_CROP 按比例扩大图片的size居中显示,使得图片长(宽)等于或大于View的长(宽)
  • CENTER_INSIDE 将图片的内容完整居中显示,通过按比例缩小或原来的size使得图片长/宽等于或小于View的长/宽
  • FIT_CENTER 把图片按比例扩大/缩小到View的宽度,居中显示
  • FIT_END 把图片按比例扩大/缩小到View的宽度,显示在View的下部分位置
  • FIT_START 把图片按比例扩大/缩小到View的宽度,显示在View的上部分位置
  • FIT_XY 把图片不按比例扩大/缩小到View的大小显示
  • MATRIX 用矩阵来绘制,动态缩小放大图片来显示。

Bitmap的四种属性,也就是指图片压缩质量参数

  • public static final Bitmap.Config ALPHA_8
  • public static final Bitmap.Config ARGB_4444
  • public static final Bitmap.Config ARGB_8888
  • public static final Bitmap.Config RGB_565

我们知道ARGB指的是一种色彩模式,里面A代表Alpha,R表示red,G表示green,B表示blue,其实所有的可见色都是右红绿蓝组成的,所以红绿蓝又称为三原色,每个原色都存储着所表示颜色的信息值

说白了就ALPHA_8就是Alpha由8位组成 ARGB_4444就是由4个4位组成即16位, ARGB_8888就是由4个8位组成即32位, RGB_565就是R为5位,G为6位,B为5位共16位

由此可见: ALPHA_8 代表8位Alpha位图 ARGB_4444 代表16位ARGB位图 ARGB_8888 代表32位ARGB位图 RGB_565 代表8位RGB位图

位图位数越高代表其可以存储的颜色信息越多,当然图像也就越逼真。

设置位图的压缩模式

BitmapFactory.Options options = new BitmapFactory.Options();
        options2.inPreferredConfig = Bitmap.Config.RGB_565;
bm = BitmapFactory.decodeFile("filepath/file.jpg", options);

ViewPager

继承自ViewGroup, 专门用以实现左右滑动切换View的效果

使用方法

使用方法和ListView类似,先定义一个Adapter类继承PagerAdapter

public class MainpageBannerViewPagerAdapter extends PagerAdapter {
    private Context context;
    private ArrayList<SimpleDraweeView> imageViews;

    public MainpageBannerViewPagerAdapter(Context context, ArrayList<SimpleDraweeView> imageViews) {
        this.context = context;
        this.imageViews = imageViews;
    }

    @Override
    public int getCount() {
        return imageViews.size();
    }

    @Override
    public boolean isViewFromObject(View view, Object object) {
        return view == object;
    }

    @Override
    public ImageView instantiateItem(ViewGroup container, int position) {
        SimpleDraweeView imageView = imageViews.get(position);
        container.addView(imageView);
        return imageView;
    }

}

重载方法

  • getCount() 获取ViewPager实际绘制的列表项的数量
  • instantiateItem(ViewGroup container, int position) 获取当前列表项,也就是正在显示的当前页,当前ViewPager所在的位置
  • destroyItem(ViewGroup container,int position,Object object) 当前项离开屏幕时回调本方法,在本方法中需要将当前想从ViewPager中移除
  • isViewFromObject(View view,Object obj) 判断view和obj是否为同一个view

最后ViewPager关联Adapter即可

viewPager.setAdapter(goodsViewPagerAdapter);

SSL加密握手过程

  1. 客户端将它所支持的算法列表和一个用作产生密钥的随机数发送给服务器
  2. 服务器从算法列表中选择一种加密算法,并将它和一份包含服务器公用密钥的证书发送给客户端;该证书还包含了用于认证目的的服务器标识,服务器同时还提供了一个用作产生密钥的随机数
  3. 客户端验证服务器证书,如果验证失败,则停止建立SSL.验证成功进行下一步骤
  4. 客户端为本次会话生成预备主密码,并用服务器公钥加密后发送给服务器.
  5. 如果服务器要求验证客户端身份,则检查客户端的CA证书是否可信,如果不在可信列表中则结束会话.检查通过后服务器用私钥解密收到的预备主密码(pre master key),并生成本次会话的主密码(master key).客户端与服务端使用pre master key独立计算出master key.
  6. 客户端通知服务端以后发送的消息均使用master key加密.通知服务端完成这次SSL握手
  7. 服务端通知客户端完成SSL握手

Android 单元测试

工程采用MVP架构时对Presenter的接口进行测试

单元测试使用Junit,AS在创建项目时已经支持,不需要手动导入

待测试的类

public class Calculator {
    public int add(int one, int another) {
        // 为了简单起见,暂不考虑溢出等情况。
        return one + another;
    }

    public int multiply(int one, int another) {
        // 为了简单起见,暂不考虑溢出等情况。
        return one * another;
    }
}

测试类

public class CalculatorTest {
    Calculator mCalculator;

    @Before
    public void setup() {
        mCalculator = new Calculator();
    }

    @Test
    public void testAdd() throws Exception {
        int sum = mCalculator.add(1, 2);
        assertEquals(3, sum);  //为了简洁,往往会static import Assert里面的所有方法。
    }

    @Test
    public void testMultiply() throws Exception {
        int product = mCalculator.multiply(2, 4);
        assertEquals(8, product);
    }

}

方法注解

@Test 是测试方法,运行单元测试时依次执行
@Before 是每个测试方法调用之前运行的方法,一般用来进行变量初始化 @After 每个测试方法结束后调用

验证结果的方法

assertEquals(expected, actual)
验证expected的值跟actual是一样的,如果是一样的话,测试通过,不然的话,测试失败。如果传入的是object,那么这里的对比用的是equals()

assertEquals(expected, actual, tolerance)
这里传入的expected和actual是float或double类型的,大家知道计算机表示浮点型数据都有一定的偏差,所以哪怕理论上他们是相等的,但是用计算机表示出来则可能不是,所以这里运行传入一个偏差值。如果两个数的差异在这个偏差值之内,则测试通过,否者测试失败。

assertTrue(boolean condition)
验证contidion的值是true

assertFalse(boolean condition)
验证contidion的值是false

assertNull(Object obj)
验证obj的值是null

assertNotNull(Object obj)
验证obj的值不是null

assertSame(expected, actual)
验证expected和actual是同一个对象,即指向同一个对象

assertNotSame(expected, actual)
验证expected和actual不是同一个对象,即指向不同的对象

fail()
让测试方法失败


Android 内存机制

Dalvik内存堆(Heap)栈(Stack)

堆(Heap: 内存数据区,存储对象实例数据,方法内部变量(复杂类型),动态属性,由new创建的对象和数组, 栈(Stack: 内存指令区,基本类型的变量和对象的引用变量,方法内部变量(基本数据类型),静态属性,类方法,对象地址,常量

对象实例数据

实际上是保存对象实例的属性,属性的类型和对象本身的类型标记等,但是不保存实例的方法。实例的方法是属于数据指令,是保存在Stack里面,也就是上面表格里面的类方法。

对象实例在Heap中分配好以后,会在stack中保存一个4字节的Heap内存地址,用来查找对象的实例。因为在Stack里面会用到Heap的实例,特别是调用实例的时候需要传入一个this指针。

方法内部变量

类方法的内部变量分为两种情况:简单类型保存在Stack中;对象类型在Stack中保存地址,在Heap 中保存值。

非静态方法和静态方法

非静态方法有一个隐含的传入参数,这个参数是dalvik虚拟机传进去的,这个隐含参数就是对象实例在Stack中的地址指针。因此非静态方法(在Stack中的指令代码)总是可以找到自己的专用数据(在Heap 中的对象属性值)。

  当然非静态方法也必须获得该隐含参数,因此非静态方法在调用前,必须先new一个对象实例,获得Stack中的地址指针,否则dalvik虚拟机将无法将隐含参数传给非静态方法。

  静态方法没有隐含参数,因此也不需要new对象,只要class文件被ClassLoader load进入JVM的Stack,该静态方法即可被调用。所以我们可以直接使用类名调用类的方法。当然此时静态方法是存取不到Heap 中的对象属性的。

静态属性和动态属性

静态属性是保存在Stack中的,而不同于动态属性保存在Heap 中。正因为都是在Stack中,而Stack中指令和数据都是定长的,因此很容易算出偏移量,所以类方法(静态和非静态)都可以访问到类的静态属性。也正因为静态属性被保存在Stack中,所以具有了全局属性。

小结

  Java的堆是一个运行时数据区,类的(对象从中分配空间。这些对象通过new、newarray、anewarray和multianewarray等指令建立,它们不需要程序代码来显式的释放。

  堆是由垃圾回收来负责的,堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,因为它是在运行时动态分配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。

  栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本类型的变量(,int, short, long, byte, float, double, boolean, char)和对象句柄。

  对比上面的解析可以看出,其实Java处理Heap和Stack的大致原理跟C++是一样的。只是多了一个内存回收机制,让程序员不用主动调用delete释放内存。就像在C++里面,一般使用new申请的内存才会放到堆里面,而一般的临时变量都是放到栈里面去。

Android的GC如何回收内存

Android的每个应用程序都会使用一个专有的Dalvik虚拟机实例来运行,它是由Zygote服务进程孵化出来的,也就是说每个应用程序都是在属于自己的进程中运行的。

Android为不同类型的进程分配了不同的内存使用上限,如果程序在运行过程中出现了内存泄漏的而造成应用进程使用的内存超过了这个上限,则会被系统视为内存泄漏,从而被kill掉,这使得仅仅自己的进程被kill掉,而不会影响其他进程(如果是system_process等系统进程出问题的话,则会引起系统重启)。

Android的GC是自动运行的,不再被引用的对象就会被GC回收.因此对于我们已经不需要使用的对象,我们可以把它设置为null,这样当GC运行的时候,就好遍历到你这个对象已经没有引用,会自动把该对象占用的内存回收。

查看内存使用情况

adb命令

adb shell dumpsys meminfo [package name]

代码中查看

ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);

获取堆内存大小(单位M)
activityManager.getMemoryClass()
activityManager.getLargeMemoryClass()

如果在AndroidManifest.xml中的标签下添加属性android:largeHeap=“true” ,将启用大堆内存模式


viewGroup和子控件

自己定义的GridView,ListView点击时会没有反应,是因为其中的Item项中有Button,CheckBox等子控件占据了焦点事件
解决方法是在Item布局的根布局上定义android:descendantFocusability属性来控制
属性值为

  • beforeDescendants:viewgroup会优先其子类控件而获取到焦点
  • afterDescendants:viewgroup只有当其子类控件不需要获取焦点时才获取焦点
  • blocksDescendants:viewgroup会覆盖子类控件而直接获得焦点

Recycler View添加滚动到底部自动加载更多

RecyclerView没有提供滚动到底部加载更多的功能,需要自己实现

自定义一个OnScrollListener

public abstract class EndlessRecyclerOnScrollListener extends RecyclerView.OnScrollListener {
    public static String TAG = EndlessRecyclerOnScrollListener.class.getSimpleName();

    private int previousTotal = 0; // The total number of items in the dataset after the last load
    private boolean loading = true; // True if we are still waiting for the last set of data to load.
    private int visibleThreshold = 5; // The minimum amount of items to have below your current scroll position before loading more.
    int firstVisibleItem, visibleItemCount, totalItemCount;

    private int current_page = 1;

    private GridLayoutManager layoutManager;

    public EndlessRecyclerOnScrollListener(GridLayoutManager layoutManager) {
        this.layoutManager = layoutManager;
    }

    @Override
    public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
        super.onScrolled(recyclerView, dx, dy);

        visibleItemCount = recyclerView.getChildCount();
        totalItemCount = layoutManager.getItemCount();
        firstVisibleItem = layoutManager.findFirstVisibleItemPosition();

        if (loading) {
            if (totalItemCount > previousTotal) {
                loading = false;
                previousTotal = totalItemCount;
            }
        }
        if (!loading && (totalItemCount - visibleItemCount)
                <= (firstVisibleItem + visibleThreshold)) {
            // End has been reached

            // Do something
            current_page++;

            onLoadMore(current_page);

            loading = true;
        }
    }

    public abstract void onLoadMore(int current_page);
}

GridView和RecyclerView只有LayoutManager不同

public abstract class EndlessGridRecyclerOnScrollListener extends
        RecyclerView.OnScrollListener {

    private int previousTotal = 0;
    private boolean loading = true;
    int firstVisibleItem, visibleItemCount, totalItemCount;

    private int currentPage = 1;

    private GridLayoutManager gridLayoutManager;

    public EndlessGridRecyclerOnScrollListener(
            GridLayoutManager gridLayoutManager) {
        this.gridLayoutManager = gridLayoutManager;
    }

    @Override
    public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
        super.onScrolled(recyclerView, dx, dy);

        CommonUtils.log(String.valueOf(dx) + " " + String.valueOf(dy) );

        visibleItemCount = recyclerView.getChildCount();
        totalItemCount = gridLayoutManager.getItemCount();
        firstVisibleItem = gridLayoutManager.findFirstVisibleItemPosition();


        if (loading) {
            if (totalItemCount > previousTotal) {
                loading = false;
                previousTotal = totalItemCount;
            }
        }
        if (!loading && (totalItemCount - visibleItemCount) <= firstVisibleItem) {
            currentPage++;
            onLoadMore(currentPage);
            loading = true;
        }
    }

    public abstract void onLoadMore(int currentPage);
}

在Activity中使用

listView.addOnScrollListener(new EndlessGridRecyclerOnScrollListener(layoutManager) {
    @Override
    public void onLoadMore(int currentPage) {
        loadPage();
    }
});

如果RecyclerView包裹在NestedScrollView中,添加ScrollListener必须添加到NestedScrollView上,不然会出现RecyclerView不响应事件,无限循环加载的问题.

<android.support.v4.widget.NestedScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    app:layout_behavior=    "@string/appbar_scrolling_view_behavior"
    tools:context=".module.branddetail.BrandDetailActivity"
    tools:showIn="@layout/activity_brand_detail"
    android:id="@+id/scroll_view">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/brand_goods_list"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_alignParentBottom="true"
        android:layout_alignParentLeft="true"
        android:layout_alignParentStart="true"
        android:nestedScrollingEnabled="false"
        />

</android.support.v4.widget.NestedScrollView>

继承NestedScrollView.OnScrollChangeListener

public abstract class EndlessParentScrollListener implements NestedScrollView.OnScrollChangeListener {
    // The current offset index of data you have loaded
    private int currentPage = 0;
    // The total number of items in the dataset after the last load
    private int previousTotalItemCount = 0;
    // True if we are still waiting for the last set of data to load.
    private boolean loading = true;
    // Sets the starting page index
    private int startingPageIndex = 0;
    // The minimum amount of pixels to have below your current scroll position
    // before loading more.
    private int visibleThresholdDistance = 300;

    GridLayoutManager mLayoutManager;

    public EndlessParentScrollListener(GridLayoutManager layoutManager) {
        this.mLayoutManager = layoutManager;
    }

    @Override
    public void onScrollChange(NestedScrollView scrollView, int x, int y, int oldx, int oldy) {
        // We take the last son in the scrollview
        View view = scrollView.getChildAt(scrollView.getChildCount() - 1);
        int distanceToEnd = (view.getBottom() - (scrollView.getHeight() + scrollView.getScrollY()));

        int totalItemCount = mLayoutManager.getItemCount();
        // If the total item count is zero and the previous isn't, assume the
        // list is invalidated and should be reset back to initial state
        if (totalItemCount < previousTotalItemCount) {
            this.currentPage = this.startingPageIndex;
            this.previousTotalItemCount = totalItemCount;
            if (totalItemCount == 0) {
                this.loading = true;
            }
        }

        // If it’s still loading, we check to see if the dataset count has
        // changed, if so we conclude it has finished loading and update the current page
        // number and total item count.
        if (loading && (totalItemCount > previousTotalItemCount)) {
            loading = false;
            previousTotalItemCount = totalItemCount;
        }

        // If it isn’t currently loading, we check to see if we have breached
        // the visibleThreshold and need to reload more data.
        // If we do need to reload some more data, we execute onLoadMore to fetch the data.
        // threshold should reflect how many total columns there are too
        if (!loading && distanceToEnd <= visibleThresholdDistance) {
            currentPage++;
            onLoadMore(currentPage, totalItemCount);
            loading = true;
        }
    }

    // Defines the process for actually loading more data based on page
    public abstract void onLoadMore(int page, int totalItemsCount);
}

Activity中使用

nestedScrollView.setOnScrollChangeListener(new EndlessParentScrollListener(gridLayoutManager) {
    @Override
    public void onLoadMore(int page, int totalItemsCount) {
        load();
    }
});

View注入框架ButterKnife

对一个成员变量使用@BindView注解,并传入一个View ID, ButterKnife 就能够帮你找到对应的View,并自动的进行转换
ButterKnife通过编译时对代码处理来执行View的查找,不占用性能

使用方法

导入库,添加依赖

dependencies {
implementation 'com.jakewharton:butterknife:8.8.1'
annotationProcessor 'com.jakewharton:butterknife-compiler:8.8.1'
}

Activity中使用

//声明view的时候添加@InjectView注解
@BindView(R.id.ok_btn) //控件对应的ID
Button mBtn;

@BindView(R.id.title_text)
TextView mTitleTextView;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main_activity);

    //ButterKnife初始化
    ButterKnife.inject(this);

    //这样之后就可以直接使用变量了
    mTitleTextView.setText("test");
}

Fragment的和adapter里也可以用,不过调用时要多加一个root view参数。
Fragegment使用时记得同时继承onDestroyView,并在其中将ButterKnife.reset

public class FancyFragment extends Fragment {
@BindView(R.id.button1) Button button1;
@BindView(R.id.button2) Button button2;

@Override View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View view = inflater.inflate(R.layout.fancy_fragment, container, false);
    //添加root view参数
    ButterKnife.inject(this, view);
    return view;
}
}

@Override void onDestroyView() {
    super.onDestroyView();
    ButterKnife.reset(this);
}

还能绑定String,Drawable,Color,Dimen资源

@BindString(R.string.title) String title;
@BindDrawable(R.drawable.graphic) Drawable graphic;
@BindColor(R.color.red) int red; // int or ColorStateList field
@BindDimen(R.dimen.spacer) Float spacer;

一次绑定多个资源的操作,使用@BindViews

@BindViews({ R.id.first_name, R.id.middle_name, R.id.last_name })
List<EditText> nameViews;

View绑定点击事件到方法

@OnClick(R.id.submit)
public void submit(View view) {
    // TODO submit data to server...
}

Window和WindowManager相关

WindowManagerService 就是位于 Framework 层的窗口管理服务,它的职责就是管理系统中的所有窗口。
Window是一个抽象类,具体实现是PhoneWindow,创建Window依赖于WindowManager
View,Dialog,Toast都有一个Window
WindowManger的作用是管理View,提供的方法:添加View,更新View,删除View. Android所有的View都是附加在Window上的.Window是View的直接管理者.事件分发机制是由Window传递给DecorView再向下分发到子View.

对Window的操作依赖WindowManager,WindowManager是一个接口,它的真正实现是 WindowManagerImpl 类

@Override
public void addView(View view, ViewGroup.LayoutParams params){
    mGlobal.addView(view, params, mDisplay, mParentWindow);
}

@Override
public void updateViewLayout(View view, ViewGroup.LayoutParams params){
    mGlobal.updateViewLayout(view, params);
}

@Override
public void removeView(View view){
    mGlobal.removeView(view, false);
}

WindowManagerImpl实现了WindowManager的接口方法,但转交给了WindowManagerGlobal处理,WindowManagerGlobal以工厂模式提供自己的实例.

WindowManagerGlobal内部有如下几个列表

private final ArrayList<View> mViews = new ArrayList<View>();
private final ArrayList<ViewRootImpl> mRoots = new ArrayList<ViewRootImpl>();
private final ArrayList<WindowManager.LayoutParams> mParams = new ArrayList<WindowManager.LayoutParams>();
private final ArraySet<View> mDyingViews = new ArraySet<View>();

mViews储存所有的对应View
mRoots储存所有Window对应的ViewRootImpl
mParams 所有Window所对应的布局参数
mDyingViews 正在被删除的View对象

创建Window的过程

创建Window的过程通过WindowManager的addView来实现,WindowManagerImpl实现了WindowManager的接口方法,又转交WindowManagerGlobal来处理.

WindowManagerGlobal的处理方式有以下几步

  • 创建ViewRootImpl
  • 通过ViewRootImpl来更新界面完成Window的添加过程

Window 有三种类型,分别是应用 Window、子 Window 和系统 Window。应用类 Window 对应一个 Acitivity,子 Window 不能单独存在,需要依附在特定的父 Window 中,比如常见的一些 Dialog 就是一个子 Window。系统 Window是需要声明权限才能创建的 Window,比如 Toast 和系统状态栏都是系统 Window。

一个应用存在多少个Window

View是Android 中的视图的呈现方式,但是View不能单独存在,它必须附着在Window这个抽象的概念上面,因此有视图的地方就有 Window。哪些地方有视图呢?Android可以提供视图的地方有Activity、Dialog、Toast,除此之外,还有一些依托 Window 而实现的视图,比如 PopUpWindow(自定义弹出窗口)、菜单,它们也是视图,有视图的地方就有 Window,因此 Activity、Dialog、Toast 等视图都对应着一个 Window。

Window是一个抽象类,他的实现类是PhoneWindow.

一个应用的Window数包括所有的Activity,Dialog,Toast的数量

创建Window的过程

Activity的Window创建过程

Window本质就是一块显示区域,所以关于Activity的Window创建应该发生在Activity的启动过程中.Activity最终会由 ActivityThread中的performLaunchActivity()来完成整个启动过程,在这个方法内部会通过类加载器创建Activity的实例对象,并调用其attach方法为其关联运行过程中所依赖的一系列上下文环境变量。

Activity 的 Window 创建就发生在 attach 方法里,系统会创建 Activity 所属的 Window 对象并为其设置回调接口

mWindow = PolicyManager.makeNewWindow(this);
mWindow.setCallback(this);
mWindow.setOnWindowDismissedCallback(this);
mWindow.getLayoutInflater().setPrivateFactory(this);

Window对象的创建是通过PolicyManager的makeNewWindow方法实现的,由于Activity实现了Window的Callback 接口,因此当 Window接受到外界的状态改变时就会回调Activity的方法。Callback接口中的方法很多有几个是我们非常熟悉的,如onAttachedToWindow、onDetachedFromWindow、dispatchTouchEvent等等。

PolicyManager主要用于创建Window类、LayoutInflater类和WindowManagerPolicy类

再回到Window的创建,可以看到Activity的Window是通过PolicyManager的一个工厂方法来创建的,但是在PolicyManager的实际调用中,PolicyManager的真正实现是Policy类,Policy类中的makeNewWindow方法的实现如下:

public Window makeNewWindow(Context context){
    return new PhoneWindow(context);
}

makeNewWindow生成的就是Window的子类PhoneWindow.到这里 Window 以及创建完成了,下面分析 Activity 的视图是怎么附属到 Window 上的,而 Activity 的视图由 setContentView 提供,所以从 setContentView 入手,它的源码如下:

public void setContentView(int layoutResID){
    getWindow().setContentView(layoutResID);
    initWindowDecorActionBar();
}

可以看到,Activity 将具体实现交给了 Window,而 Window 的具体实现是 PhoneWindow,所以只需要看 PhoneWindow 的相关逻辑即可,它的处理步骤如下:

  1. 创建DecorView,DecorView 是 Activity 中的顶级 View,是一个 FrameLayout,一般来说它的内部包含标题栏和内容栏.内容栏是一定存在的,并且有固定的 id:”android.R.id.content”,在 PhoneWindow 中,通过 generateDecor 方法创建 DecorView,通过 generateLayout 初始化主题有关布局。

  2. 将 View 添加到 DecorView 的 mContentParent 中,就是把布局xml文件添加到 DecorView 的 mContentParent 中

  3. 回调 Activity 的 onContentChanged 方法通知 Activity 视图已经发生改变

Dialog的Window创建过程

  1. 创建Window

Dialog中Window同样是通过PolicyManager的makeNewWindow方法来完成的,创建后的对象也是PhoneWindow。

  1. 初始化DecorView并将Dialog的视图添加到DecorView中

这个过程也和 Activity 类似,都是通过 Window 去添加指定布局文件:

public void setContentView(int layoutResID){
mWindow.setContentView(layoutResID);
}
  1. 将DecorView添加到Window中并显示

Toast的Window创建过程

Toast也是基于Window来实现的,但是由于Toast具有定时取消这一功能,所以系统采用了Handler。


Android Loader 相关

Android的装载器(loader)是从Android 3.0新引入的API , 主要完成单线程耗时数据异步装载功能,并在数据有更新自动通知UI刷新的作用。业内也叫加载器,装载机。

Loader用途

Loader一般用在Activity和fragment异步加载数据,无需重新启动一个线程来执行数据加载,异步加载可以用asyncTask, 但是loader自带数据结果监听机制,可以方便优雅的进行UI更新。

作用和优点:

  • 提供异步加载数据功能;
  • 对数据源变化进行监听,实时更新数据;
  • 在Activity配置发生变化(如横竖屏切换)时不避免数据重复加载;
  • 适用于任何Activity和Fragment;

异步处理AsyncTask和Handler的优缺点

ASyncTask

实现机制是线程池进行调度,线程池有5个核心工作线程.最多能产生128个工作线程 线程等待队列是10个

线程池中的工作线程少于5个时,创建新的工作线程执行异步任务
已经有5个线程的时候再添加任务就会被放到缓冲队列
缓冲队列满了之后再添加任务,线程池就会新开工作线程执行异步任务
线程池中已经有128个线程,缓冲队列已满,再添加任务就会被拒绝,抛出RejectedExecutionException

优点: 实现了异步任务的封装,结构清晰,只要实现postExecute,preExecute,doInBackground这几个方法即可
缺点: 并行工作的线程数量有限,并发数量多的情况下效率低.比如列表加载大量图片的情况下

Handler

实现异步的流程是主线程启动子线程,运行并生成Message-Looper获取Messagebing传递给Handler,Handler获取Looper中的Message,并进行UI变更.ASyncTask用到了Handler机制.在子线程任务完成之后向UI线程返回结果.

数据简单的时候用ASyncTask,数据结构复杂的情况下用Handler.


Android 底部控件,随软键盘弹出,跟着上移

androidManifest.xml activity添加属性

<activity android:name=".MainActivity"
    ...
    android:windowSoftInputMode="adjustResize|stateHidden">
</activity>

android:windowSoftInputMode属性

activity主窗口与软键盘的交互模式,可以用来避免输入法面板遮挡问题
这个属性能影响两件事情:

  1. 当有焦点产生时,软键盘是隐藏还是显示
  2. 是否减少活动主窗口大小以便腾出空间放软键盘

各值的含义:

【A】stateUnspecified:软键盘的状态并没有指定,系统将选择一个合适的状态或依赖于主题的设置

【B】stateUnchanged:当这个activity出现时,软键盘将一直保持在上一个activity里的状态,无论是隐藏还是显示

【C】stateHidden:用户选择activity时,软键盘总是被隐藏

【D】stateAlwaysHidden:当该Activity主窗口获取焦点时,软键盘也总是被隐藏的

【E】stateVisible:软键盘通常是可见的

【F】stateAlwaysVisible:用户选择activity时,软键盘总是显示的状态

【G】adjustUnspecified:默认设置,通常由系统自行决定是隐藏还是显示

【H】adjustResize:该Activity总是调整屏幕的大小以便留出软键盘的空间

【I】adjustPan:当前窗口的内容将自动移动以便当前焦点从不被键盘覆盖和用户能总是看到输入内容的部分