java 静态工厂_高效Java第一条考虑用静态工厂代替构造函数

article/2025/9/17 10:52:53

获得类的实例:

1.提供一个公有的构造函数;

2.提供一个公有的静态工厂方法,该方法只是一个返回类的实例的静态方法。

5099341ad0fc

静态工厂方法与设计模式中的工厂方法模式不同。

提供静态工厂方法的优势——静态工厂方法与构造函数不同的第一大优势在于,它们有名称

如果构造函数的参数本身没有确切地描述正被返回的对象,那么具有适当名称的静态工厂会更容易被使用,产生的客户端代码也更易于阅读。

BigInteger(int,int,Random)返回的BigInteger可能为素数,如果用名为BigInteger.probablePrime的静态方法来表示,会更为清楚。

一个类只能有一个带有指定签名的构造函数。

提供两个构造函数,它们的参数列表只在参数类型的顺序上有所不同。这不是好主意。用户永远也记不住该用哪个构造函数,因此会经常调用错误的构造函数。并且,读调用这些构造函数的代码是有难度的。

当一个类需要多个带有相同签名的构造函数时,可以用静态工厂方法代替构造函数,通过慎重地选择名称可以突出它们之间的区别。

提供静态工厂方法的优势——静态工厂方法与构造函数不同的第二大优势在于,不必在每次调用它们的时候都创建一个新的对象

可以避免创建不必要的重复对象。

Booelan.valueOf(booelan)方法从不创建对象。

类似于Flyweight模式。

如果程序经常请求创建相同的对象,并且创建对象的代价很好,则这项技术可以极大地提升性能。

实例受控的类:静态工厂方法能够为重复的调用返回相同的对象,这样有助于类总能严格控制在某个时刻那些实例应该存在。

提供静态工厂方法的优势——静态工厂方法与构造函数不同的第三大优势在于,它们可以返回原返回类型的任何子类型的对象

API可以返回对象,又不会使对象的类变成公有的。以这种方式隐藏实现类会使API变得非常简洁。

这项技术适用于基于接口的框架,接口为静态工厂方法提供了自然返回类型。

接口不能有静态方法,因此按照惯例,接口Type的静态工厂方法被放在一个名为Types的不可实例化的类中。

Collections Framework API比导出32个独立公有类的实现方式要小得多,每种便利实现都对应一个类。这不仅是API数量上减少,也是概念意义上的减少。

被返回的对象是由相关的接口精确指定的。

使用这种静态工厂方法时,甚至要求客户端通过接口来引用被返回的对象,而不是通过它的实现类来引用被返回的对象,这是一种良好的习惯。

公有的静态工厂方法所返回的对象的类不仅可以是非公有的,而且该类还可以随着每次调用而发生变化,这取决于静态工厂方法的参数值。只要是已声明的返回类型的子类型,都是允许的。为了提升软件的可维护性和性能,返回对象的类也可能随着发行版本的不同而不同。

JDK5引入的java.util.EnumSet没有公有构造函数,只有静态工厂方法。它返回两种实现类之一,取决于底层枚举类型的大小:如果元素有64个或者更少,静态工厂方法就会返回一个RegularEnumSet实例,用单个long进行支持;如果枚举类型有65个或者更多元素,工厂就返回JumboEnumSet实例,用long数组进行支持。

这两个实现类对客户来说是不可见的。如果RegularEnumSet不能再给小的枚举类型提供性能优势,就可能从未来的发行版本中被删除,这不会造成不良的影响。如果事实证明对性能有好处,在未来的发行版本中可以添加其他的EnumSet实现。客户永远不知道也不关心他们从工厂方法中得到的对象的类,他们只需要知道它是EnumSet的某个子类即可。

静态工厂方法返回的对象所属的类,在编写包含该静态工厂方法的类时可以不必存在。这种灵活的静态工厂方法构成了服务提供者框架的基础(JDBC).

提供静态工厂方法的优势——静态工厂方法与构造函数不同的第四大优势在于,在创建参数化类型实例的时候,它们使代码变得更加简洁。

在调用参数化类的构造函数时,即使类型参数很明显,也必须指明。也就是说要接连两次提供类型参数:

5099341ad0fc

使用静态工厂方法,编译器可以推导出类型参数(类型推导)。

5099341ad0fc

5099341ad0fc

JDK8竟然还会警告。

静态工厂方法的主要缺点在于:类如果不含公有的或者受保护的构造函数,就不能被子类化。

对于公有的静态工厂所返回的非公有类,也是如此。

鼓励程序员使用复合,而不是继承。

静态工厂方法的第二个缺点在于,它们与其他的静态方法没有任何区别。

在API文档中,它们没有像构造函数那样在API文档中明确地标识出来。

对于提供了静态工厂方法而不是构造函数的类来说,要想查明如何实例化一个类,是非常困难的。

静态工厂方法的惯用名称

valueOf——该方法返回的实例与它的参数具有相同的值。实际上是类型转换方法。

of——valueOf的简洁替代,在EnumSet中使用并流行起来。

