真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

大數(shù)據(jù)中如何分析語言DolphinDB腳本語言

這期內容當中小編將會給大家?guī)碛嘘P大數(shù)據(jù)中如何分析語言DolphinDB腳本語言,文章內容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

創(chuàng)新互聯(lián)是一家專注于網(wǎng)站設計、網(wǎng)站制作與策劃設計,天柱網(wǎng)站建設哪家好?創(chuàng)新互聯(lián)做網(wǎng)站,專注于網(wǎng)站建設10余年,網(wǎng)設計領域的專業(yè)建站公司;建站業(yè)務涵蓋:天柱等地區(qū)。天柱做網(wǎng)站價格咨詢:028-86922220

大數(shù)據(jù)中如何分析語言DolphinDB腳本語言

開發(fā)大數(shù)據(jù)應用,不僅需要能支撐海量數(shù)據(jù)的分布式數(shù)據(jù)庫,能高效利用多核多節(jié)點的分布式計算框架,更需要一門能與分布式數(shù)據(jù)庫和分布式計算有機融合、高性能易擴展、表達能力強、滿足快速開發(fā)和建模需要的編程語言。DolphinDB從流行的Python和SQL語言汲取了靈感,設計了大數(shù)據(jù)處理腳本語言。

提到數(shù)據(jù)庫語言,我們很容易想到標準的SQL語言。不同于標準的SQL,DolphinDB編程語言功能齊全,表達能力非常強大,完美支持命令式編程、向量化編程、函數(shù)話編程、SQL編程、遠程過程調用編程(RPC)和元編程等多種編程范式。DolphinDB編程語言的語法和表達習慣與Python和SQL非常相似,只要對Python和SQL有一定的了解,就能輕松掌握。相對而言,掌握內存時序數(shù)據(jù)庫kdb+的q語言難度要大得多。

DolphinDB的編程語言能夠滿足數(shù)據(jù)科學家快速開發(fā)和建模的需求。DolphinDB語言簡潔靈活,表達能力強,大大提高了數(shù)據(jù)科學家的開發(fā)效率。DolphinDB支持向量化計算和分布式計算,具有極快的運行速度。下面將詳細介紹DolphinDB編程語言的獨特之處。

1.命令式編程

與主流的腳本語言Python、JS等,還有強類型語言C、C++、Java等一樣,DolphinDB也支持命令式編程。命令式編程是指通過執(zhí)行一條一條的語句,實現(xiàn)最終目標。DolphinDB的命令式編程主要是用作上層模塊的處理和調度。在大數(shù)據(jù)分析中,由于需要處理的數(shù)據(jù)量非常龐大,如果我們采用命令式編程逐行處理數(shù)據(jù),效率會十分低下,性能也會有所下降。因此,我們推薦在DolphinDB中使用其他編程方式來批量處理數(shù)據(jù)。

//DolphinDB支持對單變量和多變量進行賦值
x = 1 2 3
y = 4 5
y += 2
x, y = y, x //swap the value of x and y
x, y =1 2 3, 4 5

// 1到100累加求和
s = 0
for(x in 1:101) s += x
print s

//數(shù)組中的元素求和
s = 0;
for(x in 1 3 5 9 15) s += x
print s

//打印矩陣每一列的均值
m = matrix(1 2 3, 4 5 6, 7 8 9)
for(c in m) print c.avg()

//計算product表中每一個產(chǎn)品的銷售額
t= table(["TV set", "Phone", "PC"] as productId, 1200 600 800 as price, 10 20 7 as qty)
for(row in t) print row.productId + ": " + row.price * row.qty

2.向量化編程

跟matlab、R等編程語言一樣,DolphinDB也支持向量化編程。前面提到的kdb+數(shù)據(jù)庫的q語言也是向量處理語言,它在復雜的計算上表現(xiàn)出很好的性能,并且效率很高。DolphinDB的編程語言對很多算法都進行了優(yōu)化,比如對時間序列數(shù)據(jù)計算滑動窗口指標,大大提高了向量函數(shù)的效率。

