Functional Programming in Java venkat(14) Being Lazy
admin
2024-04-15 01:07:17
0

文章目录

  • Functional Programming in Java venkat(14): Being Lazy
    • Delayed Initialization
      • A Familiar Approach
      • Providing Thread Safety
      • Adding a Level of Indirection
    • 英文

Functional Programming in Java venkat(14): Being Lazy

这里是记录学习这本书 Functional Programming in Java: Harnessing the Power Of Java 8 Lambda Expressions 的读书笔记,如有侵权,请联系删除。

本章的内容:延时创建heavyweight 的对象,然后把eager的计算的情形转变为lazy评估。

In this chapter we start with a task to postpone the creation of a heavyweight object, then we turn some eager computations into lazy evaluations.

Delayed Initialization

eager is easy to write and to reason about. But delaying commitments until the last responsible moment is a good agile practice.

A Familiar Approach

把heavyweight的资源放在Heavy类中。

move the heavyweight resources into another class—say, Heavy. Then an instance of Holder will keep a reference to an instance of Heavy and route calls to it as appropriate.

package fpij;public class Heavy {public Heavy() { System.out.println("Heavy created"); }public String toString() { return "quite heavy"; }
}

下面是最简单的延迟创建Heavy对象,当heavy = null 的时候创建新的Heavy对象,否则的话返回heavy对象。

package fpij;public class HolderNaive {private Heavy heavy;public HolderNaive() {System.out.println("Holder created");}public Heavy getHeavy() {if(heavy == null) {heavy = new Heavy();}return heavy;}//...public static void main(final String[] args) {final HolderNaive holder = new HolderNaive();System.out.println("deferring heavy creation...");System.out.println(holder.getHeavy());System.out.println(holder.getHeavy());}
}

但是存在问题:线程不安全

如果多个线程同时进来,可能会创建多个heavy对象。

That appears to work. The solution, however, is not simple; it is a familiar
but also a rather simplistic solution that fails thread safety. Let’s work through
it.

Providing Thread Safety

变成线程安全的需要synchronized关键字

  public synchronized Heavy getHeavy() {if(heavy == null) {heavy = new Heavy();}return heavy;}

多个线程进来,满足互斥。只有一个可以进入创建对象,其他的线程进入之后会发现对象已经被之前的线程创建,从而直接返回heavy对象。

If two or more threads call this method concurrently, due to mutual exclusion only one will be allowed to enter and the others will queue up for their turn. The first one to enter into the method will create the instance. When subsequent threads enter this method they will see that the instance already exists,
and will simply return it.

我们避免了竞赛条件,但这个解决方案产生了另一个负面影响。现在对getHeavy()方法的每一次调用都必须忍受同步开销;即使没有同时竞争的线程,调用线程也必须跨越内存障碍(memory barrier)。

We averted the race condition, but the solution created another negative impact. Every call to the getHeavy() method now has to endure the synchronization overhead; the calling threads have to cross the memory barrier even if there are no concurrently competing threads.

Adding a Level of Indirection


import java.util.function.Supplier;public class Holder {private Supplier heavy = () -> createAndCacheHeavy();public Holder() {System.out.println("Holder created");}public Heavy getHeavy() {return heavy.get();}//...
}

当Holder的实例被创建时,我们可以看到,Heavy的实例并没有被创建。这种设计实现了懒惰初始化的目标。我们还需要一个非苛刻的线程安全解决方案。这就是createAndCacheHeavy()方法的用处。

When an instance of Holder is created, as we can see, an instance of Heavy is not created. This design achieves the goal of lazy initialization. We also need a non-draconian solution to thread safety. This is where the createAndCacheHeavy() method comes in.

查看createAndCacheHeavy方法

 private synchronized Heavy createAndCacheHeavy() {class HeavyFactory implements Supplier {private final Heavy heavyInstance = new Heavy();public Heavy get() { return heavyInstance; }}if(!HeavyFactory.class.isInstance(heavy)) {heavy = new HeavyFactory();}return heavy.get();}

让我们考虑这样一个场景:一个新的Holder实例刚刚被创建。我们假设有两个线程同时调用getHeavy()方法,随后第三个线程在很久之后调用这个方法。当前两个线程在Holder中调用默认Supplier的get()方法时,createAndCacheHeavy()方法会让其中一个线程通过,让另一个线程等待。第一个进入的线程将检查heavy是否是HeavyFactory的一个实例。由于它不是默认的Supplier,这个线程将用HeavyFactory的一个实例来替换heavy。最后它返回这个HeavyFactory所持有的Heavy实例。

