您好,登錄后才能下訂單哦!
如何在Java 10項目中使用var關(guān)鍵字?針對這個問題,這篇文章詳細介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。
介紹
Java 10引入了一個閃亮的新功能:局部變量類型推斷。對于局部變量,現(xiàn)在可以使用特殊的保留類型名稱“var”代替實際類型,如下所示:
var name = “Mohamed Taman”;
提供這個特性是為了增強Java語言,并將類型推斷擴展到局部變量的聲明上。這樣可以減少板代碼,同時仍然保留Java的編譯時類型檢查。
由于編譯器需要通過檢查賦值等式右側(cè)(RHS)來推斷var的實際類型,因此在某些情況下,這個特性具有局限性。我會在稍后提到這個問題。現(xiàn)在,讓我們來看一些簡單的例子吧。
在開始演示代碼之前,你需要一個IDE來體驗這些新特性。現(xiàn)在有很多可選擇的IDE,所以你可以在它們當(dāng)中選擇你喜歡的能夠支持Java SE 10的IDE,比如Apache NetBeans 9、IntelliJ IDEA 2018或最新版本的Eclipse。
就個人而言,我更喜歡使用交互式的編程工具,可以快速學(xué)習(xí)Java語言語法,了解新的Java API及其特性,甚至用來進行復(fù)雜代碼的原型設(shè)計。這與枯燥的編輯、編譯和執(zhí)行代碼的繁瑣過程不太一樣:
寫一個完整的程序;
編譯并修復(fù)錯誤;
運行程序;
弄清楚它有什么問題;
修改;
重復(fù)這個過程。
除了IDE之外,現(xiàn)在還可以使用從Java SE 9以就隨ava SE JDK一起發(fā)布的JShell。
什么是JShell
現(xiàn)在,Java有了自己的REPL(Read-Evaluate-Print-Loop)實現(xiàn)JShell(Java Shell),作為交互式的編程環(huán)境。那么,它有什么神奇的地方?JShell提供了一個快速友好的環(huán)境,讓你能夠快速探索、發(fā)現(xiàn)和試驗Java語言特性及其豐富的庫。
在JShell中,你可以一次輸入一個程序元素,并可以立即看到結(jié)果,然后根據(jù)需要對代碼做出調(diào)整。因此,JShell用它的Read-Evaluate-Print循環(huán)取代了編輯、編譯和執(zhí)行的繁瑣過程。在JShell中,你不需要編寫完整的程序,只需要編寫JShell命令和Java代碼片段即可。
當(dāng)你輸入代碼段時,JShell會立即讀取、執(zhí)行并打印結(jié)果,然后準備好執(zhí)行下一個代碼片段。因此,JShell的即時反饋可以讓你保持注意力,提高你的效率,并加快學(xué)習(xí)和軟件開發(fā)過程。
對JShell的介紹就到此為止(InfoQ最近對這個工具進行過全面介紹)。為了深入了解JShell的功能,我錄制了一套視頻教程“Hands-on Java 10 Programming with JShell”,可以幫助你掌握JShell,可以從Packt或Udemy訪問這些教程。
現(xiàn)在,讓我們通過一些簡單的示例(使用JShell)來了解這個新的var類型能做些什么。
必備軟件
為了能用上JShell,我假設(shè)你安裝了Java SE或JDK 10+,并且JDK的bin目錄已經(jīng)加入到系統(tǒng)路徑中。如果還沒有安裝,可以在這里下載JDK 10+ 最新版本。
啟動JShell會話
在Windows上,打開命令提示符,輸入jshell并按回車鍵。
在Linux上,打開一個shell窗口,輸入jshell并按回車鍵。
在macOS(以前稱為OS X)上,打開終端窗口,輸入“jshell”并按回車鍵。
這個命令會啟動一個新的JShell會話,并顯示這個消息:
| Welcome to JShell -- Version 10.0.1 | For an introduction type: /help intro jshell>
使用“var”類型
現(xiàn)在你已經(jīng)安裝了JDK 10,現(xiàn)在讓我們開始玩JShell。我們直接跳到終端,通過示例來了解var類型。只需在jshell提示符下輸入我接下來要介紹的每個代碼片段,我會把結(jié)果留給你作為練習(xí)。如果你稍微有瞄過一兩眼在代碼,你會注意到它們看起來好像是錯的,因為當(dāng)中沒有分號。你可以試試看,看看能不能運行。
簡單的類型推理
這是var類型的基本用法,在下面的示例中,編譯器可以將RHS推斷為String字面量:
var name = "Mohamed Taman" var lastName = str.substring(8) System.out.println("Value: "+lastName +" ,and type is: "+ lastName.getClass().getTypeName())
這里不需要分號,因為JShell是一個交互式環(huán)境。只有當(dāng)同一行代碼有多個語句或一個類型聲明或方法聲明中有多個語句時才需要分號,你將在后面的示例中看到。
var類型和繼承
在使用var時,多態(tài)仍然有效。在繼承的世界中,var類型的子類型可以像平常一樣賦值給超類型的var類型,如下所示:
import javax.swing.* var password = new JPasswordField("Password text") String.valueOf(password.getPassword()) // // 將密碼的字符數(shù)組轉(zhuǎn)換成字符串 var textField = new JTextField("Hello text") textField = password textField.getText()
但不能將超類型var賦值給子類型var,如下所示:
password = textField
這是因為JPasswordField是JTextField的子類。
var和編譯時安全性
如果出現(xiàn)錯誤的賦值操作會怎樣?不兼容的變量類型不能相互賦值。一旦編譯器推斷出實際類型的var,就不能將錯誤的值賦值給它,如下所示:
var number = 10 number = "InfoQ"
這里發(fā)生了什么?編譯器將“var number = 10
”替換為“int number = 10”,
所以仍然可以保證安全性。
var與集合和泛型
現(xiàn)在讓我們來看看var與集合和泛型一起使用時如何進行類型推斷。我們先從集合開始。在下面的情況中,編譯器可以推斷出集合元素的類型是什么:
var list = List.of(10);
這里沒有必要進行類型轉(zhuǎn)換,因為編譯器已經(jīng)推斷出正確的元素類型為int。
int i = list.get(0); //等效于: var i = list.get(0);
下面的情況就不一樣了,編譯器只會將其作為對象集合(而不是整數(shù)),因為在使用菱形運算符時,Java需要LHS(左側(cè))的類型來推斷RHS的類型:
var list2 = new ArrayList<>(); list2.add(10); list2 int i = list2.get(0) //編譯錯誤 int i = (int) list2.get(0) //需要進行轉(zhuǎn)換,獲得int
對于泛型,最好在RHS使用特定類型(而不是菱形運算符),如下所示:
var list3 = new ArrayList<Integer>(); list3.add(10); System.out.println(list3) int i = list3.get(0)
for循環(huán)中的var類型
讓我們先來看看基于索引的For循環(huán):
for (var x = 1; x <= 5; x++) { var m = x * 2; //等效于: int m = x * 2; System.out.println(m); }
下面是在For Each循環(huán)中:
var list = Arrays.asList(1,2,3,4,5,6,7,8,9,10) for (var item : list) { var m = item + 2; System.out.println(m); }
現(xiàn)在我有一個問題,var是否適用于Java 8 Stream?讓我們看看下面的例子:
var list = List.of(1, 2, 3, 4, 5, 6, 7) var stream = list.stream() stream.filter(x -> x % 2 == 0).forEach(System.out::println)
var類型和三元運算符
那么三元運算符呢?
var x = 1 > 0 ? 10 : -10 int i = x
現(xiàn)在,如果在三元運算符的RHS中使用不同類型的操作數(shù)會怎樣?讓我們來看看:
var x = 1 > 0 ? 10 : "Less than zero"; System.out.println(x.getClass()) //Integer var x = 1 < 0 ? 10 : "Less than zero"; System.out.println(x.getClass()) // String
這兩個例子是否可以說明var的類型是在運行時決定的?絕對不是!讓我們以舊方式實現(xiàn)同樣的邏輯:
Serializable x = 1 < 0 ? 10 : "Less than zero"; System.out.println(x.getClass())
Serializable是其中兩個操作數(shù)最具兼容性和最專的有類型(最不專有的類型是java.lang.Object)。
String和Integer都實現(xiàn)了Serializable。Integer從int自動裝箱。換句話說,Serializable是兩個操作數(shù)的LUB(最小上限)。所以,這表明往前數(shù)第三個例子中的var類型也是Serializable。
讓我們轉(zhuǎn)到另一個主題:將var類型傳給方法。
var類型與方法
我們先聲明一個名為squareOf的方法,這個方法的參數(shù)為BigDecimal類型,并返回參數(shù)的平方,如下所示:
BigDecimal squareOf(BigDecimal number) { var result= number.multiply(number); return result; } var number = new BigDecimal("2.5") number = squareOf(number)
現(xiàn)在讓我們看看它如何與泛型一起使用。我們聲明一個名為toIntgerList的方法,參數(shù)類型為List<T>(泛型類型),并使用Streams API返回一個整數(shù)列表,如下所示:
<T extends Number> List<Integer> toIntgerList(List<T> numbers) { var integers = numbers.stream() .map(Number::intValue) .collect(Collectors.toList()); return integers; } var numbers = List.of(1.1, 2.2, 3.3, 4.4, 5.5) var integers = toIntgerList(numbers)
var類型與匿名類
最后,讓我們看一下var和匿名類。我們通過實現(xiàn)Runnable接口來使用線程,如下所示:
<T extends Number> List<Integer> toIntgerList(List<T> numbers) { var integers = numbers.stream() .map(Number::intValue) .collect(Collectors.toList()); return integers; } var numbers = List.of(1.1, 2.2, 3.3, 4.4, 5.5) var integers = toIntgerList(numbers)
到目前為止,我已經(jīng)介紹了Java 10的新特性——“var”類型,它減少了樣板編碼,同時保持了Java的編譯時類型檢查。我還通過實例說明了可以用它做些什么。接下來,你將了解var類型的局限性以及不能將它用在哪些地方。
var message = "running..." //effectively final var runner = new Runnable(){ @Override public void run() { System.out.println(message); }} runner.run()
“var”的局限性
接下來,你將看一些示例,以便了解var類型功能無法做到的事情。
jshell提示符將會告訴你代碼出了什么問題,你可以利用這些交互式的即時反饋。
應(yīng)該要進行初始化
第一個也是最簡單的原則就是不允許沒有初始值的變量。
var name;
你將得到一個編譯錯誤,因為編譯器無法推斷這個局部變量x的類型。
不允許復(fù)合聲明
嘗試運行這行代碼:
var x = 1, y = 3, z = 4
你將得到一個錯誤消息:復(fù)合聲明中不允許使用'var'。
不支持確定性賦值(Definite Assignment)
嘗試創(chuàng)建一個名為testVar的方法,如下所示,將下面的代碼復(fù)制并粘貼到JShell中:
void testVar(boolean b) { var x; if (b) { x = 1; } else { x = 2; } System.out.println(x); }
方法不會被創(chuàng)建,而是會拋出編譯錯誤。因為沒有設(shè)置初始值,所以不能使用'var'。
null賦值
不允許進行null賦值,如下所示:
var name = null;
這將拋出異常“variable initializer is 'null'”。因為null不是一個類型。
與Lambda一起使用
另一個例子,沒有Lambda初始化器。這與菱形操作符那個示例一樣,RHS需要依賴LHS的類型推斷。
var runnable = () -> { }
將拋出異常:“l(fā)ambda expression needs an explicit target-type”。
var和方法引用
沒有方法引用初始值,類似于Lambda和菱形運算符示例:
var abs = BigDecimal::abs
將拋出異常:“method reference needs an explicit target-type”。
var和數(shù)組初始化
并非所有數(shù)組初始化都有效,讓我們看看什么時候var與[]不起作用:
var numbers[] = new int[] { 2, 4, 6 }
以下也不起作用:
var numbers = { 2, 4, 6 }
拋出的錯誤是: “array initializer needs an explicit target-type”。
就像上一個例子一樣,var和[]不能同時用在LHS一邊:
var numbers[] = { 2, 4, 6 }
錯誤: 'var' is not allowed as an element type of an array。
只有以下數(shù)組初始化是有效的:
var numbers = new int[] { 2, 4, 6 } var number = numbers[1]number = number + 3
不允許使用var字段
class Clazz { private var name; }
不允許使用var方法參數(shù)
void doAwesomeStuffHere(var salary) { }
不能將var作為方法返回類型
var getAwesomeStuff() { return salary; }
catch子句中不能使用var
try { Files.readAllBytes(Paths.get("c:\temp\temp.txt")); } catch (var e) { }
在編譯時var類型究竟發(fā)生了什么?
“var”實際上只是一個語法糖,并且它不會在編譯的字節(jié)碼中引入任何新的結(jié)構(gòu),在運行期間,JVM也沒有為它們提供任何特殊的指令。
關(guān)于如何在Java 10項目中使用var關(guān)鍵字問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。