身为JAVA工作者必须了解的实战知识(十一)

并发测试大致分为两类:安全性测试(不发生任何错误的行为)和活跃性测试(某个良好的行为终究会发生)。

安全测试 - 通常采用测试不变性条件的形式,即判断某个类的行为是否与其他规范保持一致。

活跃性测试 - 包括进展测试和无进展测试两个方面。

性能测试与活跃性测试相关,主要包括:吞吐量、响应性、可伸缩性。

一、正确性测试

找出需要检查的不变条件和后延条件。

[java]view plaincopy

print?

importjava.util.concurrent.Semaphore;

publicclassBoundedBuffer {

privatefinalSemaphore availableItems, availableSpaces;

privatefinalE[] items;

privateintputPosition =0;

privateinttakePosition =0;

@SuppressWarnings("unchecked")

publicBoundedBuffer(intcapacity) {

availableItems =newSemaphore(0);

availableSpaces =newSemaphore(capacity);

items = (E[])newObject[capacity];

}

publicbooleanisEmpty() {

returnavailableItems.availablePermits() ==0;

}

publicbooleanisFull() {

returnavailableSpaces.availablePermits() ==0;

}

publicvoidput(E x)throwsInterruptedException {

availableSpaces.acquire();

doInsert(x);

availableItems.release();

}

publicE take()throwsInterruptedException {

availableItems.acquire();

E item = doExtract();

availableSpaces.release();

returnitem;

}

privatesynchronizedvoiddoInsert(E x) {

inti = putPosition;

items[i] = x;

putPosition = (++i == items.length)?0: i;

}

privatesynchronizedE doExtract() {

inti = takePosition;

E x = items[i];

items[i] =null;

takePosition = (++i == items.length)?0: i;

returnx;

}

}

1 基本的单元测试

[java]view plaincopy

print?

importstaticorg.junit.Assert.*;

importorg.junit.Test;

publicclassBoundedBufferTests {

@Test

publicvoidtestIsEmptyWhenConstructed() {

BoundedBuffer bb =newBoundedBuffer(10);

assertTrue(bb.isEmpty());

assertFalse(bb.isFull());

}

@Test

publicvoidtestIsFullAfterPuts()throwsInterruptedException {

BoundedBuffer bb =newBoundedBuffer(10);

for(inti =0; i <10; i++) {

bb.put(i);

}

assertTrue(bb.isFull());

assertTrue(bb.isEmpty());

}

}

2 对阻塞操作的测试

take方法是否阻塞、中断处理。从空缓存中获取一个元素。

[java]view plaincopy

print?

@Test

publicvoidtestTakeBlocksWhenEmpty(){

finalBoundedBuffer bb =newBoundedBuffer(10);

Thread taker =newThread(){

@Override

publicvoidrun() {

try{

intunused =  bb.take();

fail();//如果执行到这里,那么表示出现了一个错误

}catch(InterruptedException e) { }

}

};

try{

taker.start();

Thread.sleep(LOCKUP_DETECT_TIMEOUT);

taker.interrupt();

taker.join(LOCKUP_DETECT_TIMEOUT);

assertFalse(taker.isAlive());

}catch(InterruptedException e) {

fail();

}

}

创建一个“获取”线程,该线程将尝试从空缓存中获取一个元素。

如果take方法成功,那么表示测试失败。

执行测试的线程启动“获取”线程,等待一段时间,然后中断该线程。

如果“获取”线程正确地在take方法中阻塞,那么将抛出InterruptedException,而捕获到这个异常的catch块将把这个异常视为测试成功,并让线程退出。

然后,主测试线程会尝试与“获取”线程合并,通过调用Thread.isAlive来验证join方法是否成功返回,如果“获取”线程可以响应中断,那么join能很快地完成。

使用Thread.getState来验证线程能否在一个条件等待上阻塞,但这种方法并不可靠。被阻塞线程并不需要进入WAITING或者TIMED_WAITING等状态,因此JVM可以选择通过自旋等待来实现阻塞。

