或許這個(gè)demo讓你了解Android四大組件

welcome.png

前言
==
粉絲:按照國(guó)際慣例,這里需要放的是一個(gè)效果展示的gif扒凇略板?
很有原則的我:效果展示,恩慈缔?不存在的63啤(我會(huì)告訴你我手機(jī)沒(méi)有sim卡)
粉絲:那我要退訂?
很有原則的我:退訂,恩哼瓤檐,威脅我赂韵,沒(méi)用的。今天說(shuō)什么我都不會(huì)做效果展示的距帅。
粉絲:此生不再相見右锨,粉轉(zhuǎn)黑。
很有原則我的:恩碌秸,再見绍移。(我們高冷的氣質(zhì)必須保持)
告一段落。

后續(xù)

在我5000萬(wàn)粉絲的苦苦哀求下讥电,你知道我的蹂窖,善良又心軟。所以就勉強(qiáng)答應(yīng)了做一個(gè)效果展示圖恩敌。我沒(méi)有sim卡怎么做的效果展示瞬测,你猜啊。
直接上效果圖纠炮。

4.gif

好吧 T绿恕!恢口!我承認(rèn)孝宗,這可能是最糟糕的展示圖,由于各種原因耕肩,用了真機(jī)模擬因妇,所以錄制卡頓。大家別噴猿诸,要臉婚被。不過(guò)最重要的是你們可以運(yùn)行起來(lái)是吧!那下面老司機(jī)就開始開車了J崴洹V沸尽!滴滴滴窜觉,請(qǐng)系好安全帶是复。

簡(jiǎn)介

這是一個(gè)小應(yīng)用的詳解,這個(gè)應(yīng)用可以添加手機(jī)黑名單竖螃,攔截手機(jī)黑名單的來(lái)電淑廊。通過(guò)這個(gè)小demo,我們可以對(duì)Android四大組件的應(yīng)用場(chǎng)景有個(gè)具體的了解特咆,可以說(shuō)是一個(gè)不錯(cuò)的練手項(xiàng)目季惩。

下面給出下載地址:
1.GitHub下載地址:
https://github.com/Simon986793021/SafeCall

2.CSDN下載地址:
http://download.csdn.net/detail/simon_crystin/9809978

概述

下面先來(lái)講下Android四大組件在demo中的應(yīng)用:

Activity:這個(gè)大家肯定是比較熟悉的录粱,也是大家比較常用的。主要是用來(lái)控制加載頁(yè)面和一些操作画拾。

Service:我們的Service會(huì)一直在后臺(tái)運(yùn)行啥繁,如果是黑名單中的號(hào)碼,將會(huì)直接攔截青抛。有兩種啟動(dòng)方式旗闽,這里不具體分析。

ContentProvider:內(nèi)容提供者蜜另,它的作用主要是對(duì)應(yīng)用之間 的數(shù)據(jù)操作提供接口适室,不懂的可以看下我的這篇BLOG:
http://blog.csdn.net/Simon_Crystin/article/details/68517050
這個(gè)demo中的應(yīng)用主要是用來(lái)對(duì)本地?cái)?shù)據(jù)庫(kù)進(jìn)行操作,這里本來(lái)是是不需要用到ContentProvider举瑰,但是要強(qiáng)行裝逼捣辆,我這里用了。這個(gè)是不需要糾結(jié)的此迅。

BroadcastReceiver:這里主要是在開機(jī)時(shí)發(fā)送一個(gè)廣播汽畴,在接收廣播的時(shí)候開啟一個(gè)服務(wù)。對(duì)廣播接收者不熟悉的也可以看下這片Blog:
http://blog.csdn.net/Simon_Crystin/article/details/68062838

除了這些:里面還涉及了許多知識(shí)點(diǎn)耸序,比如:
1:正則表達(dá)式的用法忍些,用來(lái)匹配手機(jī)號(hào)碼;
2:aidl的使用坎怪,用來(lái)攔截來(lái)電罢坝;
3:各種權(quán)限的使用;
4:adapter的使用芋忿;
5:dialog的自定義炸客;
6:Toast的自定義等等疾棵,還有許多小技巧戈钢。

實(shí)現(xiàn)

1.添加黑名單。
2.判斷如果是黑名單是尔,就對(duì)其攔截殉了。

添加黑名單:

往數(shù)據(jù)庫(kù)里面添加黑名單,手機(jī)號(hào)碼拟枚,用的是ContentProvider進(jìn)行添加的(當(dāng)然薪铜,這里是完全沒(méi)有必要用這個(gè)的)