getInstance——返回的实例是通过方法的参数来描述的,但是不能够说与参数具有同样的值。对于单例来说,该方法没有参数,并返回唯一的实例。

newInstance——与getInstance一样,但newInstance确保返回的每个实例都与所有其他实例不同。

getType——在工厂方法处于不同的类中的时候使用。Type表示工厂方法所返回的对象类型。

newType——在工厂方法处于不同的类中的时候使用。Type表示工厂方法所返回的对象类型。

服务提供者框架概念

服务提供者框架:多个服务提供者实现一个服务,系统为服务提供者的客户提供多个实现,并把他们从多个实现中解耦出来。

服务提供者三个组件

服务接口:提供者实现

提供者注册API:系统用来注册实现,让客户访问它们

服务访问API:客户端用来获取服务的实例

服务访问API允许但是不要求客户端指定某个选择提供者的条件,如果没有指定,则返回默认的实现。服务访问API是灵活的静态工厂,它构成了服务提供者框架的基础。

服务提供者可选组件

服务提供者接口,这些提供者负责创建其服务实现的实例。如果没有服务提供者接口,实现就按照类名称注册,并通过反射方式进行实例化。

服务提供者例子——JDBC

Connection服务接口

DriverManager.registerDriver是提供者注册API

DriverManager.getConnection是服务访问API

Driver是服务提供者接口

5099341ad0fc

5099341ad0fc

编写实例受控的类的原因

实例受控使得类可以确保它是一个单例或是不可实例化的。

实例受控使得不可变类可以确保不会存在两个相等的实例,即当且仅当a==b的时候才有a.equals(b)为true。如果类保证了这一点,它的客户端就可以使用==操作符来代替equals(Object)方法,这样可以提升性能。

总结

静态工厂通常更加适合,因此切忌第一反应就是提供公有的构造函数,而不优先考虑静态工厂。


http://chatgpt.dhexx.cn/article/Ca9KlHl5.shtml

相关文章

【算法之高效求素数】浅析求素数算法

注意: 如果没有特殊说明, 以下讨论的都是针对n为素数时的时间复杂度 1. 根据概念判断: 如果一个正整数只有两个因子, 1和p&#xff0c;则称p为素数. 代码: bool isPrime(int n) {if(n < 2) return false;for(int i 2; i < n; i)if(n%i 0) return false;return true;…

求素数算法

注意: 如果没有特殊说明, 以下讨论的都是针对n为素数时的时间复杂度 1. 根据概念判断: 如果一个正整数只有两个因子, 1和p&#xff0c;则称p为素数. 代码: bool isPrime(int n) {if(n < 2) return false;for(int i 2; i < n; i)if(n%i 0) return false;return true;…

求素数算法-网摘

摘自&#xff1a;http://www.cnblogs.com/luluping/archive/2010/03/03/1677552.html 浅析求素数算法注意: 如果没有特殊说明, 以下讨论的都是针对n为素数时的时间复杂度1. 根据概念判断:如果一个正整数只有两个因子, 1和p&#xff0c;则称p为素数. 代码: bool isPrime(int n) …

用JAVA编写50以内的素数_java求50以内的素数

java求50以内的素数 [2021-02-01 12:46:22] 简介: python求100内的所有素数的方法&#xff1a;使用判断该数除了1和它本身以外不再有其他因数即可&#xff0c;代码为【i2 for i in range(2,100): if(i%j0):break else:num.append(i)】。相关免费学 php去除nbsp的方法&#xff…

浅析求素数算法

原文地址为&#xff1a; 浅析求素数算法 2009-12-05 16:15:31 阅读18 评论0 字号&#xff1a;大中小 浅析求素数算法时间: 2006-10-27注意: 如果没有特殊说明, 以下讨论的都是针对n为素数时的时间复杂度1. 根据概念判断:如果一个正整数只有两个因子, 1和p&#xff0c;则称p为…

Python中的所有运算符以及运算符的优先级

文章目录 前言一、算术运算符二、赋值运算符三、比较运算符四、逻辑运算符五、位运算符六、运算符的优先级案例实战总结 前言 运算符是一些人为定义的特殊符号&#xff0c;比如我们生活当中最常见的 、-、、&#xff0c;它们主要用于数学计算、比较大小和逻辑运算等等。那这些…

在c语言中逻辑运算符的优先级是,c语言中逻辑运算符优先级是什么?

c语言中逻辑运算符优先级由高到低依次是&#xff1a;&#xff01;(逻辑非)、&&(逻辑与)、||(逻辑或)。逻辑表达式的值为逻辑值&#xff1b;逻辑值分为逻辑真值和逻辑假值&#xff0c;在判断时&#xff0c;仅有零值被判断为逻辑假值(false)&#xff0c;一切非零值均可被…

算数运算符,比较运算符,逻辑运算符,赋值运算符,运算优先级

文章目录 1.算数运算符1.1注意事项2.关系运算符&#xff0c;比较运算符2.1注意事项 3.逻辑运算符3.1注意事项 4.赋值运算法4.1注意特点 5.运算优先级 1.算数运算符 package mainimport "fmt"func main() {//说明&#xff0c;如果运算的数都是整数&#xff0c;那么除后…