Let’s consider a scenario in which a new instance of Holder has just been created. Let’s assume two threads invoke the getHeavy() method concurrently, followed by a third thread calling this method much later. When the first two threads call the default supplier’s get() method in the Holder, the createAndCacheHeavy() method will let one of them through and make the other wait. The first thread to enter will check if heavy is an instance of the HeavyFactory. Since it is not the default Supplier, this thread will replace heavy with an instance of HeavyFactory. Finally it returns the Heavy instance that this HeavyFactory holds.

现在heavy已经被HeavyFactory所取代,随后对getHeavy()方法的调用将直接进入HeavyFactory的get()方法,不会产生任何同步开销。

Now that heavy has been replaced with HeavyFactory, subsequent calls to the getHeavy() method will go directly to the HeavyFactory’s get() method and will not incur any synchronization overhead

我们设计了懒惰的初始化,同时,避免了空值检查。此外,我们确保了懒惰实例创建的线程安全。这是虚拟代理模式的一个简单、轻量级的实现。接下来我们将使用lambda表达式来延缓函数的评估。

We designed lazy initialization and, at the same time, avoided null checks. Furthermore, we ensured the thread safety of the lazy instance creation. This is a simple, lightweight implementation of the virtual proxy pattern. Next we’ll use lambda expressions to postpone function evaluations.

英文

overhead: 开销

will not incur any synchronization overhead(不会产生任何同步开销)

相关内容

热门资讯

漂来的狗儿中,狗儿的创作原形是... 漂来的狗儿中,狗儿的创作原形是谁?如题答案自在施主心中。你说方是圆,方即是圆。
漫画 魔法禁书目录 具体那几... 漫画 魔法禁书目录 具体那几天更新?如果是我的建议,那就是去看原版的小说吧,毕竟动画和漫画都是从小...
十二星座最在意什么? 十二星座最在意什么?十二星座里每个星座最在意的什么?双鱼座最在意感情,他们希望她们可以有很完美的感情...
电子发票与普通纸质发票有什么区... 电子发票与普通纸质发票有什么区别?电子发票与传统发票的区别主要有两点:一、是从传统的物理介质发展为数...
五行山下,孙悟空为何对观音菩萨... 五行山下,孙悟空为何对观音菩萨说:如来佛祖骗了俺,他到底被骗了什么?他到底被骗了什么?当时好像如来佛...
辛弃疾词“求田问舍,怕应羞见,... 辛弃疾词“求田问舍,怕应羞见,刘郎才气”中刘郎指谁?题目:辛弃疾词“求田问舍,怕应羞见,刘郎才气”中...
腰间盘突出的证状是什么 腰间盘突出的证状是什么腰间盘突出的症状很多,其中比较典型的有腰部疼痛、麻木,坐骨神经痛,下肢放射痛,...
云集多地风味 澳门上演美食嘉年... 7月11日,演员在2025澳门国际美食之都嘉年华开幕仪式上表演。新华社发(张金加摄) 新华社澳门7月...
呵护生命安全成长内容简介 呵护生命安全成长内容简介【摘要】小学生由于年龄小,生活经验少,好奇好动,不知危险,受到交通、溺水、拐...
成都上半年接待游客超1.6亿人... 封面新闻记者 赖芳杰 摄影报道“成都旅游业总体实力居全国第一方阵。”7月12日,7月12日上午,中共...
丹柯的故事出自哪里? 丹柯的故事出自哪里?高尔基写的《丹柯的故事》
超过五百万字的都市小说,最好是... 超过五百万字的都市小说,最好是重生的,或者系统的《我的清纯大小姐》也挺好看的
脱线天使的恋爱魔咒好看还是幸福... 脱线天使的恋爱魔咒好看还是幸福仙履屋好看?其实我觉得两本都很好看一本比较新,一本比较旧而已不过如果说...
女朋友的前男友比我好,她评价他... 女朋友的前男友比我好,她评价他是完美的。我心里很难过。这没什么 他说她以前的男友那块做的好 那就说明...
匹诺曹是什么意思? 匹诺曹是什么意思?匹诺曹是童话故事《木偶奇遇记》中的主人公.他只要一撒谎,鼻子就会变长.所以匹诺曹指...
你们有谁亲眼见过球状闪电的 你们有谁亲眼见过球状闪电的那是很惊悚的存在
《以家人之名》后,张新成又一部... 《以家人之名》后,张新成又一部新剧开播,他的演技如何?演技是非常不错的,感觉什么角色都可以手到擒来,...
你是我的骄傲。用英文怎么说? 你是我的骄傲。用英文怎么说?I'm proud of you 我以你为傲!you're my pro...