系统应用闹钟之TabLayout的使用

此篇文章记录一些关于安卓原生应用闹钟里面TabLayout的使用,没见过用这个组件的源代码,想从它里面学习学习。就当是见见世面,顺便写个笔记。

先从印象到布局

原生闹钟给我的印象非常好,有种说不出的美感。在改工作上的问题时,顺便看它的样式,大致就是一个TabLayout加上四个fragment

首先是头部的四个图标,是一个嵌在了Toolbar中的TabLayoutToolbar也是一个ViewGroup,在它的里面放一个其他的控件也是理所当然,只是以前没试过。第一次看到这个写法,感觉非常好。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
<android.support.design.widget.AppBarLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@null"
app:elevation="0dp">
<android.support.v7.widget.Toolbar
android:id="@+id/toolbar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:contentInsetStart="0dp"
tools:ignore="RtlSymmetry">
<android.support.design.widget.TabLayout
android:id="@+id/tabs"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:tabGravity="fill"
app:tabIndicatorColor="@android:color/transparent"
app:tabMaxWidth="0dp"
app:tabMode="fixed"
app:tabPaddingEnd="0dp"
app:tabPaddingStart="0dp" />
</android.support.v7.widget.Toolbar>

然后在Toolbar下面就是一个ViewPager,存放在一个FrameLayout中。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layout_behavior="@string/appbar_scrolling_view_behavior">

<android.support.v4.view.ViewPager
android:id="@+id/desk_clock_pager"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:importantForAccessibility="no"
android:saveEnabled="false" />

<include layout="@layout/drop_shadow" />

</FrameLayout>

这样基本上大致的框架就出来了。接着看对它的初始化:
基本步骤和网上的教程是差不多的,只是其中的UiDataModel感觉非常有意思,感觉它几乎无所不能,用到的任何东西都可以往里面取。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
// Create the tabs that make up the user interface.
mTabLayout = (TabLayout) findViewById(R.id.tabs);
final int tabCount = UiDataModel.getUiDataModel().getTabCount();
final boolean showTabLabel = getResources().getBoolean(R.bool.showTabLabel);
final boolean showTabHorizontally = getResources().getBoolean(R.bool.showTabHorizontally);
for (int i = 0; i < tabCount; i++) {
final UiDataModel.Tab tabModel = UiDataModel.getUiDataModel().getTab(i);
final @StringRes int labelResId = tabModel.getLabelResId();

final TabLayout.Tab tab = mTabLayout.newTab()
.setTag(tabModel)
.setIcon(tabModel.getIconResId())
.setContentDescription(labelResId);

if (showTabLabel) {
tab.setText(labelResId);
tab.setCustomView(R.layout.tab_item);

@SuppressWarnings("ConstantConditions")
final TextView text = (TextView) tab.getCustomView()
.findViewById(android.R.id.text1);
text.setTextColor(mTabLayout.getTabTextColors());

// Bind the icon to the TextView.
final Drawable icon = tab.getIcon();
if (showTabHorizontally) {
// Remove the icon so it doesn't affect the minimum TabLayout height.
tab.setIcon(null);
text.setCompoundDrawablesRelativeWithIntrinsicBounds(icon, null, null, null);
} else {
text.setCompoundDrawablesRelativeWithIntrinsicBounds(null, icon, null, null);
}
}

打开UiDataModel,第一个引起我的关注的是一个枚举型,在其中保存了有关Tab的基本信息,如文本、图标资源id以及该Tab所对应的fragment等。第一次见到这种写法,有点震惊,同时觉得对于这种静态的应用,用枚举型去保存也符合常理,我觉得这种方法可以以后利用在我写的Android项目中,比如说我的应用中,Tab也是固定的,同时这样写可以让那些信息更加集中,修改起来更加方便,让代码的逻辑性更强。看代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
/** Identifies each of the primary tabs within the application. */
public enum Tab {
ALARMS(AlarmClockFragment.class, R.drawable.ic_tab_alarm, R.string.menu_alarm),
CLOCKS(ClockFragment.class, R.drawable.ic_tab_clock, R.string.menu_clock),
TIMERS(TimerFragment.class, R.drawable.ic_tab_timer, R.string.menu_timer),
STOPWATCH(StopwatchFragment.class, R.drawable.ic_tab_stopwatch, R.string.menu_stopwatch);

private final String mFragmentClassName;
private final @DrawableRes int mIconResId;
private final @StringRes int mLabelResId;

Tab(Class fragmentClass, @DrawableRes int iconResId, @StringRes int labelResId) {
mFragmentClassName = fragmentClass.getName();
mIconResId = iconResId;
mLabelResId = labelResId;
}

public String getFragmentClassName() { return mFragmentClassName; }
public @DrawableRes int getIconResId() { return mIconResId; }
public @StringRes int getLabelResId() { return mLabelResId; }
}

感觉自己简直被这源代码所陶醉。后面继续看关于TabLayout的初始化。这其中,对其的适配器的写法,我也是比较感兴趣的。因为自己在写应用的时候,遇到过关于PagerAdapter的问题,所以比较期待它是如何去实现的。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
// Customize the view pager.
mFragmentTabPagerAdapter = new FragmentTabPagerAdapter(this);
mFragmentTabPager = (ViewPager) findViewById(R.id.desk_clock_pager);
// Keep all four tabs to minimize jank.
mFragmentTabPager.setOffscreenPageLimit(3);
// Set Accessibility Delegate to null so view pager doesn't intercept movements and
// prevent the fab from being selected.
mFragmentTabPager.setAccessibilityDelegate(null);
// Mirror changes made to the selected page of the view pager into UiDataModel.
mFragmentTabPager.addOnPageChangeListener(new PageChangeWatcher());
mFragmentTabPager.setAdapter(mFragmentTabPagerAdapter);

// Mirror changes made to the selected tab into UiDataModel.
mTabLayout.addOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
@Override
public void onTabSelected(TabLayout.Tab tab) {
UiDataModel.getUiDataModel().setSelectedTab((UiDataModel.Tab) tab.getTag());
}

@Override
public void onTabUnselected(TabLayout.Tab tab) {
}

@Override
public void onTabReselected(TabLayout.Tab tab) {
}
});