第一步:
新建一個(gè)MyContentProvider繼承自ContentProvider,并重寫其中的方法:

package com.wind.safecall.contentprovider;

import android.content.ContentProvider;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.Context;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import android.net.Uri;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;

/**
 * Created by zhangcong on 2017/4/8.
 */

public class MyContentProvider extends ContentProvider {
    private MyOpenHelper myOpenHelper;
    private String DB_name="blackname";//數(shù)據(jù)庫(kù)名
    private String Table_name="blacknametable";//數(shù)據(jù)表名
    private SQLiteDatabase sqLiteDatabase;
    private static UriMatcher uriMatcher;
    public static  final String AUTHORITY="blacknum";
    public static final  Uri uri =Uri.parse("content://blacknum/path_simon");

    // 注冊(cè)該內(nèi)容提供者匹配的uri
    static {
        uriMatcher=new UriMatcher(UriMatcher.NO_MATCH);//Creates the root node of the URI tree.
        uriMatcher.addURI(AUTHORITY,"path_simon",1);// 代表當(dāng)前表中的所有的記錄,第三個(gè)參數(shù)必須為正數(shù)
        uriMatcher.addURI(AUTHORITY,"path_simon/1",2);// 代表當(dāng)前表中的某條特定的記錄恩溅,記錄id便是#處得數(shù)字
    }
    //數(shù)據(jù)表列名映射
    private static  final  String blacknum="blacknum";
    private static final String _id = "id";

    @Override
    public boolean onCreate() {
        try {
            myOpenHelper=new MyOpenHelper(getContext(),DB_name,null,1);
        }
        catch (Exception e)
        {
            return  false;
        }
        return true;
    }

    @Nullable
    @Override
    public Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder) {
        Cursor cursor = null;
        sqLiteDatabase = myOpenHelper.getReadableDatabase();
        int code = uriMatcher.match(uri);//addURI()傳的第三個(gè)參數(shù)
        switch (code) {
            case 1:
                cursor = sqLiteDatabase.query(Table_name, projection, selection,
                        selectionArgs, null, null, sortOrder);
                break;
            case 2:
                // 從uri中解析出ID
                long id = ContentUris.parseId(uri);
                selection = (selection == null || "".equals(selection.trim())) ? _id
                        + "=" + id
                        : selection + " and " + _id + "=" + id;
                cursor = sqLiteDatabase.query(Table_name, projection, selection,
                        selectionArgs, null, null, sortOrder);
                break;
            default:
                throw new IllegalArgumentException("參數(shù)錯(cuò)誤");
        }

        return cursor;
    }

    @Nullable
    @Override
    public String getType(@NonNull Uri uri) {
        return null;
    }

    @Nullable
    @Override
    public Uri insert(@NonNull Uri uri, @Nullable ContentValues values) {
        sqLiteDatabase=myOpenHelper.getWritableDatabase();
        sqLiteDatabase = myOpenHelper.getWritableDatabase();
        int code = uriMatcher.match(uri);//前面addURI傳的第三個(gè)參數(shù)
        switch (code) {
            case 1:
                sqLiteDatabase.insert(Table_name, blacknum, values);
                break;
            case 2:
                long id = sqLiteDatabase.insert(Table_name, blacknum, values);
                // withAppendId將id添加到uri的最后
                ContentUris.withAppendedId(uri, id);
                break;
            default:
                throw new IllegalArgumentException("異常參數(shù)");
        }

        return uri;
    }

    @Override
    public int delete(@NonNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs) {
        return 0;
    }

    @Override
    public int update(@NonNull Uri uri, @Nullable ContentValues values, @Nullable String selection, @Nullable String[] selectionArgs) {
        return 0;
    }

    private class MyOpenHelper extends SQLiteOpenHelper
    {

        public MyOpenHelper(Context context, String name, SQLiteDatabase.CursorFactory factory, int version) {
            super(context, name, factory, version);
        }

        @Override
        public void onCreate(SQLiteDatabase db) {
            /**
             * 官方解釋
             * Called when the database is created for the first time. This is where the
             * creation of tables and the initial population of the tables should happen.
             *
             * @param db The database.
             */
            String sql = " create table if not exists " + Table_name
                    + "(blacknum  varchar(20),id INTEGER)";
            db.execSQL(sql);
        }

        @Override
        public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

        }
    }
}