//兩個長度為1000萬的向量相加,采用向量化編程比命令式編程的for語句更加簡潔,耗耗時更短。
n = 10000000
a = rand(1.0, n)
b = rand(1.0, n)

//采用for語句編程,需要12秒
c = array(DOUBLE, n)
for(i in 0 : n)
    c[i] = a[i] + b[i]
Time elapsed: 12341.043 ms

//采用向量化編程,僅需36毫秒
c = a + b
Time elapsed: 36.901 ms

向量化編程通常是把整個向量加載到連續(xù)內存中。有時候因為內存碎片,沒有找到連續(xù)內存,向量就不可用了。DolphinDB針對這個問題,特意提供了big array數(shù)據(jù)類型。big array可以把物理上不連續(xù)的內存塊組成邏輯上連續(xù)的向量,即使是非常大的向量,也能在DolphinDB中使用,提高了系統(tǒng)的可用性。

3.函數(shù)化編程

DolphinDB支持函數(shù)化編程的大部分功能,包括純函數(shù)、自定義函數(shù)、λ函數(shù)、高階函數(shù)、部分應用和閉包。DolphinDB內置了400多個函數(shù),涵蓋了各種數(shù)據(jù)類型、數(shù)據(jù)結構和系統(tǒng)調用。

DolphinDB的純函數(shù)特性減少了函數(shù)的副作用。在自定義函數(shù)時,DolphinDB不能使用函數(shù)體外定義的變量。純函數(shù)特性可以大幅度提高代碼可讀性和軟件質量。

3.1 自定義函數(shù)

//定義一個函數(shù)返回工作日
def getWorkDays(dates){
    return dates[def(x):weekday(x) between 1:5]
}

getWorkDays(2018.07.01 2018.08.01 2018.09.01 2018.10.01)

[2018.08.01, 2018.10.01]

上面的例子定義一個函數(shù)getWorkDays,該函數(shù)受一組日期,返回并返回在周一和周五之間的日期。函數(shù)的實現(xiàn)采用了向量的過濾功能,也就是接受一個布爾型單目函數(shù)用于數(shù)據(jù)的過濾。

3.2 高階函數(shù)

下面的一個例子我們使用三個高階函數(shù)pivot、each和cross,干凈利落的用三行代碼,根據(jù)股票日內tick級別的報價數(shù)據(jù),計算出兩兩之間的相關性。

