Java无处不在:使用DukeScript在任何地方运行一次编写
在相當(dāng)長(zhǎng)一段時(shí)間內(nèi),Java都未能兌現(xiàn)“一次編寫(xiě),隨處運(yùn)行”的承諾。 DukeScript希望通過(guò)在跨平臺(tái)應(yīng)用程序中實(shí)現(xiàn)視圖和邏輯的清晰分離來(lái)改變這種狀況。 在本文中,一個(gè)簡(jiǎn)單的場(chǎng)景用于介紹DukeScript的基礎(chǔ)。
多年來(lái),Java Swing使開(kāi)發(fā)人員能夠編寫(xiě)可以在任何操作系統(tǒng)上運(yùn)行的應(yīng)用程序。 隨著智能手機(jī),平板電腦和嵌入式計(jì)算機(jī)的問(wèn)世,一切都結(jié)束了。 在企業(yè)中,臺(tái)式機(jī)占據(jù)了很多年。 但是,與此同時(shí),每個(gè)IT項(xiàng)目都包括未來(lái)的計(jì)劃,將來(lái)需要將應(yīng)用程序移植到移動(dòng)平臺(tái)。 為所有平臺(tái)構(gòu)建本機(jī)應(yīng)用程序需要特殊技能,并且在維護(hù)和開(kāi)發(fā)方面都非常昂貴。 什么是要做?
DukeScript( DukeScript.com )再次為您提供了基于Java的解決方案,使您可以開(kāi)發(fā)跨平臺(tái)應(yīng)用程序。 DukeScript提供了視圖和邏輯的清晰分隔,使UI設(shè)計(jì)人員可以專(zhuān)注于UI設(shè)計(jì),而編碼人員可以專(zhuān)注于編寫(xiě)經(jīng)過(guò)良好測(cè)試的應(yīng)用程序代碼。
兩全其美的
DukeScript的基本思想很簡(jiǎn)單。 每個(gè)操作系統(tǒng)都可以運(yùn)行基本的Java應(yīng)用程序。 在Android上,這是通過(guò)Dalvik Runtime和ART本身完成的,在iOS上您具有RoboVM ,在其他許多平臺(tái)上,您具有OpenJDK和Oracle的Java SE Embedded 。 同時(shí),有一系列Java虛擬機(jī)可用于瀏覽器( TeaVM , Doppio , Bck2Brwsr ),這些虛擬機(jī)無(wú)需瀏覽器插件即可運(yùn)行。 但是,這里缺少的是統(tǒng)一視圖技術(shù),而與此同時(shí),幾乎所有平臺(tái)上都提供了現(xiàn)代HTML渲染器組件。 將這些不同的技術(shù),虛擬機(jī)和組件組合在一起時(shí),便具有了全面框架的基礎(chǔ)。
通過(guò)將所有這些部分組合在一起,可以同時(shí)利用其所有優(yōu)勢(shì)。 例如,由于Java具有靜態(tài)類(lèi)型,因此在所有編程語(yǔ)言中,Java都提供了最佳的IDE支持,并使可維護(hù)的代碼得以編寫(xiě)并易于重構(gòu)。 由于這些原因,它比JavaScript更適合用作大型項(xiàng)目的語(yǔ)言。 在UI方面,對(duì)于HTML和CSS,您可以訪問(wèn)免費(fèi)的商業(yè)框架和服務(wù)庫(kù)。 當(dāng)UI和業(yè)務(wù)邏輯彼此清晰地分開(kāi)時(shí),我們可以毫無(wú)例外地,不受任何限制地利用完整的武器庫(kù)。 為了證明這些觀點(diǎn),我們現(xiàn)在將開(kāi)發(fā)和設(shè)計(jì)待辦事項(xiàng)應(yīng)用程序。
ViewModel
DukeScript使用Model-View-ViewModel(MVVM)設(shè)計(jì)模式將可視化和邏輯分離。 View是用標(biāo)記語(yǔ)言定義的,并聲明性地將活動(dòng)元素綁定到ViewModel的屬性。 通過(guò)這種體系結(jié)構(gòu),ViewModel不需要了解View。 無(wú)需任何更改,即可從ViewModel換出View。 所有View邏輯都在ViewModel中定義。 在MVVM模式中,模型是應(yīng)用程序的其余部分,而如何可視化模型則是不確定的,因此是無(wú)限的。
讓我們從ViewModel開(kāi)始。 清單1顯示了如何創(chuàng)建ViewModel。 @Model批注確保將生成名為T(mén)ask的類(lèi)。 同時(shí),將自動(dòng)創(chuàng)建屬性title和complete的 setter和getter。 這使開(kāi)發(fā)人員不必編寫(xiě)一堆代碼,并且ViewModel類(lèi)的結(jié)構(gòu)緊湊且一目了然。 創(chuàng)建過(guò)程會(huì)在后臺(tái)自動(dòng)進(jìn)行,因此在IDE中進(jìn)行開(kāi)發(fā)時(shí)可以立即使用該類(lèi)。
清單1
@Model(className = "Task", properties = { @Property(name = "title", type = String.class), @Property(name = "complete", type = boolean.class) }) public static class TaskModel {}對(duì)于更復(fù)雜的任務(wù),我們可以包裝模型。 清單2顯示了一個(gè)TaskListViewModel ,它表示Tasks列表以及其他屬性。 @Function批注標(biāo)記可以從View調(diào)用的方法。
清單2
@Model(className = "TaskListViewModel", properties = {@Property(name = "input", type = String.class),@Property(name = "tasks", type = Task.class, array = true),@Property(name = "editing", type = Task.class) }, targetId = "body") final class TaskListViewModelDefinition {@Functionpublic static void editTask(TaskListViewModel list, Task data) {list.setEditing(data);}@Functionpublic static void stopEditing(TaskListViewModel list) {list.setEditing(null);}@Function@ModelOperationpublic static void deleteTask(TaskListViewModel model, Task data) {model.getTasks().remove(data);}@Function@ModelOperationpublic static void addTask(TaskListViewModel model) {if (null == model.getInput() || model.getInput().length() == 0) {return;}Task task = new Task(model.getInput(), false);model.setInput(""); model.getTasks().add(task);} }使用DukeScript進(jìn)行單元測(cè)試
上面代碼中的兩個(gè)方法被標(biāo)記為@ModelOperation 。 在DukeScript中,您使用的方法也需要從View外部調(diào)用,在我們的示例(清單3)中,這是一個(gè)單元測(cè)試。 該測(cè)試顯示了如何使用生成的ViewModel。 在第一個(gè)測(cè)試用例中,我們模擬用戶(hù)輸入一個(gè)新任務(wù)(輸入一個(gè)輸入( setInput )并確認(rèn)輸入,例如通過(guò)按鈕或Enter鍵( addTask )來(lái)鍵入新任務(wù)。 即使還沒(méi)有View,我們已經(jīng)可以測(cè)試ViewModel的方法了。 這種情況確實(shí)很好地顯示了組件的干凈去耦。
清單3
public class TodoListTest {@Testpublic void testAddTask() {TaskListViewModel taskList = new TaskListViewModel();Assert.assertEquals(taskList.getTasks().size(), 0);taskList.setInput("Buy milk!");taskList.addTask();Assert.assertEquals(taskList.getTasks().size(), 1);Task task = taskList.getTasks().get(0);Assert.assertEquals(task.getTitle(), "Buy milk!");}@Testpublic void testDeleteTask() {TaskListViewModel taskList = new TaskListViewModel();taskList.getTasks().add(new Task("Buy milk!", false));Assert.assertEquals(taskList.getTasks().size(), 1);Task task = taskList.getTasks().get(0);taskList.deleteTask(task);Assert.assertEquals(taskList.getTasks().size(), 0);} }JSON序列化
當(dāng)您查看定義ViewModel類(lèi)的注釋時(shí),您應(yīng)該注意到它們看起來(lái)有點(diǎn)像JSON消息。 這并非巧合,因?yàn)镈ukeScript能夠輕松地與JSON集成,因此具有巨大的價(jià)值。 ViewModel類(lèi)的toString方法返回JSON字符串。 就像您可以輕松地從JSON字符串創(chuàng)建ViewModel對(duì)象一樣。 清單4展示了ViewModel如何再次序列化和反序列化。 當(dāng)您只需要一個(gè)對(duì)象的副本時(shí),請(qǐng)使用clone方法。 Models.parse的目的是反序列化來(lái)自服務(wù)器或本地持久化數(shù)據(jù)的消息 。
清單4
TaskListViewModel copy; String json = original.toString(); InputStream inputStream = new ByteArrayInputStream( json.getBytes(StandardCharsets.UTF_8)); try {copy = Models.parse(BrwsrCtx.findDefault(TaskListViewModel.class),TaskListViewModel.class, inputStream); } catch (IOException ex) {Logger.getLogger(Main.class.getName()).log(Level.SEVERE, null, ex); }風(fēng)景
默認(rèn)情況下,DukeScript使用HTML定義視圖。 由于將View與ViewModel完全分開(kāi),因此也可以使用其他格式。 dukescript-javafx項(xiàng)目演示了它是如何工作的。 通過(guò)該項(xiàng)目,可以在JavaFX應(yīng)用程序中輕松可視化DukeScript ViewModels。 在這種情況下,用于定義視圖的語(yǔ)言是FXML。 Controls.js還使用另一種格式來(lái)定義View,本文稍后將對(duì)此進(jìn)行討論。
對(duì)于本文中創(chuàng)建的“待辦事項(xiàng)列表”應(yīng)用程序,我們將僅使用清單5所示的標(biāo)準(zhǔn)HTML格式。依賴(lài)于ViewModel的元素將使用data-bind屬性。 這樣,它們以聲明方式綁定到ViewModel的屬性和方法。 同樣,可以用這種方式定義for-each循環(huán)和條件語(yǔ)句。
在大多數(shù)情況下, 數(shù)據(jù)綁定屬性可以完成您需要的所有操作。 但是,有時(shí)您需要?jiǎng)?chuàng)建一個(gè)HTML元素,盡管View可能不需要HTML元素。 對(duì)于這些情況,有特殊的評(píng)論。 在下面清單5的示例中,通過(guò)<!– ko foreach:tasks –>遍歷Tasks列表,而<!– / ko –>關(guān)閉循環(huán)。 可以從清單5中的注釋以及DukeScript網(wǎng)站中了解有關(guān)綁定語(yǔ)法的更多信息。
清單5
<ul><!-- Iterate over the List of Tasks in the TaskListViewModel --><!-- ko foreach: tasks --><li><!-- When the Task is not being edited... --><!-- ko ifnot: $root.editing()===$data --><!-- ...bind the checkbox state to the Task property named "complete" --><input type="checkbox" name="" data-bind="checked: complete"/><!--...bind the text of the span to the Task property named "title" --><span data-bind="text: title"></span><span class="btns"><!-- ...on click, call the 'editTask' method --><button data-bind="click: $root.editTask">Edit</button><!-- ...on click, call the 'deleteTask' method --><button data-bind="click: $root.deleteTask">Delete</button></span><!-- /ko --><!-- When the Task is being edited, show an input field... --><!-- ko if: $root.editing()===$data --><!-- ...on Submit (Enter) call the 'stopEditing' method --><form data-bind="submit: $root.stopEditing"><!-- ...bind the entered text to the Task property named "title"`--><input type="text" data-bind="textInput: title"/></form><!-- /ko --></li><!-- /ko --><li><!-- On Submit (Enter) call the 'addTask' method... --><form data-bind="submit: addTask"><!-- ...bind the entered text to the Task property named "input" --><input type="text" data-bind="textInput: input"/></form></li> </ul>上面定義的功能原型如下圖所示。 從視覺(jué)上看,它可能看起來(lái)有點(diǎn)謙虛,所以讓我們?cè)谝韵赂鞴?jié)中進(jìn)行更改!
設(shè)計(jì)者/開(kāi)發(fā)者實(shí)驗(yàn)
在DukeScript到來(lái)之前,已經(jīng)出現(xiàn)了其他幾個(gè)框架,它們有望將設(shè)計(jì)與開(kāi)發(fā)脫鉤。 在現(xiàn)實(shí)世界中,幾乎沒(méi)有希望保留的諾言。 設(shè)計(jì)往往需要專(zhuān)有工具,這些工具所引起的僅僅是專(zhuān)業(yè)設(shè)計(jì)師的疲倦之笑。 JavaFX就是一個(gè)例子。 使用JavaFX Scene Builder,您甚至都無(wú)法創(chuàng)建多邊形, 而該工具的未來(lái)卻一無(wú)所知 。 此外,創(chuàng)建設(shè)計(jì)的任務(wù)往往留給開(kāi)發(fā)人員,開(kāi)發(fā)人員需要花時(shí)間適應(yīng)各種不同且相互沖突的工具。
為了測(cè)試DukeScript是否可以更好地解決所有問(wèn)題,我給自己設(shè)定了一個(gè)小挑戰(zhàn)。 我首先為待辦事項(xiàng)應(yīng)用程序找到并購(gòu)買(mǎi)了完整的設(shè)計(jì) ,如下面的屏幕快照所示。
然后,我到處找人將PSD文件更改為HTML(當(dāng)您搜索“ PSD到HTML”時(shí),可以在線找到數(shù)百個(gè)服務(wù)用于此任務(wù))。 我最終選擇了Rapidxhtml,因?yàn)樵摲?wù)價(jià)格便宜,盡管如此,但受到了好評(píng)。
對(duì)于“真實(shí)”項(xiàng)目,并確保更好的溝通,我絕對(duì)希望與設(shè)計(jì)師直接互動(dòng)。 但是,對(duì)于本實(shí)驗(yàn)而言,將通信限于信用卡和Web表單是一個(gè)優(yōu)點(diǎn),因?yàn)橥ㄟ^(guò)這種方式,我們可以確保設(shè)計(jì)人員不了解DukeScript可能需要的任何特殊要求。
我將PSD文件上傳到網(wǎng)站并指出我的樣式要求。 例如,由于適用的樣式更加精細(xì),因此復(fù)選框和滾動(dòng)條會(huì)額外花費(fèi)。 因此,我決定不進(jìn)行這些更詳細(xì)的選擇。 最后,完整的訂單并不昂貴。 每頁(yè)的轉(zhuǎn)換成本,包括所有選定的額外功能(調(diào)整寬度,帶有CSS3HTML5等),約為170歐元。付款是預(yù)先付款的,原則上是在24小時(shí)內(nèi)交付。 聽(tīng)起來(lái)不錯(cuò),我等著懸念。
6小時(shí)后,我已經(jīng)收到一封包含該設(shè)計(jì)鏈接的電子郵件。 不錯(cuò)。 乍一看,結(jié)果看起來(lái)不錯(cuò),盡管寬度調(diào)整不起作用。 兩個(gè)小時(shí)后,在回復(fù)了我的評(píng)論后,我有了一個(gè)能夠正確調(diào)整大小的新版本。 下面的屏幕截圖顯示了結(jié)果。
關(guān)于小缺陷的進(jìn)一步評(píng)論被忽略。 對(duì)于“實(shí)際”項(xiàng)目,優(yōu)質(zhì)服務(wù)或具有適用建議的設(shè)計(jì)機(jī)構(gòu)可能是一個(gè)更好的選擇。 清單6顯示了我收到HTML。
清單6
<!DOCTYPE html> <!--[if lt IE 7]> <html class="no-js lt-ie9 lt-ie8 lt-ie7"> <![endif]--> <!--[if IE 7]> <html class="no-js lt-ie9 lt-ie8"> <![endif]--> <!--[if IE 8]> <html class="no-js lt-ie9"> <![endif]--> <!--[if gt IE 8]><!--> <html class="no-js"> <!--<![endif]--><head><title>TODO</title><meta name="robots" content="index, follow"><meta charset="utf-8"><meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"><meta name="author" content="RapidxHTML" /><link rel="stylesheet" href="css/normalize.css"><link rel="stylesheet" href="css/style.css"></head><body><!--[if lt IE 7]><p class="chromeframe">You are using an outdated browser.<a href="http://browsehappy.com/">Upgrade your browser today</a> or <a href="http://www.google.com/chromeframe/?redirect=true">install Google Chrome Frame</a> to better experience this site.</p><![endif]--><!-- box --><div id="box"><div class="box-cont"><header class="box-header"><div class="box-title">My tasks for today</div><div class="box-links"><a href=""><img src="images/btn-cal.png" alt="" /></a><a href=""><img src="images/btn-settings.png" alt="" /></a></div></header><section class="todo"><section class="todo-bg"><ul class="todo-list"><li class="done"><input type="checkbox" name="" class="toggle" checked="checked" />Design a to-do list <span class="btns"><a href=""><img src="images/icon-edit.png" /></a><a href=""><img src="images/icon-delete.png" /></a></span></li><li><input type="checkbox" name="" class="toggle" />Design a super task<br />with 2 lines <span class="btns"><a href=""><img src="images/icon-edit.png" /></a><a href=""><img src="images/icon-delete.png" /></a></span></li><li><input type="checkbox" name="" class="toggle" />fix the dog toy <span class="btns"><a href=""><img src="images/icon-edit.png" /></a><a href=""><img src="images/icon-delete.png" /></a></span></li><li><input type="checkbox" name="" class="toggle" />buy coffee <span class="btns"><a href=""><img src="images/icon-edit.png" /></a><a href=""><img src="images/icon-delete.png" /></a></span></li><li><input type="checkbox" name="" class="toggle" />feed the dog <span class="btns"><a href=""><img src="images/icon-edit.png" /></a><a href=""><img src="images/icon-delete.png" /></a></span></li><li><input type="checkbox" name="" class="toggle" />take a walk with the dog <img src="icon_smile.gif" alt=":)" class="wp-smiley"> <span class="btns"><a href=""><img src="images/icon-edit.png" /></a><a href=""><img src="images/icon-delete.png" /></a></span></li></ul></section></section></div></div><!-- / box --><script type="text/javascript" src="js/jquery.js"></script><script type="text/javascript" src="js/modernizr-2.6.2.min.js"></script></body> </html>接下來(lái),我不得不為UI添加生命。 從原型開(kāi)始,您已經(jīng)在下面看到的綁定已經(jīng)很熟悉了。 清單7顯示了重寫(xiě)的代碼。
清單7
<!DOCTYPE html> <head><title>TODO</title><meta charset="utf-8"><link rel="stylesheet" href="css/normalize.css"><link rel="stylesheet" href="css/style.css"> </head> <body id="body"><!-- box --><div id="box"><div class="box-cont"><header class="box-header"><div class="box-title">My tasks for today</div><div class="box-links"><a href=""><img src="images/btn-cal.png" alt="" /></a><a href=""><img src="images/btn-settings.png" alt="" /></a></div></header><section class="todo"><section class="todo-bg"><ul class="todo-list" ><!-- ko foreach: tasks --> <li><!-- ko ifnot: $root.editing()===$data --><input type="checkbox" name="" class="toggle" data-bind="checked: complete"/><span data-bind="text: title"></span><span class="btns"><img src="images/icon-edit.png" alt="" data-bind="click: $root.editTask" /><img src="images/icon-delete.png" alt="" data-bind="click: $root.deleteTask" /></span><!-- /ko --><!-- ko if: $root.editing()===$data --><form data-bind="submit: $root.stopEditing"><input type="text" data-bind="textInput: title"/></form><!-- /ko --></li><!-- /ko --><li><form data-bind="submit: addTask"><input type="text" data-bind="textInput: input"/></form></li></ul></section></section></div></div><!-- / box --> </body> </html>在下面的屏幕截圖中,您可以看到結(jié)果。
UI看起來(lái)像設(shè)計(jì)示例,并且在所有平臺(tái)上均起作用。 這樣,實(shí)驗(yàn)取得了圓滿(mǎn)成功-可以完全委托設(shè)計(jì)。 在沒(méi)有特殊要求的情況下,設(shè)計(jì)人員和轉(zhuǎn)換服務(wù)能夠提供可用的資產(chǎn),這些資產(chǎn)可以以最小的更改集成到應(yīng)用程序中。
通過(guò)這種方式,開(kāi)發(fā)人員能夠完全專(zhuān)注于所需功能的實(shí)現(xiàn),并專(zhuān)注于應(yīng)用程序的業(yè)務(wù)邏輯。 那些具有桌面應(yīng)用程序開(kāi)發(fā)經(jīng)驗(yàn)的人都知道,定期轉(zhuǎn)換應(yīng)用程序設(shè)計(jì)會(huì)浪費(fèi)大量開(kāi)發(fā)時(shí)間。 使用DukeScript,您可以放心地將這些問(wèn)題委托給設(shè)計(jì)專(zhuān)業(yè)人員,這可以節(jié)省寶貴的時(shí)間來(lái)實(shí)現(xiàn)應(yīng)用程序的功能要求。
如何開(kāi)發(fā)DukeScript應(yīng)用程序
當(dāng)前,DukeScript支持各種桌面平臺(tái)以及iOS,Android和瀏覽器。 可用的Maven原型為每個(gè)受支持的平臺(tái)創(chuàng)建一個(gè)單獨(dú)的子項(xiàng)目。 對(duì)于每個(gè)平臺(tái),可以使用特定任務(wù)來(lái)測(cè)試和打包子項(xiàng)目。 例如,Android和iOS的子項(xiàng)目提供了在模擬器或連接的設(shè)備上運(yùn)行它們的可能性,而瀏覽器的子項(xiàng)目會(huì)自動(dòng)構(gòu)建靜態(tài)網(wǎng)站。
最新的增強(qiáng)功能已啟用對(duì)嵌入式平臺(tái)的支持,以使DukeScript可以在IoT應(yīng)用程序中使用。 因此,由于Oracle 結(jié)束了對(duì)嵌入式平臺(tái)上JavaFX的支持 ,因此現(xiàn)在再次有可能在嵌入式設(shè)備上使用Java開(kāi)發(fā)專(zhuān)業(yè)的GUI。 在這些情況下,OpenJDK通常足以用作JVM,因此,即使對(duì)于商業(yè)項(xiàng)目,也不需要昂貴的Java SE嵌入式許可。
借助Maven原型,可以通過(guò)各種Java IDE完成DukeScript應(yīng)用程序的開(kāi)發(fā)。 專(zhuān)門(mén)針對(duì)NetBeans IDE,還有一個(gè)具有一系列支持功能的插件, 使開(kāi)發(fā)更加舒適 。 例如,HTML編輯器中有data-bind指令的代碼完成,而DOM Inspector使您可以檢查正在運(yùn)行的應(yīng)用程序。 對(duì)HTML和CSS的更改將由正在運(yùn)行的應(yīng)用程序自動(dòng)獲取。 從0.8版開(kāi)始,甚至Maven原型都引入了熱插拔。 代碼更改將自動(dòng)部署到正在運(yùn)行的應(yīng)用程序中,并且可以立即進(jìn)行測(cè)試,如下面的屏幕快照所示。 甚至JavaScript開(kāi)發(fā)人員也應(yīng)該嫉妒,因?yàn)榕cJavaScript開(kāi)發(fā)不同,應(yīng)用程序的狀態(tài)得以保持,而無(wú)需進(jìn)行任何手動(dòng)重新加載。
用于Java的Controls.js-不帶HTML的DukeScript
DukeScript致力于在沒(méi)有JavaScript的情況下實(shí)現(xiàn)跨平臺(tái)開(kāi)發(fā)。 通常,DukeScript應(yīng)用程序的前端是在HTML和CSS的幫助下編寫(xiě)的。 如已顯示的,可以委派應(yīng)用程序開(kāi)發(fā)的這一方面,同時(shí)它繼續(xù)要求編寫(xiě)測(cè)試并針對(duì)各種平臺(tái)進(jìn)行修改,以及手動(dòng)編輯HTML文件。
作為一種替代方法, Controls.js for Java項(xiàng)目使您可以通過(guò)拖放來(lái)開(kāi)發(fā)完整的UI。 采用這種方法時(shí),您將擁有Maven原型和一個(gè)NetBeans插件來(lái)為您提供支持。 Controls.js利用其自己的組件庫(kù)。 每個(gè)單獨(dú)的控件都可以通過(guò)外觀顯示,而外觀編輯器用于創(chuàng)建自定義外觀。 ViewModel保持不變,而綁定是在可視編輯器的幫助下完成的。
結(jié)論
最后,讓我們研究一下百萬(wàn)美元的問(wèn)題:“ DukeScript是否適合我的項(xiàng)目?” 優(yōu)點(diǎn)很明顯。
- 使用通用的代碼庫(kù),可以為許多平臺(tái)開(kāi)發(fā)應(yīng)用程序。
- 工作流程結(jié)構(gòu)合理,同時(shí)可用豐富且完善的工具來(lái)為您提供支持。
- 可以委派設(shè)計(jì)任務(wù),從而極大地減少了應(yīng)用程序維護(hù)和開(kāi)發(fā)的成本。
盡管如此,DukeScript并不是每個(gè)應(yīng)用程序的最佳解決方案。 對(duì)于那些對(duì)自己的品牌重視并具有統(tǒng)一的跨平臺(tái)設(shè)計(jì)的人,DukeScript可以為他們提供更好的服務(wù),而對(duì)于那些對(duì)其應(yīng)用程序要部署到的設(shè)備本身的外觀和感覺(jué)感興趣的人,DukeScript可以提供更好的服務(wù)。 我也不會(huì)嘗試通過(guò)DukeScript創(chuàng)建3D建模工具,也不會(huì)嘗試通過(guò)優(yōu)化的渲染管道來(lái)使用任何類(lèi)型的應(yīng)用程序。
但是,DukeScript是業(yè)務(wù)應(yīng)用程序的很好選擇。 對(duì)于簡(jiǎn)單的業(yè)務(wù)應(yīng)用程序,使用Controls.js for Java會(huì)有所回報(bào),從而可以實(shí)現(xiàn)快速的應(yīng)用程序開(kāi)發(fā)工作流程。 另外,由于其簡(jiǎn)單的通信機(jī)制,對(duì)于具有服務(wù)器后端的應(yīng)用程序,DukeScript非常適合。 總體而言,DukeScript為Java開(kāi)發(fā)人員提供了一個(gè)跨平臺(tái)開(kāi)發(fā)的順利切入點(diǎn),而無(wú)需放棄世界上使用最廣泛的編程語(yǔ)言,該語(yǔ)言以靜態(tài)方式提供了最佳的IDE支持,無(wú)疑比當(dāng)今世界上任何其他編程語(yǔ)言都要好。
- 本文經(jīng)Geertjan Wielenga的翻譯,由Anton Epple 用德語(yǔ)翻譯 。
翻譯自: https://www.javacodegeeks.com/2015/08/java-everywhere-write-once-run-anywhere-with-dukescript.html
總結(jié)
以上是生活随笔為你收集整理的Java无处不在:使用DukeScript在任何地方运行一次编写的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: (linux usb驱动)
- 下一篇: 十大有用但又偏执的Java编程技术