3 安全性测试

在构建对并发类的安全性测试中,需要解决地关键性问题在于,要找出那些容易检查的属性,这些属性在发生错误的情况下极有可能失败,同时又不会使得错误检查代码人为地限制并发性。理想情况是,在测试属性中不需要任何同步机制。

[java]view plaincopy

print?

importjava.util.concurrent.CyclicBarrier;

importjava.util.concurrent.ExecutorService;

importjava.util.concurrent.Executors;

importjava.util.concurrent.atomic.AtomicInteger;

importjunit.framework.TestCase;

publicclassPutTakeTestextendsTestCase {

privatestaticfinalExecutorService pool = Executors.newCachedThreadPool();

privatefinalAtomicInteger putSum =newAtomicInteger(0);

privatefinalAtomicInteger takeSum =newAtomicInteger(0);

privatefinalCyclicBarrier barrier;

privatefinalBoundedBuffer bb;

privatefinalintnTrials, nPairs;

publicstaticvoidmain(String[] args) {

newPutTakeTest(10,10,100000).test();// 示例参数

pool.shutdown();

}

staticintxorShift(inty) {

y ^= (y <<6);

y ^= (y >>>21);

y ^= (y <<7);

returny;

}

publicPutTakeTest(intcapacity,intnPairs,intnTrials) {

this.bb =newBoundedBuffer(capacity);

this.nTrials = nTrials;

this.nPairs = nPairs;

this.barrier =newCyclicBarrier(nPairs *2+1);

}

voidtest() {

try{

for(inti =0; i < nPairs; i++) {

pool.execute(newProducer());

pool.execute(newConsumer());

}

barrier.await();// 等待所有的线程就绪

barrier.await();// 等待所有的线程执行完成

assertEquals(putSum.get(), takeSum.get());

}catch(Exception e) {

thrownewRuntimeException(e);

}

}

classProducerimplementsRunnable {

@Override

publicvoidrun() {

try{

intseed = (this.hashCode() ^ (int) System.nanoTime());

intsum =0;

barrier.await();

for(inti = nTrials; i >0; --i) {

bb.put(seed);

sum += seed;

seed = xorShift(seed);

}

putSum.getAndAdd(sum);

barrier.await();

}catch(Exception e) {

thrownewRuntimeException(e);

}

}

}

classConsumerimplementsRunnable {

@Override

publicvoidrun() {

try{

barrier.await();

intsum =0;

for(inti = nTrials; i >0; --i) {

sum += bb.take();

}

takeSum.getAndAdd(sum);

barrier.await();

}catch(Exception e) {

thrownewRuntimeException(e);

}

}

}

}

4 资源管理的测试

对于任何持有或管理其他对象的对象,都应该在不需要这些对象时销毁对他们的引用。测试资源泄露的例子:

[java]view plaincopy

print?

classBig {

double[] data =newdouble[100000];

};

voidtestLeak()throwsInterruptedException{

BoundedBuffer bb =newBoundedBuffer(CAPACITY);

intheapSize1 =/* 生成堆的快照 */;

for(inti =0; i < CAPACITY; i++){

bb.put(newBig());

}

for(inti =0; i < CAPACITY; i++){

bb.take();

}

intheapSize2 =/* 生成堆的快照 */;

assertTrue(Math.abs(heapSize1 - heapSize2) < THRESHOLD);

}

5 使用回调

6 产生更多的交替操作

二、性能测试

性能测试的目标 - 根据经验值来调整各种不同的限值。例如:线程数量、缓存容量等。

1 在PutTakeTest中增加计时功能

基于栅栏的定时器

[java]view plaincopy

print?

this.timer =newBarrierTimer();

this.barrier =newCyclicBarrier(nPairs *2+1, timer);

publicclassBarrierTimerimplementsRunnable{

privatebooleanstarted ;

privatelongstartTime ;

privatelongendTime ;

@Override

publicsynchronizedvoidrun() {

longt = System.nanoTime();

if(!started ){

started =true;

startTime = t;

}else{

endTime = t;

}

}

publicsynchronizedvoidclear(){

started =false;

}

publicsynchronizedlonggetTime(){

returnendTime - startTime;

}

}