//模擬生成10000000萬個數(shù)據(jù)點(股票代碼,交易時間和價格)
n=10000000
syms = rand(`FB`GOOG`MSFT`AMZN`IBM, n)
time = 09:30:00.000 + rand(21600000, n)
price = 500.0 + rand(500.0, n)

//利用pivot函數(shù)生成透視表
priceMatrix = pivot(avg, price, time.minute(), syms)
//each和ratios函數(shù)的配合使用,為每個股票(矩陣的列)生成每分鐘的回報序列
retMatrix = each(ratios, priceMatrix) - 1
//cross和corr函數(shù)的配合使用,計算股票兩兩之間的相關性
corrMatrix = cross(corr, retMatrix, retMatrix)

     AMZN      FB        GOOG      IBM       MSFT
     --------- --------- --------- --------- ---------
AMZN|1         0.015181  -0.056245 0.005822  0.084104
FB  |0.015181  1         -0.028113 0.034159  -0.117279
GOOG|-0.056245 -0.028113 1         -0.039278 -0.025165
IBM |0.005822  0.034159  -0.039278 1         -0.049922
MSFT|0.084104  -0.117279 -0.025165 -0.049922 1

3.3 部分應用

高階函數(shù)中的函數(shù)參數(shù)通常對參數(shù)有限制,通過部分應用,可以確保參數(shù)符合要求。例如,給定一個向量 a = 12 14 18,計算與矩陣中的每一列的相關性。因為要計算矩陣的每一列的相關性,當然可以使用高階函數(shù)each。但是corr函數(shù)需要兩個參數(shù),而矩陣只提供其中的一個參數(shù),另一個參數(shù)必須事先給定,所以部分應用可以解決這個問題。當然我們也可以用for語句來解決這個問題,但代碼冗長而低效。

a = 12 14 18
m = matrix(5 6 7, 1 3 2, 8 7 11)

//使用each和部分應用計算矩陣中的每一列與給定向量a的相關性
each(corr{a}, m)

//使用for語句解決上面的問題
cols = m.columns()
c = array(DOUBLE, cols)
for(i in 0:cols)
    c[i] = corr(a, m[i])

部分應用的另一個作用是使函數(shù)保持狀態(tài)。例如,在流計算中,用戶通常需要給定一個消息處理函數(shù)(message handler),接受一條新的信息,返回一個結果。但是我們希望消息處理函數(shù)返回的是迄今為止所有數(shù)的平均數(shù)。這個問題我們可以通過部分應用來解決。

def cumavg(mutable stat, newNum){
    stat[0] = (stat[0] * stat[1] + newNum)/(stat[1] + 1)
    stat[1] += 1
    return stat[0]
}

msgHandler = cumavg{0.0 0.0}
each(msgHandler, 1 2 3 4 5)

[1,1.5,2,2.5,3]

4.SQL編程

DolphinDB的編程語言不僅支持標準的SQL,還針對時間序列數(shù)據(jù)擴展了SQL的功能,如分組計算(context by)、數(shù)據(jù)透視(pivot by)、窗口函數(shù)、asof連接和窗口連接等,更便于分析時間序列數(shù)據(jù)。單純的SQL引擎表達能力有限,很難滿足更加復雜的數(shù)據(jù)分析和算法實現(xiàn),影響開發(fā)效率。在DolphinDB中,腳本語言與SQL語言是完全融合在一起的。

4.1 SQL與編程語言融合

//生成一個員工工資表
emp_wage = table(take(1..10, 100) as id, take(2017.10M + 1..10, 100).sort() as month, take(5000 5500 6000 6500, 100) as wage)

//計算給定的一組員工的平均工資。員工列表存儲在一個本地變量empIds中
empIds = 3 4 6 7 9
select avg(wage) from emp_wage where id in empIds group by id
id avg_wage
-- --------
3  5500
4  6000
6  6000
7  5500
9  5500

//除計算平均工資外,同時顯示員工的姓名。員工姓名使用一個字典empName來獲取。
empName = dict(1..10, `Alice`Bob`Jerry`Jessica`Mike`Tim`Henry`Anna`Kevin`Jones)
select empName[first(id)] as name, avg(wage) from emp_wage where id in empIds group by id
id name    avg_wage
-- ------- --------
3  Jerry   5500
4  Jessica 6000
6  Tim     6000
7  Henry   5500
9  Kevin   5500

上面的例子,SQL語句的where子句和select子句分別用到了上下文中定義的數(shù)組和字典,使得本來需要通過子查詢和多表聯(lián)結來解決的問題,通過簡單的hash table解決了。如果SQL涉及到分布式數(shù)據(jù)庫,這些上下文變量會自動序列化到需要的節(jié)點。這不僅讓代碼看上去更簡潔,有更好的可讀性,而且提升了性能。在大數(shù)據(jù)分析中,很多數(shù)據(jù)表關聯(lián),即使SQL優(yōu)化器做了很多優(yōu)化,也難免帶來性能問題。

4.2 context by——對面板數(shù)據(jù)的友好支持

DolphinDB提供了類似其他數(shù)據(jù)庫系統(tǒng)的window function——context by。但是與window function相比,context by的語法更簡潔,并且沒有那么多限制,可以與select或update一起使用。

//按股票代碼進行分組,計算每個股票每天的回報。假設數(shù)據(jù)是時間順序排列的。
update trades set ret = ratios(price) - 1.0 context by sym

//按日期進行分組,計算每天每個股票的ret降序排名。
select date, symbol,  ret, rank(ret, false) + 1 as rank from trades where isValid(ret) context by date

//選擇每天ret排名前10的股票
select date, symbol, ret from trades where isValid(ret) context by date having rank(ret, false) < 10

4.3 asof join和window join——對時序數(shù)據(jù)的友好支持

t1 =  table(09:30m 09:31m 09:33m 09:34m as minute, 29.2 28.9 29.3 30.1 as price)
t2 =  table(09:30m 09:31m 09:34m 09:36m as minute,  51.2 52.4 51.9 52.8 as price)
select * from aj(t1, t2, `minute)

