打造一款代碼命名工具的詳細(xì)教程
你是否還在為代碼命名而糾結(jié)不已?
here are only two hard things in Computer Science: cache invalidation and naming things.-- Phil Karlton
那么如何更好的命名呢? 是否有好的工具可以支持我們命名呢?網(wǎng)上搜索一圈沒(méi)有發(fā)現(xiàn)滿(mǎn)意的,于是自己動(dòng)手豐衣足食,https://jadepeng.gitee.io/code-naming-tool/。
使用方法: 打開(kāi)網(wǎng)頁(yè)后,在中文輸入框中輸入 中文命名,然后回車(chē)即可。也可以直接在英文輸入框輸入英文,搜索候選。
現(xiàn)有的工具
unbug.github.io/codelf/ 提供了一個(gè)選擇,作者先調(diào)用有道、百度等翻譯,然后調(diào)用searchcode搜索代碼,從搜索的代碼中提取變量名。
界面做的很酷,但是推薦出來(lái)的變量名稱(chēng)質(zhì)量參差不齊,失去了參考意義。
新的思路
我們常說(shuō)以史為鑒,換一個(gè)思路,我們可以從優(yōu)秀的開(kāi)源庫(kù)中去吸收他們命名的經(jīng)驗(yàn),看看他們是如何命名的,來(lái)供我們參考。
實(shí)現(xiàn)思路:
1. 從spring、apache等代碼庫(kù),讀取變量、方法、類(lèi)名稱(chēng)
2. 根據(jù)關(guān)鍵詞匹配出候選命名
3. 候選結(jié)果排序
獲取優(yōu)秀命名
要獲取命名,首先想到的是讀取代碼庫(kù),需要先下載代碼,然后解析 ———— 工作量巨大,PASS。
那怎么做呢,換個(gè)角度,可以通過(guò)java的反射來(lái)實(shí)現(xiàn)。
首先添加一個(gè)輔助庫(kù):
<dependency> <groupId>org.reflections</groupId> <artifactId>reflections</artifactId> <version>0.9.12</version> </dependency>
然后初始化Reflections,FilterBuilder可以用來(lái)過(guò)濾類(lèi)庫(kù),我們?cè)O(shè)置"org","javax","com","io", 基本上囊?guī)炝酥饕拈_(kāi)源類(lèi)庫(kù),比如spring,apache等.
List<ClassLoader> classLoadersList = new LinkedList<ClassLoader>(); classLoadersList.add(ClasspathHelper.contextClassLoader()); classLoadersList.add(ClasspathHelper.staticClassLoader()); Reflections reflections = new Reflections(new ConfigurationBuilder() .setScanners(new SubTypesScanner(false), new ResourcesScanner()) .setUrls(ClasspathHelper.forClassLoader(classLoadersList.toArray(new ClassLoader[0]))) .filterInputsBy(new FilterBuilder().includePackage("org","javax","com","io")));
然后,可以通過(guò)reflections.getSubTypesOf(Object.class);
來(lái)獲取相關(guān)的class了,注意,我們初始化一個(gè) Map<String, Integer> name2count = new HashMap<String, Integer>();
用來(lái)存儲(chǔ)代碼命名以及對(duì)應(yīng)的出現(xiàn)次數(shù)。
Set<Class<? extends Object>> allClasses = reflections.getSubTypesOf(Object.class); Map<String, Integer> name2count = new HashMap<String, Integer>(); for (Class<?> clazz : allClasses) { System.out.println(clazz.getName()); try { appendToNameMap(name2count, clazz.getSimpleName()); Field[] fields = clazz.getDeclaredFields(); for (Field field : fields) { String name = field.getName(); appendToNameMap(name2count, name); } Method[] methods = clazz.getMethods(); for (Method method : methods) { String name = method.getName(); appendToNameMap(name2count, name); // parameters Parameter[] parameters = method.getParameters(); for (Parameter param : parameters) { name = param.getName(); appendToNameMap(name2count, name); } } }catch(Throwable t) { }
其中appendToNameMap
:
private static void appendToNameMap(Map<String, Integer> name2count, String name) { // filter if(name.contains("-") || name.contains("_")|| name.contains("$")){ return; } if (!name2count.containsKey(name)) { name2count.put(name, 1); } else { name2count.put(name, name2count.get(name) +1); } }
最后把結(jié)果存儲(chǔ)到文件,作為我們的資源。
FileUtils.writeAllText(JSON.toJSONString(name2count), new File("name2count.txt"));
可以到https://gitee.com/jadepeng/code-naming-tool/blob/master/vars.js
查看結(jié)果。
命名推薦
命名推薦,還是遵循,先翻譯,然后根據(jù)翻譯結(jié)果搜索并召回。
其中翻譯直接調(diào)用網(wǎng)易有道的,但是搜索如何搞定呢?
最簡(jiǎn)單的方法,肯定是分詞,然后建立索引,lucene是標(biāo)配。但是上lucene就要上服務(wù)器,PASS!
我們來(lái)找一個(gè)瀏覽器端的lucene,google 后選定flexsearch
.
flexsearch github上有6.5k star,因此優(yōu)先選擇。
下面來(lái)看具體的實(shí)現(xiàn)。
建立索引
初始化FlexSearch,然后將之前獲取的代碼命名建立索引。
var index = new FlexSearch({ encode: "advanced", tokenize: "reverse", suggest: true, cache: true }) var data = [] var i = 0 for (var name in names) { var indexName = name.replace(/([A-Z])/g, " $1") data[i] = { "name": name, "count": names[name] } index.add(i++, indexName) }
這里有個(gè)小技巧,name.replace(/([A-Z])/g, " $1")
可以將駝峰命名拆分成單詞。
同時(shí)data數(shù)組會(huì)保存所有的命名和響應(yīng)的出現(xiàn)次數(shù)。
搜索候選
先翻譯,然后將翻譯結(jié)果給FlexSearch搜索。
function searchFromIndex(value) { var results = index.search(value, 25) results = results.map(function (i) { return data[i] }) results = results.sort(function (a, b) { return b.count - a.count }) return results }
先搜索,出來(lái)的結(jié)果是data中的index序號(hào),轉(zhuǎn)換為list對(duì)象,然后按照count倒排。
tips: 理論上,翻譯的結(jié)果可以去除一些停用詞,搜索效果應(yīng)該更好,這里先放著。
顯示結(jié)果
對(duì)結(jié)果進(jìn)行格式化:
function formatSuggestion(item){ return `${item.name} <span class='tips'>代碼庫(kù)共出現(xiàn)${item.count}次 (相關(guān)搜索: <a target='_blank' >codelf</a> <a target='_blank' >searchcode</a>)</span>`; }
增加到codelf 和 searchcode的鏈接,顯示結(jié)果如下:
開(kāi)源地址
- github: https://github.com/jadepeng/code-naming-tool
- gitee: https://gitee.com/jadepeng/code-naming-tool
命名工具地址: https://jadepeng.gitee.io/code-naming-tool/
總結(jié)
到此這篇關(guān)于來(lái),我們一起打造一款代碼命名工具的文章就介紹到這了,更多相關(guān)代碼命名工具內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java字符串寫(xiě)入文件三種方式的實(shí)現(xiàn)
這篇文章主要介紹了 Java字符串寫(xiě)入文件三種方式的實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下2017-06-06Java編程實(shí)現(xiàn)基于TCP協(xié)議的Socket聊天室示例
這篇文章主要介紹了Java編程實(shí)現(xiàn)基于TCP協(xié)議的Socket聊天室,結(jié)合實(shí)例形式詳細(xì)分析了java基于TCP協(xié)議的Socket聊天室客戶(hù)端與服務(wù)器端相關(guān)實(shí)現(xiàn)與使用技巧,需要的朋友可以參考下2018-01-01spring @Lazy延遲注入的邏輯實(shí)現(xiàn)
有時(shí)候我們會(huì)在屬性注入的時(shí)候添加@Lazy注解實(shí)現(xiàn)延遲注入,今天咱們通過(guò)閱讀源碼來(lái)分析下原因,感興趣的可以了解一下2021-08-08Java連接操作Oracle數(shù)據(jù)庫(kù)代碼詳解
這篇文章主要介紹了Java連接操作Oracle數(shù)據(jù)庫(kù)代碼詳解的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06引入QQ郵箱發(fā)送驗(yàn)證碼進(jìn)行安全校驗(yàn)功能實(shí)現(xiàn)
最近遇到這樣的需求用戶(hù)輸入自己的郵箱,點(diǎn)擊獲取驗(yàn)證碼,后臺(tái)會(huì)發(fā)送一封郵件到對(duì)應(yīng)郵箱中,怎么實(shí)現(xiàn)呢?下面小編給大家?guī)?lái)了引入QQ郵箱發(fā)送驗(yàn)證碼進(jìn)行安全校驗(yàn)功能,需要的朋友可以參考下2023-02-02Springboot啟動(dòng)報(bào)錯(cuò)時(shí)實(shí)現(xiàn)異常定位
這篇文章主要介紹了Springboot啟動(dòng)報(bào)錯(cuò)時(shí)實(shí)現(xiàn)異常定位,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06Java基礎(chǔ)知識(shí)之Java語(yǔ)言概述
這篇文章主要介紹了Java基礎(chǔ)知識(shí)之Java語(yǔ)言概述,本文介紹了Java語(yǔ)言相關(guān)的基礎(chǔ)知識(shí)、歷史介紹、主要應(yīng)用方向等內(nèi)容,需要的朋友可以參考下2015-03-03