摘要:就其本身而言,是當前以及最近幾年最受歡迎的免費信息檢索程序庫。這樣完全和數(shù)據(jù)庫進行了隔離。當一個文檔出現(xiàn)在了搜索結(jié)果中,這就意味著該文檔與用戶給定的查詢語句是相匹配的。
公眾號閱讀
https://mp.weixin.qq.com/s/M3...
[TOC]
什么是Lucene ???The Apache LuceneTM project develops open-source search software, including:
Lucene Core, our flagship sub-project, provides Java-based indexing and search technology, as well as spellchecking, hit highlighting and advanced analysis/tokenization capabilities.
lucene官網(wǎng)(http://lucene.apache.org/)
Lucene是apache軟件基金會4 jakarta項目組的一個子項目,是一個開放源代碼的全文檢索引擎工具包,但它不是一個完整的全文檢索引擎,而是一個全文檢索引擎的架構(gòu),提供了完整的查詢引擎和索引引擎,部分文本分析引擎(英文與德文兩種西方語言)。Lucene的目的是為軟件開發(fā)人員提供一個簡單易用的工具包,以方便的在目標系統(tǒng)中實現(xiàn)全文檢索的功能,或者是以此為基礎建立起完整的全文檢索引擎。Lucene是一套用于全文檢索和搜尋的開源程式庫,由Apache軟件基金會支持和提供。Lucene提供了一個簡單卻強大的應用程式接口,能夠做全文索引和搜尋。在Java開發(fā)環(huán)境里Lucene是一個成熟的免費開源工具。就其本身而言,Lucene是當前以及最近幾年最受歡迎的免費Java信息檢索程序庫。人們經(jīng)常提到信息檢索程序庫,雖然與搜索引擎有關(guān),但不應該將信息檢索程序庫與搜索引擎相混淆。為什么使用Lucene
現(xiàn)在Lucene在互聯(lián)網(wǎng)行業(yè)的用的非常廣泛,尤其是大數(shù)據(jù)時代的今天,那么根據(jù)自己的理解給大家簡單的介紹一下為什么要學習Lucene。
傳統(tǒng)的sql查詢方式,數(shù)據(jù)量過多時,數(shù)據(jù)庫的壓力就會變得很大,查詢速度會變得非常慢。我們需要使用更好的解決方案來分擔數(shù)據(jù)庫的壓力。為了解決數(shù)據(jù)庫壓力和速度的問題,我們的數(shù)據(jù)庫就變成了索引庫,我們使用Lucene的API的來操作服務器上的索引庫。這樣完全和數(shù)據(jù)庫進行了隔離。
現(xiàn)在我們已經(jīng)了解了Lucene。
Lucene是一套用于全文檢索和搜尋的開源程序庫,由Apache軟件基金會支持和提供
Lucene提供了一個簡單卻強大的應用程序接口(API),能夠做全文索引和搜尋,在Java開發(fā)環(huán)境里Lucene是一個成熟的免費開放源代碼工具
Lucene并不是現(xiàn)成的搜索引擎產(chǎn)品,但可以用來制作搜索引擎產(chǎn)品
總結(jié):Lucene全文檢索就是對文檔中全部內(nèi)容進行分詞,然后對所有單詞建立倒排索引的過程。
目前最新的版本是7.x系列,但是在企業(yè)中還是用4.x比較多,所以我們學習4.x的版本。
檢索數(shù)據(jù)需要我們先分詞,存入索引庫。
文檔Document:數(shù)據(jù)庫中一條具體的記錄
字段Field:數(shù)據(jù)庫中的每個字段
目錄對象Directory:物理存儲位置
寫出器的配置對象:需要分詞器和lucene的版本
UTF-8 1.7 1.7 4.10.2 com.janeluo ikanalyzer 2012_u6 org.apache.lucene lucene-core ${lunece.version} org.apache.lucene lucene-queryparser ${lunece.version} org.apache.lucene lucene-analyzers-common ${lunece.version} org.apache.lucene lucene-highlighter ${lunece.version} junit junit 4.11 test
創(chuàng)建文檔對象
創(chuàng)建存儲目錄
創(chuàng)建分詞器
創(chuàng)建索引寫入器的配置對象
創(chuàng)建索引寫入器對象
將文檔交給索引寫入器
提交
關(guān)閉
// 創(chuàng)建索引 @Test public void testCreate() throws Exception { // 1 創(chuàng)建文檔對象 Document document = new Document(); // 創(chuàng)建并添加字段信息。參數(shù):字段的名稱、字段的值、是否存儲,這里選Store.YES代表存儲到文檔列表。Store.NO代表不存儲 document.add(new StringField("id", "1", Store.YES)); // 這里我們title字段需要用TextField,即創(chuàng)建索引又會被分詞。StringField會創(chuàng)建索引,但是不會被分詞 document.add(new TextField("title", "谷歌地圖之父跳槽facebook", Store.YES)); // 2 索引目錄類,指定索引在硬盤中的位置 Directory directory = FSDirectory.open(new File("E:luceneTest")); // 3 創(chuàng)建分詞器對象 // Analyzer analyzer = new StandardAnalyzer(); Analyzer analyzer = new IKAnalyzer(); // 4 索引寫出工具的配置對象 IndexWriterConfig conf = new IndexWriterConfig(Version.LATEST, analyzer); // 是否清空索引庫;設置打開方式:OpenMode.APPEND // 會在索引庫的基礎上追加新索引。OpenMode.CREATE會先清空原來數(shù)據(jù),再提交新的索引 conf.setOpenMode(OpenMode.CREATE); // 5 創(chuàng)建索引的寫出工具類。參數(shù):索引的目錄和配置信息 IndexWriter indexWriter = new IndexWriter(directory, conf); // 6 把文檔交給IndexWriter indexWriter.addDocument(document); // 7 提交 indexWriter.commit(); // 8 關(guān)閉 indexWriter.close(); }
索引查看工具
啟動run.bat
*** @Test public void testSearch() throws Exception { // 索引目錄對象 Directory directory = FSDirectory.open(new File("E:luceneTest")); // 索引讀取工具 IndexReader reader = DirectoryReader.open(directory); // 索引搜索工具 IndexSearcher searcher = new IndexSearcher(reader); // 創(chuàng)建查詢解析器,兩個參數(shù):默認要查詢的字段的名稱,分詞器 QueryParser parser = new QueryParser("title", new IKAnalyzer()); // 創(chuàng)建查詢解析器,倆個參數(shù):默認要查詢的字段名稱,分詞器 // MultiFieldQueryParser parser2 = new MultiFieldQueryParser(new // String[] { // "id", "title" }, new IKAnalyzer()); // Query query2 = parser2.parse("1"); // 創(chuàng)建查詢對象 Query query = parser.parse("谷歌之父"); // 搜索數(shù)據(jù),兩個參數(shù):查詢條件對象要查詢的最大結(jié)果條數(shù) // 返回的結(jié)果是 按照匹配度排名得分前N名的文檔信息(包含查詢到的總條數(shù)信息、所有符合條件的文檔的編號信息)。 TopDocs topDocs = searcher.search(query, 10); // 獲取總條數(shù) System.out.println("本次搜索共找到" + topDocs.totalHits + "條數(shù)據(jù)"); // 獲取得分文檔對象(ScoreDoc)數(shù)組.SocreDoc中包含:文檔的編號、文檔的得分 ScoreDoc[] scoreDocs = topDocs.scoreDocs; for (ScoreDoc scoreDoc : scoreDocs) { // 取出文檔編號 int docID = scoreDoc.doc; // 根據(jù)編號去找文檔 Document doc = reader.document(docID); System.out.println("id: " + doc.get("id")); System.out.println("title: " + doc.get("title")); // 取出文檔得分 System.out.println("得分: " + scoreDoc.score); } }二、工具類 查詢
注:代碼中加了必要注釋
public void search(Query query) throws Exception { // 索引目錄對象 Directory directory = FSDirectory.open(new File("E:luceneTest")); // 索引讀取工具 IndexReader reader = DirectoryReader.open(directory); // 索引搜索工具 IndexSearcher searcher = new IndexSearcher(reader); // 搜索數(shù)據(jù),兩個參數(shù):查詢條件對象要查詢的最大結(jié)果條數(shù) // 返回的結(jié)果是 按照匹配度排名得分前N名的文檔信息(包含查詢到的總條數(shù)信息、所有符合條件的文檔的編號信息)。 TopDocs topDocs = searcher.search(query, 10); // 獲取總條數(shù) System.out.println("本次搜索共找到" + topDocs.totalHits + "條數(shù)據(jù)"); // 獲取得分文檔對象(ScoreDoc)數(shù)組.SocreDoc中包含:文檔的編號、文檔的得分 ScoreDoc[] scoreDocs = topDocs.scoreDocs; for (ScoreDoc scoreDoc : scoreDocs) { // 取出文檔編號 int docID = scoreDoc.doc; // 根據(jù)編號去找文檔 Document doc = reader.document(docID); System.out.println("id: " + doc.get("id")); System.out.println("title: " + doc.get("title")); // 取出文檔得分 System.out.println("得分: " + scoreDoc.score); } }
注:普通詞條查詢
/* * 測試普通詞條查詢 注意:Term(詞條)是搜索的最小單位,不可再分詞。值必須是字符串! */ @Test public void testTermQuery() throws Exception { // 創(chuàng)建詞條查詢對象 Query query = new TermQuery(new Term("title", "谷歌地圖")); search(query); }
注:通配符查詢
/* * 測試通配符查詢 ? 可以代表任意一個字符 * 可以任意多個任意字符 */ @Test public void testWildCardQuery() throws Exception { // 創(chuàng)建查詢對象 Query query = new WildcardQuery(new Term("title", "*歌*")); search(query); }
注:模糊查詢;數(shù)組范圍查詢;布爾查詢
/* * 測試模糊查詢 */ @Test public void testFuzzyQuery() throws Exception { // 創(chuàng)建模糊查詢對象:允許用戶輸錯。但是要求錯誤的最大編輯距離不能超過2 // 編輯距離:一個單詞到另一個單詞最少要修改的次數(shù) facebool --> facebook 需要編輯1次,編輯距離就是1 // Query query = new FuzzyQuery(new Term("title","fscevool")); // 可以手動指定編輯距離,但是參數(shù)必須在0~2之間 Query query = new FuzzyQuery(new Term("title", "facevool"), 2); search(query); } /*************************************************************** * 測試:數(shù)值范圍查詢 注意:數(shù)值范圍查詢,可以用來對非String類型的ID進行精確的查找 */ @Test public void testNumericRangeQuery() throws Exception { // 數(shù)值范圍查詢對象,參數(shù):字段名稱,最小值、最大值、是否包含最小值、是否包含最大值 Query query = NumericRangeQuery.newLongRange("id", 2L, 2L, true, true); search(query); } /***************************************************************** * 布爾查詢: 布爾查詢本身沒有查詢條件,可以把其它查詢通過邏輯運算進行組合! 交集:Occur.MUST + Occur.MUST * 并集:Occur.SHOULD + Occur.SHOULD 非:Occur.MUST_NOT */ // @Test // public void testBooleanQuery() throws Exception { // // Query query1 = NumericRangeQuery.newLongRange("id", 1L, 3L, true, true); // Query query2 = NumericRangeQuery.newLongRange("id", 2L, 4L, true, true); // // 創(chuàng)建布爾查詢的對象 // BooleanQuery query = new BooleanQuery(); // // 組合其它查詢 // query.add(query1, BooleanClause.Occur.MUST_NOT); // query.add(query2, BooleanClause.Occur.SHOULD); // // search(query); // }修改-刪除
注:修改和刪除操作
/* * 測試:修改索引 注意: A:Lucene修改功能底層會先刪除,再把新的文檔添加。 * B:修改功能會根據(jù)Term進行匹配,所有匹配到的都會被刪除。這樣不好 C:因此,一般我們修改時,都會根據(jù)一個唯一不重復字段進行匹配修改。例如ID * D:但是詞條搜索,要求ID必須是字符串。如果不是,這個方法就不能用。 * 如果ID是數(shù)值類型,我們不能直接去修改??梢韵仁謩觿h除deleteDocuments(數(shù)值范圍查詢鎖定ID),再添加。 */ @Test public void testUpdate() throws Exception { // 創(chuàng)建目錄對象 Directory directory = FSDirectory.open(new File("E:luceneTest")); // 創(chuàng)建配置對象 IndexWriterConfig conf = new IndexWriterConfig(Version.LATEST, new IKAnalyzer()); // 創(chuàng)建索引寫出工具 IndexWriter writer = new IndexWriter(directory, conf); // 創(chuàng)建新的文檔數(shù)據(jù) Document doc = new Document(); doc.add(new StringField("id", "1", Store.YES)); doc.add(new TextField("title", "谷歌地圖之父跳槽facebook ", Store.YES)); /* * 修改索引。參數(shù): 詞條:根據(jù)這個詞條匹配到的所有文檔都會被修改 文檔信息:要修改的新的文檔數(shù)據(jù) */ writer.updateDocument(new Term("id", "1"), doc); // 提交 writer.commit(); // 關(guān)閉 writer.close(); } /* * 演示:刪除索引 注意: 一般,為了進行精確刪除,我們會根據(jù)唯一字段來刪除。比如ID 如果是用Term刪除,要求ID也必須是字符串類型! */ @Test public void testDelete() throws Exception { // 創(chuàng)建目錄對象 Directory directory = FSDirectory.open(new File("E:luceneTest")); // 創(chuàng)建配置對象 IndexWriterConfig conf = new IndexWriterConfig(Version.LATEST, new IKAnalyzer()); // 創(chuàng)建索引寫出工具 IndexWriter writer = new IndexWriter(directory, conf); // 根據(jù)詞條進行刪除 // writer.deleteDocuments(new Term("id", "1")); // 根據(jù)query對象刪除,如果ID是數(shù)值類型,那么我們可以用數(shù)值范圍查詢鎖定一個具體的ID // Query query = NumericRangeQuery.newLongRange("id", 2L, 2L, true, // true); // writer.deleteDocuments(query); // 刪除所有 writer.deleteAll(); // 提交 writer.commit(); // 關(guān)閉 writer.close(); }高亮顯示
創(chuàng)建目錄 對象
創(chuàng)建索引讀取工具
創(chuàng)建索引搜索工具
創(chuàng)建查詢解析器
創(chuàng)建查詢對象
創(chuàng)建格式化器
創(chuàng)建查詢分數(shù)工具
準備高亮工具
搜索
獲取結(jié)果
用高亮工具處理普通的查詢結(jié)果
關(guān)鍵字增加css樣式
// 高亮顯示 @Test public void testHighlighter() throws Exception { // 目錄對象 Directory directory = FSDirectory.open(new File("indexDir")); // 創(chuàng)建讀取工具 IndexReader reader = DirectoryReader.open(directory); // 創(chuàng)建搜索工具 IndexSearcher searcher = new IndexSearcher(reader); QueryParser parser = new QueryParser("title", new IKAnalyzer()); Query query = parser.parse("谷歌地圖"); // 格式化器 Formatter formatter = new SimpleHTMLFormatter("", ""); QueryScorer scorer = new QueryScorer(query); // 準備高亮工具 Highlighter highlighter = new Highlighter(formatter, scorer); // 搜索 TopDocs topDocs = searcher.search(query, 10); System.out.println("本次搜索共" + topDocs.totalHits + "條數(shù)據(jù)"); ScoreDoc[] scoreDocs = topDocs.scoreDocs; for (ScoreDoc scoreDoc : scoreDocs) { // 獲取文檔編號 int docID = scoreDoc.doc; Document doc = reader.document(docID); System.out.println("id: " + doc.get("id")); String title = doc.get("title"); // 用高亮工具處理普通的查詢結(jié)果,參數(shù):分詞器,要高亮的字段的名稱,高亮字段的原始值 String hTitle = highlighter.getBestFragment(new IKAnalyzer(), "title", title); System.out.println("title: " + hTitle); // 獲取文檔的得分 System.out.println("得分:" + scoreDoc.score); } }排序
// 排序 @Test public void testSortQuery() throws Exception { // 目錄對象 Directory directory = FSDirectory.open(new File("indexDir")); // 創(chuàng)建讀取工具 IndexReader reader = DirectoryReader.open(directory); // 創(chuàng)建搜索工具 IndexSearcher searcher = new IndexSearcher(reader); QueryParser parser = new QueryParser("title", new IKAnalyzer()); Query query = parser.parse("谷歌地圖"); // 創(chuàng)建排序?qū)ο?需要排序字段SortField,參數(shù):字段的名稱、字段的類型、是否反轉(zhuǎn)如果是false,升序。true降序 Sort sort = new Sort(new SortField("id", SortField.Type.LONG, true)); // 搜索 TopDocs topDocs = searcher.search(query, 10,sort); System.out.println("本次搜索共" + topDocs.totalHits + "條數(shù)據(jù)"); ScoreDoc[] scoreDocs = topDocs.scoreDocs; for (ScoreDoc scoreDoc : scoreDocs) { // 獲取文檔編號 int docID = scoreDoc.doc; Document doc = reader.document(docID); System.out.println("id: " + doc.get("id")); System.out.println("title: " + doc.get("title")); } }分頁
// 分頁 @Test public void testPageQuery() throws Exception { // 實際上Lucene本身不支持分頁。因此我們需要自己進行邏輯分頁。我們要準備分頁參數(shù): int pageSize = 2;// 每頁條數(shù) int pageNum = 3;// 當前頁碼 int start = (pageNum - 1) * pageSize;// 當前頁的起始條數(shù) int end = start + pageSize;// 當前頁的結(jié)束條數(shù)(不能包含) // 目錄對象 Directory directory = FSDirectory.open(new File("indexDir")); // 創(chuàng)建讀取工具 IndexReader reader = DirectoryReader.open(directory); // 創(chuàng)建搜索工具 IndexSearcher searcher = new IndexSearcher(reader); QueryParser parser = new QueryParser("title", new IKAnalyzer()); Query query = parser.parse("谷歌地圖"); // 創(chuàng)建排序?qū)ο?需要排序字段SortField,參數(shù):字段的名稱、字段的類型、是否反轉(zhuǎn)如果是false,升序。true降序 Sort sort = new Sort(new SortField("id", Type.LONG, false)); // 搜索數(shù)據(jù),查詢0~end條 TopDocs topDocs = searcher.search(query, end,sort); System.out.println("本次搜索共" + topDocs.totalHits + "條數(shù)據(jù)"); ScoreDoc[] scoreDocs = topDocs.scoreDocs; for (int i = start; i < end; i++) { ScoreDoc scoreDoc = scoreDocs[i]; // 獲取文檔編號 int docID = scoreDoc.doc; Document doc = reader.document(docID); System.out.println("id: " + doc.get("id")); System.out.println("title: " + doc.get("title")); } }三、優(yōu)化 Lucene打分算法
當談論到查詢的相關(guān)性,很重要的一件事就是對于給定的查詢語句,如何計算文檔得分。文檔得分是一個用來描述查詢語句和文檔之間匹配程度的變量。如果你希望通過干預Lucene查詢來改變查詢結(jié)果的排序,你就需要對Lucene的得分計算有所理解。
當一個文檔出現(xiàn)在了搜索結(jié)果中,這就意味著該文檔與用戶給定的查詢語句是相匹配的。Lucene會對匹配成功的文檔給定一個分數(shù)。至少從Lucene這個層面,從打分公式的結(jié)果來看,分數(shù)值越高,代表文檔相關(guān)性越高。 自然而然,我們可以得出:兩個不同的查詢語句對同一個文檔的打分將會有所不同,但是比較這兩個得分是沒有意義的。用戶需要記住的是:我們不僅要避免去比較不同查詢語句對同一個文檔的打分結(jié)果,還要避免比較不同查詢語句對文檔打分結(jié)果的最大值。這是因為文檔的得分是多個因素共同影響的結(jié)果,不僅有權(quán)重(boosts)和查詢語句的結(jié)構(gòu)起作用,還有匹配關(guān)鍵詞的個數(shù),關(guān)鍵詞所在的域,查詢歸一化因子中用到的匹配類型……。在極端情況下,只是因為我們用了自定義打分的查詢對象或者由于倒排索引中詞的動態(tài)變化,相似的查詢表達式對于同一個文檔都會產(chǎn)生截然不同的打分。
計算文檔得分,考慮因素如下:
文檔權(quán)重(Document boost):在索引時給某個文檔設置的權(quán)重值。
域權(quán)重(Field boost):在查詢的時候給某個域設置的權(quán)重值。
調(diào)整因子(Coord):基于文檔中包含查詢關(guān)鍵詞個數(shù)計算出來的調(diào)整因子。一般而言,如果一個文檔中相比其它的文檔出現(xiàn)了更多的查詢關(guān)鍵詞,那么其值越大。
逆文檔頻率(Inerse document frequency):基于Term的一個因子,存在的意義是告訴打分公式一個詞的稀有程度。其值越低,詞越稀有(這里的值是指單純的頻率,即多少個文檔中出現(xiàn)了該詞;而非指Lucene中idf的計算公式)。打分公式利用這個因子提升包含稀有詞文檔的權(quán)重。
長度歸一化(Length norm):基于域的一個歸一化因子。其值由給定域中Term的個數(shù)決定(在索引文檔的時候已經(jīng)計算出來了,并且存儲到了索引中)。域越的文本越長,因子的權(quán)重越低。這表明Lucene打分公式偏向于域包含Term少的文檔。
詞頻(Term frequency):基于Term的一個因子。用來描述給定Term在一個文檔中出現(xiàn)的次數(shù),詞頻越大,文檔的得分越大。
查詢歸一化因子(Query norm):基于查詢語句的歸一化因子。其值為查詢語句中每一個查詢詞權(quán)重的平方和。查詢歸一化因子使得比較不同查詢語句的得分變得可行,當然比較不同查詢語句得分并不總是那么易于實現(xiàn)和可行的。
Lucene打分公式Lucene概念上的打分公式是這樣的:(TF/IDF公式的概念版)
上面的公式展示了布爾信息檢索模型和向量空間信息檢索模型的組合。我們暫時不去討論它,直接見識下Lucene實際應用的打分公式:
可以看到,文檔的分數(shù)實際上是由查詢語句q和文檔d作為變量的一個函數(shù)值。打分公式中有兩部分不直接依賴于查詢詞,它們是coord和queryNorm。公式的值是這樣計算的,coord和queryNorm兩大部分直接乘以查詢語句中每個查詢詞計算值的總和。另一方面,這個總和也是由每個查詢詞的詞頻(tf),逆文檔頻率(idf),查詢詞的權(quán)重,還有norm,也就是前面說的length norm相乘而得的結(jié)果。聽上去有些復雜吧?不用擔心,這些東西不需要全部記住。你只需要知道在進行文檔打分的時候,哪些因素是起決定作用的就可以了。基本上,從前面的公式中可以提煉出以下的幾個規(guī)則:
匹配到的關(guān)鍵詞越稀有,文檔的得分就越高。
文檔的域越小(包含比較少的Term),文檔的得分就越高。
設置的權(quán)重(索引和搜索時設置的都可以)越大,文檔得分越高。
正如我們所看到的那樣,Lucene會給具有這些特征的文檔打最高分:文檔內(nèi)容能夠匹配到較多的稀有的搜索關(guān)鍵詞,文檔的域包含較少的Term,并且域中的Term多是稀有的。
簡而言之
將IKAnalyzer.cfg.xml和stopword.dic和xxx.dic文件復制到MyEclipse的src目錄下,再進行配置
IK Analyzer默認的停用詞詞典為IKAnalyzer2012_u6/stopword.dic,這個停用詞詞典并不完整,只有30多個英文停用詞??梢詳U展停用詞字典,新增ext_stopword.dic,文件和IKAnalyzer.cfg.xml在同一目錄,編輯IKAnalyzer.cfg.xml把新增的停用詞字典寫入配置文件,多個停用詞字典用逗號隔開,如下所示。
stopword.dic;ext_stopword.dic
接下來就可以構(gòu)建自己的搜索引擎了。
上面展示了lucene一些基本操作,更詳細的的工具類可以訪問https://github.com/wangshiyu777/usefulDemo,分享了詳細Demo。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/74975.html
摘要:那我們現(xiàn)在是不知道記錄是否真真正正存儲到索引庫中的,因為我們看不見。索引庫存放的數(shù)據(jù)放在文件下,我們也是不能打開文件的。 什么是Lucene?? Lucene是apache軟件基金會發(fā)布的一個開放源代碼的全文檢索引擎工具包,由資深全文檢索專家Doug Cutting所撰寫,它是一個全文檢索引擎的架構(gòu),提供了完整的創(chuàng)建索引和查詢索引,以及部分文本分析的引擎,Lucene的目的是為軟件開發(fā)...
摘要:說到檔案系統(tǒng),選文檔數(shù)據(jù)庫再合適不過了。熟悉的人看這個會很眼熟,沒錯,這就是從借鑒過來,并用在我的關(guān)系數(shù)據(jù)庫查詢上。接口規(guī)范接口的主要目是為了傳遞數(shù)據(jù),數(shù)據(jù)結(jié)構(gòu)已經(jīng)在上面給出。為便于不同系統(tǒng)不同終端的數(shù)據(jù)交換,也將應當在接口支持之內(nèi)。 說到檔案系統(tǒng),選文檔數(shù)據(jù)庫再合適不過了。談到文檔數(shù)據(jù)庫一般想到的是 MongoDB、CouchDB 之類的,可這里要說的不是這些,而是另一個 NoSQL...
摘要:說到檔案系統(tǒng),選文檔數(shù)據(jù)庫再合適不過了。熟悉的人看這個會很眼熟,沒錯,這就是從借鑒過來,并用在我的關(guān)系數(shù)據(jù)庫查詢上。接口規(guī)范接口的主要目是為了傳遞數(shù)據(jù),數(shù)據(jù)結(jié)構(gòu)已經(jīng)在上面給出。為便于不同系統(tǒng)不同終端的數(shù)據(jù)交換,也將應當在接口支持之內(nèi)。 說到檔案系統(tǒng),選文檔數(shù)據(jù)庫再合適不過了。談到文檔數(shù)據(jù)庫一般想到的是 MongoDB、CouchDB 之類的,可這里要說的不是這些,而是另一個 NoSQL...
閱讀 1032·2022-07-19 10:19
閱讀 1804·2021-09-02 15:15
閱讀 1018·2019-08-30 15:53
閱讀 2661·2019-08-30 13:45
閱讀 2663·2019-08-26 13:57
閱讀 1993·2019-08-26 12:13
閱讀 1015·2019-08-26 10:55
閱讀 555·2019-08-26 10:46