minute price t2_minute t2_price
------ ----- --------- --------
09:30m 29.2  09:30m    51.2
09:31m 28.9  09:31m    52.4
09:33m 29.3  09:31m    52.4
09:34m 30.1  09:34m    51.9

上面的例子中,t2中沒有與09:33m、09:34m對應的記錄,asof join(aj)會分別取t2中在09:33m、09:34m之前最近時間對應的記錄,即取t2中09:31m的記錄。

p = table(1 2 3 as id, 2018.06M 2018.07M 2018.07M as month)
s = table(1 2 1 2 1 2 as id, 2018.04M 2018.04M 2018.05M 2018.05M 2018.06M 2018.06M as month, 4500 5000 6000 5000 6000 4500 as wage)
select * from wj(p, s, -3:-1,,`id`month)

id month    avg_wage
-- -------- -----------
1  2018.06M 5250
2  2018.07M 4833.333333
3  2018.07M

上面的例子說明了window join(wj)的用法。wj首先取表p第一行記錄,即id=1,month=2018.06M。然后在表s中選擇id=1并且month在(2018.06M-3)到(2018.06M-1),即2018.03M到2018.05M之間的記錄來計算avg(wage)。因此avg_wage=(4500+6000)/2=5250。如此類推。

asof join和window join在金融分析領域有著廣泛的應用。一個經(jīng)典的應用是將交易表和報價表進行關聯(lián),計算個股交易成本。詳情可以參考使用Window Join快速估計個股交易成本。

4.4 SQL其它擴展

為了滿足大數(shù)據(jù)分析的要求,DolphinDB對SQL還做了很多擴展。比如,用戶的自定義函數(shù)無需編譯、打包或部署,即可在SQL中使用。又比如DolphinDB支持組合字段(Composite Column),可以將復雜分析函數(shù)的多個返回值輸出到數(shù)據(jù)表的一行。

factor1=3.2 1.2 5.9 6.9 11.1 9.6 1.4 7.3 2.0 0.1 6.1 2.9 6.3 8.4 5.6
factor2=1.7 1.3 4.2 6.8 9.2 1.3 1.4 7.8 7.9 9.9 9.3 4.6 7.8 2.4 8.7
t=table(take(1 2 3, 15).sort() as id, 1..15 as y, factor1, factor2)

//在輸出參數(shù)的同時,輸出t統(tǒng)計值。使用自定義函數(shù)包裝輸出結果
def myols(y,x){
    r=ols(y,x,true,2)
    return r.Coefficient.beta join r.RegressionStat.statistics[0]
}
select myols(y,[factor1,factor2]) as `alpha`beta1`beta2`R2 from t group by id

id alpha     beta1     beta2     R2
-- --------- --------- --------- --------
1  1.063991  -0.258685 0.732795  0.946056
2  6.886877  -0.148325 0.303584  0.992413
3  11.833867 0.272352  -0.065526 0.144837

5.遠程過程調用編程