第二步:在你的監(jiān)聽事件下執(zhí)行這個(gè)方法:

 private void insertData(String blacknum) {
        ContentValues contentValues=new ContentValues();
        contentValues.put("blacknum",blacknum);//與數(shù)據(jù)庫(kù)字段對(duì)應(yīng)隔箍,第二個(gè)參數(shù)是你添加的給名單
        Uri uri=getContentResolver().insert(MyContentProvider.uri,contentValues);
        Utils.showToast(uri.toString(),MainActivity.this);

    }

第三步:查詢你所添加的黑名單:
這里分了兩個(gè)步驟:
查詢黑名單和顯示黑名單:

查詢黑名單

private void queryBlackNum() {
        String array[]={"blacknum"};
        Cursor cursor=getContentResolver().query(MyContentProvider.uri,array,null,null,null);
        int blacknumindex=cursor.getColumnIndex("blacknum");
        Log.i(">>>",blacknumindex+"");
        cursor.moveToFirst();
        ArrayList<String> list=new ArrayList<>();
        while (!cursor.isAfterLast())
        {
            String blacknum=cursor.getString(blacknumindex);
            list.add(blacknum);
            cursor.moveToNext();
            Log.i(">>>",">>>>");
            Log.i("Simon",list.toString());
        }
        Intent intent=new Intent(MainActivity.this,BlackNumActivity.class);
        intent.putStringArrayListExtra("list", list);
//        Bundle bundle=new Bundle();
//        bundle.putSerializable("list", (Serializable) list);
//        intent.putExtras(bundle);
        startActivity(intent);


    }

在另一個(gè)Activity顯示黑名單:

package com.wind.safecall.activity;

import android.app.Activity;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.util.Log;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.TextView;

import com.wind.safecall.R;

import java.util.ArrayList;
import java.util.List;


/**
 * Created by zhangcong on 2017/4/10.
 */

public class BlackNumActivity extends Activity{
    private TextView activitytitle;
    private TextView back;
    private StringBuffer sb;
    private ListView listview;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_black_num);
        activitytitle= (TextView) findViewById(R.id.tv_activity_toolbar_center);
        activitytitle.setText("黑名單");

        listview= (ListView) findViewById(R.id.lv_black_num);

        back= (TextView) findViewById(R.id.tv_back);
        back.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                finish();
            }
        });
        showBlackNum();
    }

    private void showBlackNum() {
        ArrayList<String> list = this.getIntent().getStringArrayListExtra("list");
        Log.i(">>>>>", list.toString());
        if (list != null) {
            listview.setAdapter(new ArrayAdapter<>(BlackNumActivity.this, R.layout.item_black_num, R.id.tv_black_num, list));
        }
    }
}

到這里,黑名單就添加完成脚乡。

接下來(lái)就是對(duì)其進(jìn)行攔截了:
攔截黑名單來(lái)電:

第一步:新建一個(gè)BlackNumService繼承自Service蜒滩,在這里面判斷是不是黑名單,是的話對(duì)其攔截:
這里需要補(bǔ)充一點(diǎn)的是:由于我們是直接攔截的來(lái)電,相當(dāng)于是跨進(jìn)程間的通信俯艰,這個(gè)時(shí)候aidl就登場(chǎng)了捡遍,好在Android有這些接口:
因?yàn)橄旅娴拇a會(huì)調(diào)用這些接口,所以我們提交準(zhǔn)備好:

1.在根目錄新建一個(gè)android.telephony的包竹握,在里面新建一個(gè)aidl文件画株,命名為NeighboringCellInfo;
里面的內(nèi)容為:

// NeighboringCellInfo.aidl
package android.telephony;

// Declare any non-default types here with import statements


parcelable NeighboringCellInfo;

2.在根目錄下新建一個(gè)com.android.internal.telephony的包啦辐,在里面新建一個(gè)aidl文件谓传,命名為ITelephony;里面的內(nèi)容為:

package com.android.internal.telephony;
interface ITelephony{
    boolean endCall();
    void answerRingingCall();
}

這里的包名和文件名一定要一致昧甘,通過(guò)包名識(shí)別來(lái)電的良拼,再進(jìn)行通信的

package com.wind.safecall.service;

import android.app.Notification;
import android.app.NotificationManager;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.database.Cursor;
import android.net.Uri;
import android.os.IBinder;
import android.support.annotation.Nullable;
import android.telephony.PhoneStateListener;
import android.telephony.TelephonyManager;
import android.util.Log;

import com.android.internal.telephony.ITelephony;
import com.wind.safecall.contentprovider.MyContentProvider;

import org.w3c.dom.ls.LSInput;

import java.lang.reflect.Method;
import java.util.ArrayList;