适配器的源代码不完全贴出来,感觉都对不起这源代码。它继承自ViewPager,自己实现了其中的方法,具体如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
/**
* This adapter produces the DeskClockFragments that are the content of the DeskClock tabs. The
* adapter presents the tabs in LTR and RTL order depending on the text layout direction for the
* current locale. To prevent issues when switching between LTR and RTL, fragments are registered
* with the manager using position-independent tags, which is an important departure from
* FragmentPagerAdapter.
*/
final class FragmentTabPagerAdapter extends PagerAdapter {

private final DeskClock mDeskClock;

/** The manager into which fragments are added. */
private final FragmentManager mFragmentManager;

/** A fragment cache that can be accessed before {@link #instantiateItem} is called. */
private final Map<UiDataModel.Tab, DeskClockFragment> mFragmentCache;

/** The active fragment transaction if one exists. */
private FragmentTransaction mCurrentTransaction;

/** The current fragment displayed to the user. */
private Fragment mCurrentPrimaryItem;

FragmentTabPagerAdapter(DeskClock deskClock) {
mDeskClock = deskClock;
mFragmentCache = new ArrayMap<>(getCount());
mFragmentManager = deskClock.getFragmentManager();
}

@Override
public int getCount() {
return UiDataModel.getUiDataModel().getTabCount();
}

/**
* @param position the left-to-right index of the fragment to be returned
* @return the fragment displayed at the given {@code position}
*/
DeskClockFragment getDeskClockFragment(int position) {
// Fetch the tab the UiDataModel reports for the position.
final UiDataModel.Tab tab = UiDataModel.getUiDataModel().getTabAt(position);

// First check the local cache for the fragment.
DeskClockFragment fragment = mFragmentCache.get(tab);
if (fragment != null) {
return fragment;
}

// Next check the fragment manager; relevant when app is rebuilt after locale changes
// because this adapter will be new and mFragmentCache will be empty, but the fragment
// manager will retain the Fragments built on original application launch.
fragment = (DeskClockFragment) mFragmentManager.findFragmentByTag(tab.name());
if (fragment != null) {
fragment.setFabContainer(mDeskClock);
mFragmentCache.put(tab, fragment);
return fragment;
}

// Otherwise, build the fragment from scratch.
final String fragmentClassName = tab.getFragmentClassName();
fragment = (DeskClockFragment) Fragment.instantiate(mDeskClock, fragmentClassName);
fragment.setFabContainer(mDeskClock);
mFragmentCache.put(tab, fragment);
return fragment;
}

@Override
public void startUpdate(ViewGroup container) {
if (container.getId() == View.NO_ID) {
throw new IllegalStateException("ViewPager with adapter " + this + " has no id");
}
}

@Override
public Object instantiateItem(ViewGroup container, int position) {
if (mCurrentTransaction == null) {
mCurrentTransaction = mFragmentManager.beginTransaction();
}

// Use the fragment located in the fragment manager if one exists.
final UiDataModel.Tab tab = UiDataModel.getUiDataModel().getTabAt(position);
Fragment fragment = mFragmentManager.findFragmentByTag(tab.name());
if (fragment != null) {
mCurrentTransaction.attach(fragment);
} else {
fragment = getDeskClockFragment(position);
mCurrentTransaction.add(container.getId(), fragment, tab.name());
}

if (fragment != mCurrentPrimaryItem) {
FragmentCompat.setMenuVisibility(fragment, false);
FragmentCompat.setUserVisibleHint(fragment, false);
}

return fragment;
}

@Override
public void destroyItem(ViewGroup container, int position, Object object) {
if (mCurrentTransaction == null) {
mCurrentTransaction = mFragmentManager.beginTransaction();
}
final DeskClockFragment fragment = (DeskClockFragment) object;
fragment.setFabContainer(null);
mCurrentTransaction.detach(fragment);
}

@Override
public void setPrimaryItem(ViewGroup container, int position, Object object) {
final Fragment fragment = (Fragment) object;
if (fragment != mCurrentPrimaryItem) {
if (mCurrentPrimaryItem != null) {
FragmentCompat.setMenuVisibility(mCurrentPrimaryItem, false);
FragmentCompat.setUserVisibleHint(mCurrentPrimaryItem, false);
}
if (fragment != null) {
FragmentCompat.setMenuVisibility(fragment, true);
FragmentCompat.setUserVisibleHint(fragment, true);
}
mCurrentPrimaryItem = fragment;
}
}

@Override
public void finishUpdate(ViewGroup container) {
if (mCurrentTransaction != null) {
mCurrentTransaction.commitAllowingStateLoss();
mCurrentTransaction = null;
mFragmentManager.executePendingTransactions();
}
}

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

修改framework后如何编译、生效!生效!

被framework生效问题困了一天, 一定要记下来。试了网上各种答案,得到的结果都没生效。最终还是从同事那里得到的一份答案,还是同事靠谱啊。一、framework编译方法一般修改framework层的内容分为两种,一种是res,一种是源代码。前者只需要在其目录下,通过mm的方式即可将framewo
阅读更多

插入耳机时安全音量提示

插入耳机的状态下,当音量达到某个值得时候,会弹出一个“继续提高音量将损害听力”的提示框。这只是一普通的对话框而已,但是这与需求不太一样,需要做一些微微的调整。源码的目录:Android 8.0中,位于:frameworks/base/packages/SystemUI/src/com/android
阅读更多

对Ubuntu开(关)机启动慢的处理

没有及时截图,只能来个恢复现场,所以主要是以解决的思路为主吧,其它的就当个参考,随便看看就好。Ubuntu升级后开(关)机贼慢开(关)机的时候,老是卡在这个地方(其实是有光的),是升级16.04之后出现的。这种情况,我总是觉得有点不对,因为之前用过的Ubuntu都是启动飞快的。为什么?它为什么开机要
阅读更多

Linux日常使用之软件安装失败的处理方式

从古董系统Ubuntu12.04一路升级到较新的16.04,发现有些软件,如samba,不能用了。用一般的处理方式如apt -f install也没办法处理。一直放着,今天还是觉得samba有一些方便,便花了一点时间去看这个问题。错误详情➜ ~ sudo apt install sambaReadi
阅读更多

【工具使用】Linux实用命令之文件搜索

在工作中,经常遇到只知道一些模糊、大致的名字的一些类名或字段,从已知的一个目录下去搜索。然而有些命令实在是太好用了,好用到赞不绝口,也有一些命令,老是忘记。。。

find

最常用的场景是在framework目录下,有一些类,只知道类名,但是不知道它的具体的路径,总不可能去一个一个搜吧,还有一个可行的是去百度,但是这也不是很适合,因为还有更快的。

这个命令的功能实在是太强大了,所以按需而罗列吧。

