草庐IT

java - java 8 供应商是否正在替换构造函数调用

coder 2024-03-15 原文

我一直在研究 Java 8 的供应商和消费者接口(interface),并且据我所知它可以替代构造函数调用。
我在 dzone ( link here ) 上看到了一个使用 ShapeFactory 的示例。
代码非常简单,是一个形状类的简单工厂。
但她是这样使用的:

Supplier<ShapeFactory> supplier = ShapeFactory::new;
supplier.get().getShape("rectangle").draw(); 

但为什么它比做一个简单的经典更好:
ShapeFactory factory = new ShapeFactory();
factory.getShape("rectangle").draw()

这足够简单和有效。此外,如果ShapeFactory 类的构造函数有参数,Supplier 将无法工作,我们将不得不使用Function 或其他方法。

那么为什么在这种精确的情况下使用这样的供应商呢?

最佳答案

TLDR

使用 Supplier带来更好的可读性/可维护性 而不是显式构造函数调用,例如 new MyObject()成语由 switch 处理/if-else if声明。
它还带来了更好的类型安全通过反射实例化,替代 new MyObject()习惯用法,通常在 Java 8 之前用于解决可维护性问题,但这会引入其他问题。

实例化 Factory class 不一定是说明 Supplier 好处的最佳示例.
所以,假设我们要实例化 Shape来自 Factory 的类类(class)。
Shape一家工厂应该实例化不同种类的 Shape (它的子类)。

1) 无 Supplier并带有 new MyShape()习惯用法,你将完成一个包含一些 if-else if 的方法。/switch检查一种标准/参数并根据此标准/参数实例化预期类的语句。
例如 :

public static Shape createShape(String criteria) {
    if (criteria.equals("circle")){
        return new Circle();
    }
    else if (criteria.equals("square")){
        return new Square();
    }
    ...
}  

这很糟糕,因为当您添加要由该方法处理的类时,您必须使用新的 if-else if 更改此方法。/switch以便将其考虑在内。
这会导致不可维护的代码,您可能会在其中快速创建副作用。

2)为了避免这个问题,我们经常使用反射Class.newInstance() .
它消除了 if-else if/switch问题,但它经常会产生其他问题,因为反射可能不起作用(安全问题、类不可实例化等),您只会在运行时知道它。
结果仍然是脆弱的代码。

以下是青睐 Supplier 的理由:

通过提供 Supplier , 检查在编译时执行 : 如果类不可实例化,编译器将发出错误。
此外,使用/接受 Supplier<Shape> 的方法不需要使用if-else if/switch声明。