/**
 * Created by zhangcong on 2017/4/10.
 */

public class BlackNumService extends Service {
    private TelephonyManager tm;
    private MyPhoneStateListener listener;
    private NotificationManager nm;
    @Override
    public void onCreate() {
        super.onCreate();
        tm= (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
        listener=new MyPhoneStateListener();
        tm.listen(listener,PhoneStateListener.LISTEN_CALL_STATE);
       // nm= (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    }
    private final class MyPhoneStateListener extends PhoneStateListener{
        //private long startTime = 0;
        @Override
        public void onCallStateChanged(int state, String incomingNumber) {
            // TODO Auto-generated method stub
            super.onCallStateChanged(state, incomingNumber);
            switch (state) {
                case TelephonyManager.CALL_STATE_RINGING:
                    String array[]={"blacknum"};
                    Cursor cursor=getContentResolver().query(MyContentProvider.uri,array,null,null,null);
                    int blacknumindex=cursor.getColumnIndex("blacknum");
                    Log.i(">>>",blacknumindex+"");
                    cursor.moveToFirst();
                    ArrayList<String> list=new ArrayList<>();
                    while (!cursor.isAfterLast())
                    {
                        String blacknum=cursor.getString(blacknumindex);
                        list.add(blacknum);
                        cursor.moveToNext();
                        Log.i(">>>",">>>>");
                        Log.i("Simon",list.toString());
                    }
                    if (list!=null&&list.contains(incomingNumber))
                    {
                        endCall();
                        return;
                    }
                    //判斷來(lái)電黑名單是否開啟
//                    boolean isblackstart = sp.getBoolean("isblacknumber", false);
//                    if(isblackstart){
//                        boolean isBlackNumber = blackNumberDao.isBlackNumber(incomingNumber);
//                        if(isBlackNumber){
//
//                        }


                    //startTime = System.currentTimeMillis();

                    break;
                case TelephonyManager.CALL_STATE_OFFHOOK:
                    break;
                case TelephonyManager.CALL_STATE_IDLE:
                    break;

                default:
                    break;
            }
        }

    }
    //掛斷電話
    private void endCall(){
        try {
            Class<?> clazz = Class.forName("android.os.ServiceManager");
            Method method = clazz.getMethod("getService", String.class);
            IBinder ibinder = (IBinder) method.invoke(null, Context.TELEPHONY_SERVICE);
            ITelephony iTelephony = ITelephony.Stub.asInterface(ibinder);
            iTelephony.endCall();

        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
}

第二步:在廣播中或者應(yīng)用中開啟服務(wù):

1.在應(yīng)用中開啟服務(wù):

 /*
    啟動(dòng)服務(wù)
     */
    private void stService() {
        Intent intent=new Intent(MainActivity.this, BlackNumService.class);
        startService(intent);
    }

你可以在mainactivity中的oncreate方法中調(diào)用這個(gè)方法,只要這個(gè)進(jìn)程不被殺死充边,這個(gè)服務(wù)也就一致存在庸推,這是startService的性質(zhì)。

2.在開機(jī)廣播中開啟服務(wù):有些手機(jī)是接收不到廣播的:

需要新建一個(gè)BootCompletedReceiver繼承自BroadcastReceiver:

package com.wind.safecall.broadcastreceiver;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

import com.wind.safecall.service.BlackNumService;


/**
 * Created by zhangcong on 2017/4/10.
 */

public class BootCompletedReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent){
        Log.i(">>>>>>>>","已經(jīng)開機(jī)");
        Intent intent1=new Intent(context, BlackNumService.class);
        context.startService(intent1);
    }
}

這樣浇冰,整個(gè)流程就結(jié)束了贬媒,最后貼出所有的權(quán)限和注冊(cè):

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.wind.safecall">

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
    <uses-permission android:name="android.permission.RECEIVE_SMS"/>
    <uses-permission android:name="android.permission.READ_PHONE_STATE"/>
    <!--添加可以向外撥打電話的權(quán)限  -->
    <uses-permission android:name="android.permission.CALL_PHONE"></uses-permission>
    <permission android:name="blacknum.permission"
        android:protectionLevel="normal"/>
    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <!-- 開機(jī)廣播 -->
        <receiver android:name=".broadcastreceiver.BootCompletedReceiver">
            <intent-filter >
                <action android:name="android.intent.action.BOOT_COMPLETED"/>
                <category android:name="android.intent.category.HOME" />
            </intent-filter>
        </receiver>
        <service android:name=".service.BlackNumService"/>
        <provider
            android:authorities="blacknum"
            android:name=".contentprovider.MyContentProvider"
            android:permission="blacknum.permission"
            />
        <activity
            android:name=".activity.MainActivity"
            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=".activity.PhoneCallActivity"
            android:screenOrientation="portrait"/>
        <activity android:name=".activity.BlackNumActivity"
            android:screenOrientation="portrait"/>
    </application>

</manifest>

最后再次給出github地址:
https://github.com/Simon986793021/SafeCall

大家覺(jué)得有點(diǎn)幫助可以來(lái)一份star嗎?

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末肘习,一起剝皮案震驚了整個(gè)濱河市际乘,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌漂佩,老刑警劉巖脖含,帶你破解...
    沈念sama閱讀 221,576評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異投蝉,居然都是意外死亡养葵,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,515評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門瘩缆,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)关拒,“玉大人,你說(shuō)我怎么就攤上這事庸娱∽虐恚” “怎么了?”我有些...
    開封第一講書人閱讀 168,017評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵熟尉,是天一觀的道長(zhǎng)归露。 經(jīng)常有香客問(wèn)我,道長(zhǎng)斤儿,這世上最難降的妖魔是什么剧包? 我笑而不...
    開封第一講書人閱讀 59,626評(píng)論 1 296
  • 正文 為了忘掉前任腮考,我火速辦了婚禮,結(jié)果婚禮上玄捕,老公的妹妹穿的比我還像新娘踩蔚。我一直安慰自己,他們只是感情好枚粘,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,625評(píng)論 6 397
  • 文/花漫 我一把揭開白布馅闽。 她就那樣靜靜地躺著,像睡著了一般馍迄。 火紅的嫁衣襯著肌膚如雪福也。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,255評(píng)論 1 308
  • 那天攀圈,我揣著相機(jī)與錄音暴凑,去河邊找鬼。 笑死赘来,一個(gè)胖子當(dāng)著我的面吹牛现喳,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播犬辰,決...
    沈念sama閱讀 40,825評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼嗦篱,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了幌缝?” 一聲冷哼從身側(cè)響起灸促,我...
    開封第一講書人閱讀 39,729評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎涵卵,沒(méi)想到半個(gè)月后浴栽,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,271評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡轿偎,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,363評(píng)論 3 340
  • 正文 我和宋清朗相戀三年典鸡,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片贴硫。...
    茶點(diǎn)故事閱讀 40,498評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡椿每,死狀恐怖伊者,靈堂內(nèi)的尸體忽然破棺而出英遭,到底是詐尸還是另有隱情,我是刑警寧澤亦渗,帶...
    沈念sama閱讀 36,183評(píng)論 5 350
  • 正文 年R本政府宣布挖诸,位于F島的核電站,受9級(jí)特大地震影響法精,放射性物質(zhì)發(fā)生泄漏多律。R本人自食惡果不足惜痴突,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,867評(píng)論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望狼荞。 院中可真熱鬧辽装,春花似錦、人聲如沸相味。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,338評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)丰涉。三九已至拓巧,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間一死,已是汗流浹背肛度。 一陣腳步聲響...
    開封第一講書人閱讀 33,458評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留投慈,地道東北人承耿。 一個(gè)月前我還...
    沈念sama閱讀 48,906評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像伪煤,于是被迫代替她去往敵國(guó)和親瘩绒。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,507評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,282評(píng)論 25 707
  • HandlerThread是一個(gè)Android 已封裝好的輕量級(jí)異步類带族。HandlerThread本質(zhì)上是一個(gè)線程...
    kjy_112233閱讀 1,298評(píng)論 0 9
  • 前幾天整理了Java面試題集合,今天再來(lái)整理下Android相關(guān)的面試題集合.如果你希望能得到最新的消息,可以關(guān)注...
    Boyko閱讀 3,641評(píng)論 8 135
  • 明天正式去簽約了锁荔,,并不是很開心蝙砌,今天太累了阳堕,天氣悶悶的,坐公車一身汗择克,覺(jué)得對(duì)待工作也有點(diǎn)隨便恬总,今天發(fā)現(xiàn)了許多錯(cuò)誤...
    Oxygen大一個(gè)閱讀 173評(píng)論 0 0
  • 辛年僅為重春度, 家老屈歲終凌寒肚邢。 返鄉(xiāng)道軌駛不盡壹堰, 未見遺容淚已干。 懷念已故親人
    匠門裝飾周佳偉閱讀 132評(píng)論 0 0