【C语言】运算符的优先级

目录 算术运算符&#xff1a; 关系运算符&#xff1a; 逻辑运算符 三种运算符优先级高低比较&#xff1a; 算术运算符&#xff1a; 5种算术运算符&#xff1a;, -, *, /, % 优先级&#xff1a;*&#xff0c;/&#xff0c;%大于&#xff0c;- 关系运算符&#xff1a; 6种…

逻辑运算符与++的使用与优先级差别

我们的教材通常会按级次区别优先级&#xff0c;如下图所示&#xff1a; 那对于一个简单的编程题&#xff1a; int a,b,c,y;abc1;y a||b&&c 此时输出结果为&#xff1a;y1,a2,b1,c1 按照逻辑运算符&&优先于||&#xff0c;结果不应该为2&#xff0c;2&#xff…

逻辑运算符、位运算符、移位运算符、三目运算符、运算符的优先级

一、逻辑运算符 &#xff1a;逻辑运算符的作用是用于连接布尔表达式的。 1、& &#xff08;与&#xff0c;并且&#xff09; 规律&#xff1a; 只有左右变量同时为true&#xff0c;那么结果才是true&#xff0c;否则就false。 2、| &#xff08;或&#xff0c;或者&…

C/C++逻辑运算符 | | 和 的优先级误区——优先级决定运算顺序吗?

C运算符优先级表&#xff1a; 根据这个表格&#xff0c;我们很容易得出结论&#xff1a;&&的优先级大于|| 下面看一段代码&#xff0c;来验证这个结论 #include<iostream> using namespace std; int main() {int x 3;int y 1 || (x 1) && (x 1);pr…

运算符的优先级顺序

运算符优先级口诀&#xff1a;单目算术位关系&#xff0c;逻辑三目后赋值。 单目&#xff1a; 、- - 算术&#xff1a;、-、*、/、% 位&#xff1a;位移运算符&#xff1a;<<&#xff08;左移位&#xff09;、>>&#xff08;带符号右移位&#xff09;、>>>…

c++运算符优先级归纳

C一共有 18个优先级&#xff0c;运算中按优先级进行性计算&#xff0c;当优先级相同时&#xff0c;根据结合性规则来决定。 结合性&#xff1a; 1.从左到右&#xff08;L-R&#xff09;&#xff1a;操作数和操作符结合的顺序大部分是从左到右结合性的&#xff0c;例如&#xff…

C++运算符的优先级

有括号的优先级最高&#xff0c;涉及对象的运算符&#xff1b; 自增自减&#xff0c;取值&#xff0c;取地址运算符&#xff0c;逻辑非&#xff0c;按位取反&#xff0c;强制类型转换&#xff0c;长度运算符 先乘除后加减 左移右移运算符 比较运算符、三目运算符 各类赋值运算…

运算符优先级

C语言中&#xff0c;运算符除了常见的三大类&#xff0c;算术运算符、关系运算符与逻辑运算符之外&#xff0c;还有一些用于完成特殊任务的运算符。 运算符的运算优先级共分为15 级&#xff0c;1 级最高&#xff0c;15 级最低。 在表达式中&#xff0c;优先级较高的先于优先级…

【博客624】MAC地址表、ARP表、路由表(RIB表)、转发表(FIB表)

MAC地址表、ARP表、路由表(RIB表/FIB表) MAC地址表 MAC地址表是交换机等网络设备记录MAC地址和端口的映射关系&#xff0c;代表了交换机从哪个端口学习到了某个MAC地址&#xff0c;交换机把这个信息记录下来&#xff0c;后续交换机需要转发数据的时候就可以根据报文的目的MAC地…

【新星计划-2023】什么是ARP?详解它的“解析过程”与“ARP表”。

一、什么是ARP ARP&#xff08;地址解析协议&#xff09;英文全称“Address Resolution Protocol”&#xff0c;是根据IP地址获取物理地址的一个TCP/IP协议。主机发送信息时将包含目标IP地址的ARP请求广播到局域网络上的所有主机&#xff0c;并接收返回消息&#xff0c;以此确…

ARP表 MAC表 路由表

ARP表是一个动态表&#xff0c;存储在计算机当中&#xff0c;目的是做一个ip地址与mac地址的对应。 假设在同一子网段&#xff0c;计算机A与计算机B通信 计算机A的ip地址192.168.0.1 MAC地址AA-AA-AA-AA-AA-AA 计算机B的ip地址已知192.168.0.2 MAC地址未知 那么计算机A就要…

Linux上ARP表,路由表的解释

为厘清数据在网络中的传输流向和具体的实现过程&#xff0c;现对ARP表&#xff0c;路由表进行学习如下。 ARP&#xff08;Address Resolution Protocol&#xff09;地址解析协议。IP数据包经常通过以太网进行发送&#xff0c;以32位的IP地址作为源/目的地址。但是以太网设备不能…