当您使用 Supplier ,您通常会遇到两种情况(当然不是相互排斥的):
  • Supplier<Shape>对象由工厂类实例化。
    例如,我们可以在工厂中使用 Map存储 Supplier<Shape>实例和
    修改代码以添加/删除 map 中的元素确实更清晰,因为将新分支添加到 if-else if/switch声明因为它不那么冗长,并且更改 map 填充的方式(添加 map.put() 或删除 map.put() 语句)不太容易产生副作用。
  • Supplier<Shape>对象由客户端类实例化和提供。
    在这种情况下,工厂甚至不需要改变。
    所以 map 甚至不需要 .
    而从客户端,同时客户端提供了一个有效的 Supplier<Shape>参数,没问题 .

  • 类型安全,可维护代码:正如您可能注意到的,这两种方式使用 Supplier<Shape>彻底解决new MyShape()的弊端和通过反射习语实例化。

    我将举两个例子来说明这两种方式。

    示例其中 Shape Supplier s 是在工厂中创建的:
    public class SimpleShapeFactory {
    
    
        private static Map<String, Supplier<Shape>> shapesByCriteria = new HashMap<>();
    
        static {
            shapesByCriteria.put("square", Square::new);
            shapesByCriteria.put("circle", Circle::new);
        }
    
        public static Shape createShape(String criteria) {
            return shapesByCriteria.get(criteria).get();
        }           
    
    }
    

    客户端可以这样调用它:
    Shape square = SimpleShapeFactory.createShape("square");
    Shape circle = SimpleShapeFactory.createShape("circle");
    

    由于 Square 的实例化,此代码不会在运行时失败或 Circle因为它是在编译时检查的。
    以及实例化的任务 Shape实例在同一个地方并且很容易改变:
    static {
        shapesByCriteria.put("square", Square::new);
        shapesByCriteria.put("circle", Circle::new);
    }
    

    示例其中 Shape Supplier s 由客户提供:
    public class ComplexShapeFactory {
    
        public static Shape composeComplexShape(List<Supplier<Shape>> suppliers) {
            Shape shape = suppliers.get(0);
            for (int i = 1; i < suppliers.size() - 1; i++) {
                shape = shape.compose(suppliers.get(i + 1).get());
            }
    
            return shape;
        }    
    
    }
    

    客户端可以通过链接 Supplier 创建复杂的形状以这种方式调用方法:
    Shape squareWithTwoCircles = ComplexShapeFactory.composeComplexShape(Arrays.asList(Square::new, Circle::new, Circle::new));
    

    检查仍然在编译时完成,由于供应商由客户提供,客户可以添加新类 Shape无需进行工厂更改。

    关于java - java 8 供应商是否正在替换构造函数调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45464032/

    有关java - java 8 供应商是否正在替换构造函数调用的更多相关文章

    1. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

      给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

    2. ruby - 检查数组是否在增加 - 2

      这个问题在这里已经有了答案:Checktoseeifanarrayisalreadysorted?(8个答案)关闭9年前。我只是想知道是否有办法检查数组是否在增加?这是我的解决方案,但我正在寻找更漂亮的方法:n=-1@arr.flatten.each{|e|returnfalseife

    3. ruby - 在没有 sass 引擎的情况下使用 sass 颜色函数 - 2

      我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re

    4. java - 等价于 Java 中的 Ruby Hash - 2

      我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

    5. ruby-on-rails - 在 ruby​​ 中使用 gsub 函数替换单词 - 2

      我正在尝试用ruby​​中的gsub函数替换字符串中的某些单词,但有时效果很好,在某些情况下会出现此错误?这种格式有什么问题吗NoMethodError(undefinedmethod`gsub!'fornil:NilClass):模型.rbclassTest"replacethisID1",WAY=>"replacethisID2andID3",DELTA=>"replacethisID4"}end另一个模型.rbclassCheck 最佳答案 啊,我找到了!gsub!是一个非常奇怪的方法。首先,它替换了字符串,所以它实际上修改了

    6. ruby - 检查字符串是否包含散列中的任何键并返回它包含的键的值 - 2

      我有一个包含多个键的散列和一个字符串,该字符串不包含散列中的任何键或包含一个键。h={"k1"=>"v1","k2"=>"v2","k3"=>"v3"}s="thisisanexamplestringthatmightoccurwithakeysomewhereinthestringk1(withspecialcharacterslike(^&*$#@!^&&*))"检查s是否包含h中的任何键的最佳方法是什么,如果包含,则返回它包含的键的值?例如,对于上面的h和s的例子,输出应该是v1。编辑:只有字符串是用户定义的。哈希将始终相同。 最佳答案

    7. ruby-on-rails - Ruby 检查日期时间是否为 iso8601 并保存 - 2

      我需要检查DateTime是否采用有效的ISO8601格式。喜欢:#iso8601?我检查了ruby​​是否有特定方法,但没有找到。目前我正在使用date.iso8601==date来检查这个。有什么好的方法吗?编辑解释我的环境,并改变问题的范围。因此,我的项目将使用jsapiFullCalendar,这就是我需要iso8601字符串格式的原因。我想知道更好或正确的方法是什么,以正确的格式将日期保存在数据库中,或者让ActiveRecord完成它们的工作并在我需要时间信息时对其进行操作。 最佳答案 我不太明白你的问题。我假设您想检查

    8. ruby - 检查日期是否在过去 7 天内 - 2

      我的日期格式如下:"%d-%m-%Y"(例如,今天的日期为07-09-2015),我想看看是不是在过去的七天内。谁能推荐一种方法? 最佳答案 你可以这样做:require"date"Date.today-7 关于ruby-检查日期是否在过去7天内,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/32438063/

    9. ruby - 在 Ruby 中有条件地定义函数 - 2

      我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

    10. ruby - 如何验证 IO.copy_stream 是否成功 - 2

      这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下

    随机推荐