  • 按文件名查找
    find dirname -name "filename"
    find dirname -name "*filename*" (使用正则表达式)
    find dirname -iname "filename" (忽略大小写)
    find dirname -name "filename 2> /dev/null" (忽略错误信息)

grep

这个也是一个强大的命令,配合正则表达式也是一个逆天的存在,但是一般情况下会我用到它的场景也就是两种:

  • 查找某个目录下所有的某列包含某字符串的文件
    grep -Rn "PatternSequence" . (找到并显示行号,不要太好用!)
  • 查找某文本中所有的含有某字符串的列
    这个不解释

单例模式

有很多关于单例模式的博客,书上也有些介绍,了解它也算挺久了。但是部觉得少了点什么,怕自己记了,也怕自己后面找起来麻烦,所以还是写一篇博客来记录一下吧。

单例模式的概念

我的认识是这样,可能不是非常标准,一点自己的想法,不想copy而已:

  1. 只有一个实例
  2. 构造方法为private, 无法被直接实例化;类中持有一个本类的静态私有对象,并提供静态方法给外界提供访问。

实现方式

不同的方式有不同应用场景,可以按需要选择。其实我还是有个小小的疑问,就是关于饿汉式,instance实例化的时机是当前类被引用的时候,可以认为是在调用getInstance()的时候 ,此时类中的变量按照相应的初始化顺序,依次初始化。这样的话,就和懒汉式的效果是一样的了=.=可能是我的想法不太对吧

饿汉式

即初始化类的时候就创建好实例,不用担心多线程的环境下出现问题,但是有可能浪费资源。

1
2
3
4
5
6
7
public class SingletonHangry {
private static SingletonHangry instance;
private SingletonHangry(){}
public static SingletonHangry getInstance(){
return instance;
}
}

懒汉式

即在被需要时才创建实例,这在并发的环境下会出现一些问题。

