摘要:使用過(guò)語(yǔ)言的朋友們可能使用過(guò),它是一個(gè)偽造數(shù)據(jù)的工具。但是沒(méi)有語(yǔ)言版本的,所以就動(dòng)手折騰吧。它的重點(diǎn)在于所以我們需要來(lái)看它的所在位置文件。的基本目錄數(shù)據(jù)源具體功能實(shí)現(xiàn)加載數(shù)據(jù)根據(jù)版本的我們也來(lái)創(chuàng)建對(duì)應(yīng)的目錄。本人就是做了一些翻譯的的工作。
使用過(guò)Python語(yǔ)言的朋友們可能使用過(guò)forgery_py,它是一個(gè)偽造數(shù)據(jù)的工具。能偽造一些常用的數(shù)據(jù)。在我們開(kāi)發(fā)過(guò)程和效果展示是十分有用。但是沒(méi)有Go語(yǔ)言版本的,所以就動(dòng)手折騰吧。
從源碼入手在forgery_py的PyPi有一段的實(shí)例代碼:
>>> import forgery_py >>> forgery_py.address.street_address() u"4358 Shopko Junction" >>> forgery_py.basic.hex_color() "3F0A59" >>> forgery_py.currency.description() u"Slovenia Tolars" >>> forgery_py.date.date() datetime.date(2012, 7, 27) >>> forgery_py.internet.email_address() u"brian@zazio.mil" >>> forgery_py.lorem_ipsum.title() u"Pretium nam rhoncus ultrices!" >>> forgery_py.name.full_name() u"Mary Peters" >>> forgery_py.personal.language() u"Hungarian"
從以上的方法調(diào)用我們可以看出forgery_py下有一系列的*.py文件,里面有各種方法,實(shí)現(xiàn)各種功能,我們?cè)趤?lái)通過(guò)分析下Python版本的forgery_py的源碼來(lái)看看它的實(shí)現(xiàn)原理。
# ForgeryPy 包的一級(jí)目錄 ├── dictionaries # 偽造內(nèi)容和來(lái)源目錄,目錄下存放的都是一些文本文件 ├── dictionaries_loader.py # 加載文件腳本 ├── forgery # 主目錄,實(shí)現(xiàn)各種數(shù)據(jù)偽造功能,目錄下存放的都是python文件 ├── __init__.py
我們?cè)趤?lái)看下forgery目錄下的腳本
$ cat name.py import random from ..dictionaries_loader import get_dictionary __all__ = [ "first_name", "last_name", "full_name", "male_first_name", "female_first_name", "company_name", "job_title", "job_title_suffix", "title", "suffix", "location", "industry" ] def first_name(): """Random male of female first name.""" _dict = get_dictionary("male_first_names") _dict += get_dictionary("female_first_names") return random.choice(_dict).strip()
__all__設(shè)置能被調(diào)用的方法。
first_name()方法是forgery_py中一個(gè)典型偽造數(shù)據(jù)方法,我們只要來(lái)分析它就可以知道forgery_py的工作原理了。
這個(gè)方法代碼很少,能容易就看出_dict = get_dictionary("male_first_names")和_dict += get_dictionary("female_first_names")獲取的數(shù)據(jù)合并,在最后的return random.choice(_dict).strip()返回隨機(jī)的數(shù)據(jù)。它的重點(diǎn)在于get_dictionary(),所以我們需要來(lái)看它的所在位置dictionaries_loader.py文件。
$ cat dictionaries_loader import random DICTIONARIES_PATH = abspath(join(dirname(__file__), "dictionaries")) dictionaries_cache = {} def get_dictionary(dict_name): """ Load a dictionary file ``dict_name`` (if it"s not cached) and return its contents as an array of strings. """ global dictionaries_cache if dict_name not in dictionaries_cache: try: dictionary_file = codecs.open( join(DICTIONARIES_PATH, dict_name), "r", "utf-8" ) except IOError: None else: dictionaries_cache[dict_name] = dictionary_file.readlines() dictionary_file.close() return dictionaries_cache[dict_name]
以上就是dictionaries_loader.py文件去掉注釋后的所以要內(nèi)容。它的主要實(shí)現(xiàn)就是:定義一個(gè)全局的字典參數(shù)dictionaries_cache作為緩存,然后定義方法get_dictionary()獲取源數(shù)據(jù),get_dictionary()中每次forgery目錄底下方法調(diào)用時(shí)先查看緩存,緩存字典中存在數(shù)據(jù)就直接輸出,不存在就讀取dictionaries底下的對(duì)應(yīng)文件,并存入緩存。最后是返回?cái)?shù)據(jù)。
總的來(lái)說(shuō)forgery_py的原理就是:一個(gè)方法調(diào)用,去讀內(nèi)存中的緩存,存在就直接返回,不存在就到對(duì)應(yīng)的文本文件中讀取并寫(xiě)入緩存并返回。返回來(lái)的數(shù)據(jù)再隨機(jī)選取輸出結(jié)果。
在了解了forgery_py的工作原理之后,我們就可以來(lái)使用Go語(yǔ)言來(lái)實(shí)現(xiàn)了。
# forgery的基本目錄 $ cat forgery ├── dictionaries # 數(shù)據(jù)源 │?? ├── male_first_names ├── name.go # 具體功能實(shí)現(xiàn) └── loader.go # 加載數(shù)據(jù)
根據(jù)python版本的我們也來(lái)創(chuàng)建對(duì)應(yīng)的目錄。
實(shí)現(xiàn)數(shù)據(jù)的讀取的緩存:
// forgery/loader.go package forgery import ( "os" "io" "bufio" "math/rand" "time" "strings" ) // 全局的緩存map var dictionaries map[string][]string = make(map[string][]string) // 在獲取數(shù)據(jù)之后隨機(jī)輸出 func random(slice []string) string { rand.Seed(time.Now().UnixNano()) n := rand.Intn(len(slice)) return strings.TrimSpace(slice[n]) } // 主要的數(shù)據(jù)加載方法 func loader(name string) (slice []string, err error) { slice, ok := dictionaries[name] // 緩存中存在數(shù)據(jù),直接返回 if ok { return slice, nil } // 讀取對(duì)應(yīng)文件 file, err := os.Open("./dictionaries/" + name) if err != nil { return slice, err } defer file.Close() rd := bufio.NewReader(file) for { line, err := rd.ReadString(" ") slice = append(slice, line) if err != nil || io.EOF == err { break } } dictionaries[name] = slice return slice, nil } // 統(tǒng)一的錯(cuò)誤處理 func checkErr(err error) (string, error) { return "", err }
實(shí)現(xiàn)具體的功能:
// forgery/name.go // Random male of female first name. func FirstName() (string, error) { slice, err := loader("male_first_names") checkErr(err) slice1, err := loader("female_first_names") checkErr(err) slice = append(slice, slice1...) return random(slice), nil }
這樣就將python語(yǔ)言版本的forgery_py使用Go來(lái)實(shí)現(xiàn)了。
最后上面只是提及了一些工作原理,具體的源代碼可以看https://github.com/xingyys/fo...,也十分感謝https://github.com/tomekwojci...,具體的思路和里面的數(shù)據(jù)源都是他提供的。本人就是做了一些翻譯的的工作。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.hztianpu.com/yun/42168.html
摘要:閑言最近需要看簡(jiǎn)寫(xiě)為和相關(guān)的技術(shù)資料,順帶學(xué)一下語(yǔ)言。語(yǔ)言始于年月的三個(gè)工程師,年月正式宣布。項(xiàng)目包括語(yǔ)言工具和標(biāo)準(zhǔn)庫(kù),以及一切從簡(jiǎn)的理念。語(yǔ)言本身成熟且穩(wěn)定,并且保證向下兼容。 [TOC] 閑言 最近需要看Kubernetes(簡(jiǎn)寫(xiě)為k8s)和docker相關(guān)的技術(shù)資料,順帶學(xué)一下Go語(yǔ)言。 嘗試了通過(guò)minikube部署遇到鏡像下載和網(wǎng)絡(luò)配置等等各種問(wèn)題。雖然k8s很火熱,但是資料...
摘要:在本次受訪者中,也有的開(kāi)發(fā)者表示主要使用框架。這不剛發(fā)布了三個(gè)月,就已進(jìn)入了特性凍結(jié)階段。根據(jù)官方統(tǒng)計(jì),有的開(kāi)發(fā)人員使用進(jìn)行單元測(cè)試,而的人使用。此外,與開(kāi)發(fā)者有所不同,開(kāi)發(fā)者更習(xí)慣使用。對(duì)于語(yǔ)言的使用,表示,多數(shù)人使用單個(gè)全局。 showImg(https://upload-images.jianshu.io/upload_images/13825820-feaee185c3c95b...
閱讀 3396·2021-11-15 11:37
閱讀 2545·2021-09-29 09:48
閱讀 4164·2021-09-22 15:55
閱讀 3081·2021-09-22 10:02
閱讀 2705·2021-08-25 09:40
閱讀 3311·2021-08-03 14:03
閱讀 1771·2019-08-29 13:11
閱讀 1637·2019-08-29 12:49