DolphinDB與其他系統(tǒng)相比,在遠程過程調用(RPC)上的優(yōu)勢主要體現(xiàn)在兩個方面:第一,在DolphinDB中,無論是自定義函數(shù)還是內置函數(shù),我們都可以通過遠程過程調用發(fā)送到其他節(jié)點上運行,而其他系統(tǒng)不能遠程調用與自定義函數(shù)相關的函數(shù)。第二,DolphinDB的遠程過程調用無需編譯或者部署。系統(tǒng)會自動把相關函數(shù)定義和所需數(shù)據(jù)序列化到遠程節(jié)點。數(shù)據(jù)科學家或數(shù)據(jù)分析師在編寫與遠程過程調用相關的函數(shù)時,不需要工程師配合編譯和部署,可以直接在線使用,極大地提高了開發(fā)和分析效率。

下面的例子是使用remoteRun執(zhí)行遠程函數(shù):

h = xdb("localhost", 8081)
//在遠程節(jié)點上執(zhí)行一段腳本
remoteRun(h, "sum(1 3 5 7)")
16

//上述遠程調用也可以簡寫成
h("sum(1 3 5 7)")
16

//在遠程節(jié)點上執(zhí)行一個在遠程節(jié)點注冊的函數(shù)
h("sum", 1 3 5 7)
16

//在遠程系節(jié)點上執(zhí)行本地的自定義函數(shù)
def mysum(x) : reduce(+, x)
h(mysum, 1 3 5 7)
16

//在遠程節(jié)點(localhost:8081)上創(chuàng)建一個共享表sales
h("share table(2018.07.02 2018.07.02 2018.07.03 as date, 1 2 3 as qty, 10 15 7 as price) as sales")
//如果本地的自定義函數(shù)有依賴,依賴的自定義函數(shù)也會序列化到遠程節(jié)點
defg salesSum(tableName, d): select mysum(price*qty) from objByName(tableName) where date=d
h(salesSum, "sales", 2018.07.02)
40

DolphinDB還提供了與分布式計算相關的函數(shù)。mr和imr分別用于開發(fā)基于map-reduce和迭代的map-reduce分布式算法。用戶只需要指定分布式數(shù)據(jù)源和定制的核心函數(shù),譬如map函數(shù),reduce函數(shù),final函數(shù)等。下面我們先創(chuàng)建一個分布式表,添加一些模擬數(shù)據(jù),然后演示開發(fā)計算中位數(shù)和線性回歸的例子。

//模擬生成分布式表sample,用id分區(qū)
//y = 0.5 + 3x1 -0.5x2
n=10000000
x1 = pow(rand(1.0,n), 2)
x2 = norm(3.0:1.0, n)
y = 0.5 + 3 * x1 - 0.5*x2 + norm(0.0:1.0, n)
t=table(rand(10, n) as id, y, x1, x2)

login(`admin,"123456")
db = database("dfs://testdb", VALUE, 0..9)
db.createPartitionedTable(t, "sample", "id").append!(t)

利用自定義的map函數(shù)myOLSMap,內置的reudce函數(shù)加函數(shù)(+),自定義的final函數(shù)myOLSFinal,以及內置的map-reduce框架函數(shù)mr,快速構建了一個在分布式數(shù)據(jù)源上運行線性回歸的函數(shù)myOLSEx。

def myOLSMap(table, yColName, xColNames){
    x = matrix(take(1.0, table.rows()), table[xColNames])
    xt = x.transpose();
    return xt.dot(x), xt.dot(table[yColName])
}

def myOLSFinal(result){
    xtx = result[0]
    xty = result[1]
    return xtx.inv().dot(xty)[0]
}

def myOLSEx(ds, yColName, xColNames){
  return mr(ds, myOLSMap{, yColName, xColNames}, +, myOLSFinal)
}

//使用自己開發(fā)的分布式算法和分布式數(shù)據(jù)源計算線性回歸系數(shù)
sample = loadTable("dfs://testdb", "sample")
myOLSEx(sqlDS(), `y, `x1`x2)
[0.4991, 3.0001, -0.4996]

//使用內置的函數(shù)ols和未分的數(shù)據(jù)計算線性回歸的系數(shù),得到相同的結果
ols(y, [x1,x2],true)
[0.4991, 3.0001, -0.4996]