  • 不加任何措施
    脑洞一下,都知道这个不一定是只有一个实例。
    1
    2
    3
    4
    5
    6
    7
    8
    9
    class SingletonLazy {
    private SingletonLazy(){}
    private static SingletonLazy instance;
    public static SingletonLazy getInstance(){
    if (instance == null)
    instance = new SingletonLazy();
    return instance;
    }
    }
  • 加上同步锁
    虽然这样就保证了只有一个实例,但是这个同步锁在每次调用getInstance()时都会工作,而显然这些同步工作中非常大的一部分是不需要的。所以在性能上必定会有所损失。
    1
    2
    3
    4
    5
    6
    7
    8
    9
    class SingletonLazy {
    private SingletonLazy(){}
    private static SingletonLazy instance;
    public synchronized static SingletonLazy getInstance(){
    if (instance == null)
    instance = new SingletonLazy();
    return instance;
    }
    }
  • Double Check Lock(DCL)
    说起这个,我刚开始的时候只了解到这有两个if。后来才知道这就是大名鼎鼎的DCL,失敬失敬!
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    class SingletonLazy {
    private SingletonLazy(){}
    private static volatile SingletonLazy instance;
    public static SingletonLazy getInstance(){
    if (instance == null) { // 使用这个if是为了在instance已被初始化之后,避免同步
    synchronized (SingletonLazy.class) {
    if (instance == null) {
    // 这个if是为了因为,如果同时有两个进程A,B执行了getInstance(),它们是都会有机会通过第1个if,
    // 也就是说会依次执行同步锁里面的内容。因此没有这个if也是有可能不能保证单例的。
    instance = new SingletonLazy();
    }
    }
    }
    return instance;
    }
    }
    至于那个关键字volatile,与Java中的内存模型有关,加上了这个才能保证单例。它大致做的事情是这样的:
    instance = new SingletonLazy()并非是原子操作,事实上在JVM中这句话做了三件事:

1.给instance分配内存
2.调用SingletonLazy()的构造函数来初始化成员变量
3.将instance对象指向分配的空间(执行完这一步instance就不为null)

但是在JVM的即时编译器中存在指令重排序的优化,也就是说上面的第二步和第三步是不能保证顺序的,最终执行的顺序可能是1-2-3或者是1-3-2。如果是后者,则在3执行完毕,2执行之前,被线程2抢占了,这时instance已经是非null了(但却没有初始化),所以线程2会直接返回instance,然后使用,然后会报错。使用了volatile就保证了:取操作必须在执行完1-2-3之后或者1-3-2之后,不存在执行到1-3然后取到值的情况。