修改后的test方法中使用了基于栅栏的计时器

[java]view plaincopy

print?

voidtest(){

try{

timer.clear();

for(inti =0; i < nPairs; i++){

pool .execute(newProducer());

pool .execute(newConsumer());

}

barrier .await();

barrier .await();

longnsPerItem = timer.getTime() / ( nPairs * (long)nTrials );

System. out .println("Throughput: "+ nsPerItem +" ns/item");

assertEquals(putSum.get(), takeSum.get() )

}catch(Exception e) {

thrownewRuntimeException(e);

}

. 生产者消费者模式在不同参数组合下的吞吐率

. 有界缓存在不同线程数量下的伸缩性

. 如何选择缓存的大小

[java]view plaincopy

print?

publicstaticvoidmain(String[] args)throwsInterruptedException {

inttpt =100000;// 每个线程中的测试次数

for(intcap =1; cap <= tpt; cap *=10){

System. out .println("Capacity: "+ cap);

for(intpairs =1; pairs <=128; pairs *=2){

TimedPutTakeTest t =newTimedPutTakeTest(cap, pairs, tpt);

System. out .println("Pairs: "+ pairs +"\t");

t.test();

System. out .println("\t");

Thread. sleep(1000);

t.test();

System. out .println();

Thread. sleep(1000);

}

}

pool .shutdown();

}

查看吞吐量/线程数量的关系

2 多种算法的比较

3 响应性衡量

三、避免性能测试的陷阱

1 垃圾回收

2 动态编译

3 对代码路径的不真实采样

4 不真实的竞争程度

5 无用代码的消除

四、其他的测试方法

1 代码审查

2 静态分析工具

FindBugs、Lint

3 面向方面的测试技术

4 分析与监测工具

以上就是我推荐给Java开发者们的一面试经典知识。但是这些知识里面并没有太多Java全栈、Java晋阶、JAVA架构之类的题,不是我不推荐,而是希望大家更多的从基本功做起,打好基础,太多复杂的内容一会儿也说不明白。

好了同学们,我能介绍的也都全部介绍完给你们了,如果下获得更多JAVA教学资源,可以选择来我们这里共同交流,群:240448376,很多大神在这里切磋学习,不懂可以直接问,晚上还有大牛免费直播教学。

注:加群要求

1、具有一定工作经验的,面对目前流行的技术不知从何下手,需要突破技术瓶颈的可以加,有些应届生和实习生也可以加。

2、在公司待久了,过得很安逸,但跳槽时面试碰壁。需要在短时间内进修、跳槽拿高薪的可以加。

3、如果没有工作经验,但基础非常扎实,对java工作机制,常用设计思想,常用java开发框架掌握熟练的,可以加。

4、觉得自己很牛B,一般需求都能搞定。但是所学的知识点没有系统化,很难在技术领域继续突破的可以加。

5.阿里Java高级大牛直播讲解知识点,分享知识,多年工作经验的梳理和总结,带着大家全面、科学地建立自己的技术体系和技术认知!

PS:现在主要讲解的内容是(反射原理枚举原理与应用注解原理常用设计模式、正规表达式高级应用、JAVA操作Office原理详解JAVA图像处理技术,等多个知识点的详解和实战)

6.小号或者小白之类加群一律不给过,谢谢。

最后,每一位读到这里的网友,感谢你们能耐心地看完。觉得对你有帮助可以给个喜欢!希望在成为一名更优秀的Java程序员的道路上,我们可以一起学习、一起进步

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,298评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,701评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 107,078评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,687评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,018评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,410评论 1 211
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,729评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,412评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,124评论 1 239
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,379评论 2 242
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,903评论 1 257
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,268评论 2 251
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,894评论 3 233
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,014评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,770评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,435评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,312评论 2 260

推荐阅读更多精彩内容