下面這個例子,我們構造一個算法,在分布式數(shù)據(jù)源上計算一組數(shù)據(jù)的近似中位數(shù)。算法的基本原理是利用bucketCount函數(shù),在每一個節(jié)點上分別計算一組bucket內的數(shù)據(jù)個數(shù),然后把各個節(jié)點上的數(shù)據(jù)累加。這樣我們可以找到中位數(shù)應該落在哪個區(qū)間內。如果這個區(qū)間不夠小,進一步細分這個區(qū)間,直到小于給定的精度要求。中位數(shù)的算法需要多次迭代,我們因此使用了迭代計算框架imr。

def medMap(data, range, colName): bucketCount(data[colName], double(range), 1024, true)

def medFinal(range, result){
    x= result.cumsum()
    index = x.asof(x[1025]/2.0)
    ranges = range[1] - range[0]
    if(index == -1)
        return (range[0] - ranges*32):range[1]
    else if(index == 1024)
        return range[0]:(range[1] + ranges*32)
    else{
        interval = ranges / 1024.0
        startValue = range[0] + (index - 1) * interval
        return startValue : (startValue + interval)
    }
}

def medEx(ds, colName, range, precision){
    termFunc = def(prev, cur): cur[1] - cur[0] <= precision
    return imr(ds, range, medMap{,,colName}, +, medFinal, termFunc).avg()
}

//使用自己開發(fā)的近似中位數(shù)算法,計算分布式數(shù)據(jù)的中位數(shù)。
sample = loadTable("dfs://testdb", "sample")
medEx(sqlDS(), `y, 0.0 : 1.0, 0.001)
-0.052973

//使用內置的med函數(shù)計算未分區(qū)的數(shù)據(jù)的中位數(shù)。
med(y)
-0.052947

6.元編程

DolphinDB支持使用元編程來動態(tài)創(chuàng)建表達式,如函數(shù)調用的表達式和SQL查詢表達式。元編程的一個典型應用是定制報表。用戶只需要輸入數(shù)據(jù)表、字段名稱和字段格式就能生成報表。具體實現(xiàn)如下:

//根據(jù)輸入的數(shù)據(jù)表,字段名稱和格式,以及過濾條件,動態(tài)生成SQL表達式并執(zhí)行
def generateReport(tbl, colNames, colFormat, filter){
	colCount = colNames.size()
	colDefs = array(ANY, colCount)
	for(i in 0:colCount){
		if(colFormat[i] == "") 
			colDefs[i] = sqlCol(colNames[i])
		else
			colDefs[i] = sqlCol(colNames[i], format{,colFormat[i]})
	}
	return sql(colDefs, tbl, filter).eval()
}

//模擬生成一個100行的數(shù)據(jù)表
t = table(1..100 as id, (1..100 + 2018.01.01) as date, rand(100.0, 100) as price, rand(10000, 100) as qty)

//輸入過濾條件,字段和格式,定制報表。過濾條件使用了元編程。
generateReport(t, ["id","date","price","qty"], ["000","MM/dd/yyyy", "00.00", "#,###"], < id<5 or id>95 >)

id  date       price qty
--- ---------- ----- -----
001 01/02/2018 50.27 2,886
002 01/03/2018 30.85 1,331
003 01/04/2018 17.89 18
004 01/05/2018 51.00 6,439
096 04/07/2018 57.73 8,339
097 04/08/2018 47.16 2,425
098 04/09/2018 27.90 4,621
099 04/10/2018 31.55 7,644
100 04/11/2018 46.63 8,383

DolphinDB編程語言為數(shù)據(jù)分析而生,天生具備處理海量數(shù)據(jù)的能力,功能強大,簡單易用。

上述就是小編為大家分享的大數(shù)據(jù)中如何分析語言DolphinDB腳本語言了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關知識,歡迎關注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


本文標題:大數(shù)據(jù)中如何分析語言DolphinDB腳本語言
鏈接地址:http://weahome.cn/article/piigds.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部