何靖的博客


  • 首页

  • 标签

  • 分类

  • 归档

  • 搜索

Android 进程间通讯方式

发表于 2019-03-08 | 分类于 Android

1、通过单向数据管道传递数据

管道(使用PipedWriter/ 创建PipedReader)是java.io包的一部分。也就是说,它们是一般的Java功能,而不是Android特定的。管道为同一进程内的两个线程提供了一种连接和建立单向数据通道的方法。生产者线程将数据写入管道,而使用者线程从管道读取数据。

2、共享内存通信

共享内存(使用编程中已知的内存区域作为堆)是在线程之间传递信息的常用方法。应用程序中的所有线程都可以访问进程中的相同地址空间。

3、实现消费者 - 生产者模式 BlockingQueue

线程信令是一种低级,高度可配置的机制,可以适应许多用例,但它也可能被认为是最容易出错的技术。因此,Java平台在线程信令机制上构建高级抽象,以解决线程之间的任意对象的单向切换。抽象通常被称为“解决生产者 - 消费者同步问题。”问题包括可能存在生成内容的线程(生产者线程)和消费内容的线程(消费者线程)的用例。生产者将消息传递给消费者进行处理。线程之间的中介是具有阻塞行为的队列,即java.util.concurrent.BlockingQueue。

4、消息队列

消息处理机制是Android平台的基础,API位于android.os包中,其中包含一组实现该功能的类Handler。

android.os.Looper
与唯一的消费者线程关联的消息调度程序。

android.os.Handler
消费者线程消息处理器,以及生产者线程将消息插入队列的接口。Looper可以有许多关联的处理程序,但它们都将消息插入到同一队列中。

android.os.MessageQueue
要在消费者线程上处理的无限链接消息列表。每个Looper和Thread-最多只有一个MessageQueue。

android.os.Message
要在使用者线程上执行的消息。

原文链接

Git如何把本地代码推送到远程仓库

发表于 2019-03-08 | 分类于 Git
1、 初始化版本库
1
$ git init
2、 添加文件到版本库(只是添加到缓存区),.代表添加文件夹下所有文件
1
$ git add .
3、 把添加的文件提交到版本库,并填写提交备注
1
$ git commit -m "first commit"
4、 把本地库与远程库关联
1
$ git remote add origin 远程库地址
5、 第一次推送时
1
$ git push -u origin master
6、第一次推送后,直接使用该命令即可推送修改
1
$ git push origin master

Git回滚代码到某个commit

发表于 2019-03-08 | 分类于 Git

回退命令:

$git reset --hard HEAD^      回退到上个版本  
$git reset --hard HEAD~3     回退三次提交
$git reset --hard commit-id  退到/进到指定commit

强推到远程:

$git push origin HEAD --force 

相关链接

JVM内存回收机制

发表于 2019-03-08 | 分类于 Java , GC

1、回收算法

  • 标记回收算法(Mark and Sweep GC)
    从GC Roots集合开始,将内存整个遍历一次,保留所有可以被GC Roots直接或间接引用到的对象,而剩下的对象都当作垃圾对待并回收,这个算法需要中断进程内其它组件的执行并且可能产生内存碎片

  • 复制算法 (Copying)
    将现有的内存空间分为两快,每次只使用其中一块,在垃圾回收时将正在使用的内存中的存活对象复制到未被使用的内存块中,之后,清除正在使用的内存块中的所有对象,交换两个内存的角色,完成垃圾回收。

  • 标记-压缩算法 (Mark-Compact)
    先需要从根节点开始对所有可达对象做一次标记,但之后,它并不简单地清理未标记的对象,而是将所有的存活对象压缩到内存的一端。之后,清理边界外所有的空间。这种方法既避免了碎片的产生,又不需要两块相同的内存空间,因此,其性价比比较高。

  • 分代
    将所有的新建对象都放入称为年轻代的内存区域,年轻代的特点是对象会很快回收,因此,在年轻代就选择效率较高的复制算法。当一个对象经过几次回收后依然存活,对象就会被放入称为老年代的内存空间。对于新生代适用于复制算法,而对于老年代则采取标记-压缩算法。

2、复制算法和标记-压缩算法的区别

乍一看这两个算法似乎并没有多大的区别,都是标记了然后挪到另外的内存地址进行回收,那为什么不同的分代要使用不同的回收算法呢?

其实2者最大的区别在于前者是用空间换时间后者则是用时间换空间。

前者的在工作的时候是不没有独立的mark与copy段的,而是合在一起做一个动作,就叫scavenge(或evacuate,或者就叫copy)。也就是说,每发现一个这次收集中尚未访问过的活对象就直接copy到新地方,同时设置forwarding pointer。这样的工作方式就需要多一份空间。

后者在工作的时候则需要分别的mark与compact阶段,mark阶段用来发现并标记所有活的对象,然后compact阶段才移动对象来达到compact的目的。如果compact方式是sliding compaction,则在mark之后就可以按顺序一个个对象滑动到空间的某一侧。因为已经先遍历了整个空间里的对象图,知道所有的活对象了,所以移动的时候就可以在同一个空间内而不需要多一份空间。

所以新生代的回收会更快一点,老年代的回收则会需要更长时间,同时压缩阶段是会暂停应用的,所以给我们应该尽量避免对象出现在老年代。

Java 二分查找算法

发表于 2019-03-08 | 分类于 Java , 算法

1、二分查找又称折半查找,它是一种效率较高的查找方法。

2、二分查找要求:

  • *(1)必须采用顺序存储结构 *
  • (2)必须按关键字大小有序排列

3、原理:

将数组分为三部分,依次是中值(所谓的中值就是数组中间位置的那个值)前,中值,中值后;将要查找的值和数组的中值进行比较,若小于中值则在中值前 面找,若大于中值则在中值后面找,等于中值时直接返回。然后依次是一个递归过程,将前半部分或者后半部分继续分解为三部分。

4、实现:二分查找的实现用递归和循环两种方式

5、代码

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
package other;

public class BinarySearch {
/*
* 循环实现二分查找算法arr 已排好序的数组x 需要查找的数-1 无法查到数据
*/
public static int binarySearch(int[] arr, int x) {
int low = 0;
int high = arr.length-1;
while(low <= high) {
int middle = (low + high)/2;
if(x == arr[middle]) {
return middle;
}else if(x <arr[middle]) {
high = middle - 1;
}else {
low = middle + 1;
}
}
return -1;
}
//递归实现二分查找
public static int binarySearch(int[] dataset,int data,int beginIndex,int endIndex){
int midIndex = (beginIndex+endIndex)/2;
if(data <dataset[beginIndex]||data>dataset[endIndex]||beginIndex>endIndex){
return -1;
}
if(data <dataset[midIndex]){
return binarySearch(dataset,data,beginIndex,midIndex-1);
}else if(data>dataset[midIndex]){
return binarySearch(dataset,data,midIndex+1,endIndex);
}else {
return midIndex;
}
}

public static void main(String[] args) {
int[] arr = { 6, 12, 33, 87, 90, 97, 108, 561 };
System.out.println("循环查找:" + (binarySearch(arr, 87) + 1));
System.out.println("递归查找"+binarySearch(arr,87,2,arr.length-1));
}
}
1234
何靖

何靖

18 日志
13 分类
28 标签
GitHub E-Mail CSDN
© 2021 何靖
由 Hexo 强力驱动
|
主题 — NexT.Pisces v5.1.4