  • 静态内部类
    因为类的初始化是虚拟机保证只加载一次,因此是线程安全的。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    class SingletonStaticInnerClass {
    static class SingletonHolder {
    private final static SingletonStaticInnerClass instance = new SingletonStaticInnerClass();
    }
    private SingletonStaticInnerClass(){}
    private static SingletonStaticInnerClass getInstance(){
    return SingletonHolder.instance;
    }
    }
  • 枚举
    枚举的相信息:http://blog.csdn.net/u013256816/article/details/50562905

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    enum  SingletonEnum {
    INSTANCE1, INSTANCE2;
    class Singleton {
    private Singleton(){}
    }
    private Singleton instance;
    private SingletonEnum(){
    instance = new Singleton();
    }

    public Singleton getInstance() {
    return instance;
    }
    }
    public static void main(String[] args) {
    SingletonEnum.INSTANCE1.getInstance();
    SingletonEnum.INSTANCE2.getInstance();
    }

容器

这个有点迷

常用排序算法

慢慢再温习一遍之前学习过的排序算法吧,一点点地积累。

插入排序

插入排序的基本思想是在遍历数组的过程中,假设在序号 i (i>=1)之前的元素即 [0..i-1] 都已经排好序,本趟需要找到 i 对应的元素 x 的正确位置 k ,并且在寻找这个位置 k 的过程中逐个将比较过的元素往后移一位,为元素 x “腾位置”,最后将 k 对应的元素值赋为 x ,一般情况下,插入排序的时间复杂度和空间复杂度分别为O(n2) 和 O(1)。(通俗说法:把数组后面那些没排序的元素换到数组前面已经排好序的部分里对应的位置)

例如:45 80 48 40 22 78
第一轮:45 80 48 40 22 78 —> 45 80 48 40 22 78 i=1
第二轮:45 80 48 40 22 78 —> 45 48 80 40 22 78 i=2
第三轮:45 48 80 40 22 78 —> 40 45 48 80 22 78 i=3
第四轮:40 45 48 80 22 78 —> 22 40 45 48 80 78 i=4
第五轮:22 40 45 48 80 78 —> 22 40 45 48 78 80 i=5
(红色代表此轮要插入的元素,红色左边是已经排好序的,右边是待排序的)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/**
* @param int[] 未排序数组
* @return int[] 排完序数组
*/
public static int[] InsertSort(int[] array){
for(int i =1;i<array.length;i++){
int temp = array[i];
int j = i-1;
while(j>=0 && temp < array[j] ){
array[j+1] = array[j];
j--;
}
array[j+1] = temp;
}
return array;
}

选择排序

选择排序的基本思想是遍历数组的过程中,以 i 代表当前需要排序的序号,则需要在剩余的 [i+1,…n-1] 中找出其中的最小值,然后将找到的最小值与 i 指向的值进行交换。因为每一趟确定元素的过程中都会有一个选择最大值/最小值的子流程,所以人们形象地称之为选择排序。选择排序的时间复杂度和空间复杂度分别为O(n2)和O(1)。(通俗说法:每次把剩余数组里最小的选出来放在数组的前面。所以第一次选出来的就是数组里面最小的,第二次选出来的就是数组里面第二小的,依次。。。。。)

例如:45 80 48 40 22 78
第一轮:45 80 48 40 22 78 —> 22 80 48 40 45 78 i=0
第二轮:22 80 48 40 45 78 —> 22 40 48 80 45 78 i=1
第三轮:22 40 48 80 45 78 —> 22 40 45 80 48 78 i=2
第四轮:22 40 45 80 48 78 —> 22 40 45 48 80 78 i=3
第五轮:22 40 45 48 80 78 —> 22 40 45 48 78 80 i=4
(红色代表此轮需要排序的序号的元素,红色左边是已经排好序的,右边是待排序的)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
/**
* @param int[] 未排序数组
* @return int[] 排完序数组
*/
public int[] sortSelect(int[] arr){
for (int i = 0; i < arr.length-1; i++) {
int miniPost = i;
for (int m = i + 1; m < arr.length; m++) {
if (arr[m] < arr[miniPost])
  miniPost = m;
}
if (arr[i] > arr[miniPost]) {
int temp = arr[i];
arr[i] = arr[miniPost];
arr[miniPost] = temp;
}
}
return arr;
}

归并排序

来自:https://www.cnblogs.com/chengxiao/p/6194356.html,图非常简洁明了。

归并排序(MERGE-SORT)是利用归并的思想实现的排序方法,该算法采用经典的分治(divide-and-conquer)策略(分治法将问题分(divide)成一些小的问题然后递归求解,而治(conquer)的阶段则将分的阶段得到的各答案”修补”在一起,即分而治之)。

分而治之

这里写图片描述
可以看到这种结构很像一棵完全二叉树,本文的归并排序我们采用递归去实现(也可采用迭代的方式去实现)。分阶段可以理解为就是递归拆分子序列的过程,递归深度为log2n。

合并相邻有序子序列

再来看看治阶段,我们需要将两个已经有序的子序列合并成一个有序序列,比如上图中的最后一次合并,要将[4,5,7,8]和[1,2,3,6]两个已经有序的子序列,合并为最终序列[1,2,3,4,5,6,7,8],来看下实现步骤。
这里写图片描述
这里写图片描述

代码实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
package sortdemo;

import java.util.Arrays;

/**
* Created by chengxiao on 2016/12/8.
*/
public class MergeSort {
public static void main(String []args){
int []arr = {9,8,7,6,5,4,3,2,1};
sort(arr);
System.out.println(Arrays.toString(arr));
}
public static void sort(int []arr){
int []temp = new int[arr.length];//在排序前,先建好一个长度等于原数组长度的临时数组,避免递归中频繁开辟空间
sort(arr,0,arr.length-1,temp);
}
private static void sort(int[] arr,int left,int right,int []temp){
if(left<right){
int mid = (left+right)/2;
sort(arr,left,mid,temp);//左边归并排序,使得左子序列有序
sort(arr,mid+1,right,temp);//右边归并排序,使得右子序列有序
merge(arr,left,mid,right,temp);//将两个有序子数组合并操作
}
}
private static void merge(int[] arr,int left,int mid,int right,int[] temp){
int i = left;//左序列指针
int j = mid+1;//右序列指针
int t = 0;//临时数组指针
while (i<=mid && j<=right){
if(arr[i]<=arr[j]){
temp[t++] = arr[i++];
}else {
temp[t++] = arr[j++];
}
}
while(i<=mid){//将左边剩余元素填充进temp中
temp[t++] = arr[i++];
}
while(j<=right){//将右序列剩余元素填充进temp中
temp[t++] = arr[j++];
}
t = 0;
//将temp中的元素全部拷贝到原数组中
while(left <= right){
arr[left++] = temp[t++];
}
}
}

/**
执行结果
[1, 2, 3, 4, 5, 6, 7, 8, 9]
**/

归并排序是稳定排序,它也是一种十分高效的排序,能利用完全二叉树特性的排序一般性能都不会太差。java中Arrays.sort()采用了一种名为TimSort的排序算法,就是归并排序的优化版本。从上文的图中可看出,每次合并操作的平均时间复杂度为O(n),而完全二叉树的深度为|log2n|。总的平均时间复杂度为O(nlogn)。而且,归并排序的最好,最坏,平均时间复杂度均为O(nlogn)。

快速排序

大概的思想是(假设为升序),先以一个数为参照,然后将所有的大于它的数移到它的后面,小于它的数移到前面,再以此数的位置,将数组分成两部分,再对这两部分做相同的操作…如此递归,直至完毕。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
public static void main(String[] args) {
int[] arr = new int[]{6,2,4,99,22,56,11,5,7,4,1};
QuickSort(arr, 0, arr.length - 1);
printArray(arr);
}

static void QuickSort(int[] arr, int left, int right){
if (left > right)
return;
int pivot_index = partition1(arr, left, right);
QuickSort(arr, left, pivot_index - 1);
QuickSort(arr, pivot_index + 1, right);
}

static int partition1(int[] arr, int left, int right){
int pivot = arr[left];
while(left < right){
while(left < right && arr[right] >= pivot){
right--;
}
arr[left] = arr[right];

while(left < right && arr[left] <= pivot){
left++;
}
arr[right] = arr[left];

}
arr[left] = pivot;
return left;
}

基础Shell脚本大法

shell中的变量变量的设置规则读取变量的时候,$PATH 与 $是等同的双引号内的特殊符号如$等,可以保持原有的特性。单引号内的特殊符号则仅为一般字符(纯文本)`命令` 与 $(命令)是等同的变量如何加1语法((i=i+1));let i=i+1;x=$(( $x + 1 ))x=`expr $x
阅读更多

MTK和Qualcomm刷机备忘录

这两个处理器厂商的刷机方式有一些不一样,然而很久不刷的我还是有些迷。还是总结一下,不要再用错误的方式,来刷机了~

MTK刷机

从服务器上面下载target_files-package.zip的编译包,然后解压;也可以选择使用自己编译源代码出来得到的rom。
这里写图片描述
烧录的时候:
1、Download agent选择路径(默认一般已经选好):
MTK_driver_and_flashtool\SP_Flash_Tool_exe_Windows_v5.1612.00.000 \MTK_AllInOne_DA.bin
2、Scatter-loading选择编译出来的android目录中的scatter文件,路径如下:
\out\target\product\*****\*****_Android_scatter.txt
选择完成,之后按download开始,这个时候,将手机关机,插入USB即可开始烧录,等待烧录完成,然后重启(扣电池也行)手机,即可。

Qualcomm刷机

高通用QFILE,不论在edl模式还是在开机状态都可以。
第一栏选择好相应的.mbn文件后,第二栏的配置文件一般做如下选择:

  • 第一次选择:
  1. rawprogram_unsparse.xml || rawprogram_unsparse_without_QCN.xml
  2. rawprogram2.xml
  • 第二次选择:
    两个都选上。