<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>
"); //-->

博客專(zhuān)欄

EEPW首頁(yè) > 博客 > Python實(shí)現對腦電數據情緒分析

Python實(shí)現對腦電數據情緒分析

發(fā)布人:AI科技大本營(yíng) 時(shí)間:2021-08-07 來(lái)源:工程師 發(fā)布文章

引言

腦電波是一類(lèi)由大腦中局部群體神經(jīng)元同步放電所形成的具有時(shí)空特征的腦電活動(dòng)電波。德國醫生漢斯·伯格(Hans Berger)在1924年首次在人的頭骨上記錄到腦電波圖(electroencephalography,EEG)。心理學(xué)研究表明,人類(lèi)的認知和感知可以通過(guò)腦電波來(lái)表達。當大腦的嗅覺(jué)、聽(tīng)覺(jué)、視覺(jué)、味覺(jué)及觸覺(jué)神經(jīng)受到刺激時(shí),其刺激反應信號可以通過(guò)腦電波表達出來(lái),從而揭示感官和人員之間的心理關(guān)聯(lián)性。其中大量研究展示了使用腦電信號連續確定個(gè)人舒適感的可行性,并且可以得到更加客觀(guān)的數據。近來(lái)則有研究表明觸覺(jué)刺激與腦電波的θ,α,β這三個(gè)頻段均存在關(guān)聯(lián)性。

傳統的情緒識別主要是基于面部特征、肢體動(dòng)作和語(yǔ)音的研究,這些外在特征容易偽裝,并不能反應出真實(shí)的情緒,腦電信號可以反映大腦在加工情緒時(shí)所伴隨的神經(jīng)電生理活動(dòng),能夠很好的彌補傳統研究方法的缺陷。

1.png

故本項目通過(guò)使用python語(yǔ)言搭建KNN機器學(xué)習算法實(shí)現對EEG腦電波數據的情緒分析分類(lèi)。其最終實(shí)現的效果如下圖可見(jiàn):

2.png

No.1 “基本介紹”

環(huán)境要求

本次環(huán)境使用的是python3.6.5+windows平臺。主要用的庫有:

csv模塊。CSV庫在這里用來(lái)讀取CSV數據集文件。其中CSV文件是逗號分隔值文件,是一種常用的文本格式,用以存儲表格數據,包括數字或者字符。很多程序在處理數據時(shí)都會(huì )碰到csv這種格式的文件,它的使用是比較廣泛的。

scipy模塊。scipy作為高級科學(xué)計算庫:和numpy聯(lián)系很密切,scipy一般都是操控numpy數組來(lái)進(jìn)行科學(xué)計算、統計分析,所以可以說(shuō)是基于numpy之上了。scipy有很多子模塊可以應對不同的應用,例如插值運算,優(yōu)化算法等等。scipy則是在numpy的基礎上構建的更為強大,應用領(lǐng)域也更為廣泛的科學(xué)計算包。正是出于這個(gè)原因,scipy需要依賴(lài)numpy的支持進(jìn)行安裝和運行。以Python為基礎的scipy的另一個(gè)好處是,它還提供了一種強大的編程語(yǔ)言,可用于開(kāi)發(fā)復雜的程序和專(zhuān)門(mén)的應用程序。使用scipy的科學(xué)應用程序受益于世界各地的開(kāi)發(fā)人員在軟件領(lǐng)域的許多小眾領(lǐng)域中開(kāi)發(fā)的附加模塊。

pathlib模塊。該模塊提供了一些使用語(yǔ)義表達來(lái)表示文件系統路徑的類(lèi),這些類(lèi)適合多種操作系統。

Pickle模塊。python的pickle模塊實(shí)現了基本的數據序列和反序列化。通過(guò)pickle模塊的序列化操作我們能夠將程序中運行的對象信息保存到文件中去,永久存儲;通過(guò)pickle模塊的反序列化操作,我們能夠從文件中創(chuàng )建上一次程序保存的對象。

以及其他常用模塊,如OpenCV等不一一介紹了。

KNN算法介紹

其中k近鄰算法(k Nearest Neighbor,kNN)是一種理論上比較成熟的智能算法,最初由Cover和Hart于1968年提出。kNN算法思路簡(jiǎn)單直觀(guān):對于給定的測試樣本,基于某種距離度量找出訓練集中與其最靠近的k個(gè)訓練樣本,然后基于這k個(gè)“鄰居”的信息來(lái)進(jìn)行預測。

kNN方法可以實(shí)現分類(lèi)任務(wù)和回歸任務(wù)。在分類(lèi)任務(wù)中,一般使用“投****法”,即在k個(gè)“近鄰”樣本中出現最多的種類(lèi)標記作為預測結果。在回歸任務(wù)中,一般使用“平均法”,即將這k個(gè)樣本的實(shí)值輸出標記的平均值作為預測結果。在定義了相似度評價(jià)準則后,還可以基于相似度的大小進(jìn)行加權投****和加權平均。相似度越大的樣本權重越大。其中KNN算法分類(lèi)流程如下圖:

3.png

No.2 “模型搭建”

數據集準備

首先我們使用官方提供的EEG數據集,放置data文件夾下:

4.png

數據特征提取

首先我們使用官方提供的EEG數據集,放置data文件夾下:

通過(guò)使用pickle實(shí)現對dat數據文件的讀取,獲取各個(gè)數據文件中特征向量,并在每個(gè)信道中進(jìn)行fft。

其中輸入:維數為N × M的通道數據,N為通道個(gè)數,M為每個(gè)通道的腦電圖數據個(gè)數。

輸出:維度為N x M的FFT結果。N表示信道數,M表示每個(gè)信道的FFT數據數。

關(guān)鍵代碼如下:

for file in os.listdir("../data"):
      fname = Path("../data/"+file)
      x = cPickle.load(open(fname, 'rb'), encoding="bytes")
      for i in range(40):
                num+=1
                eeg_realtime = x[b'data'][i]
                label = x[b'labels'][i]
                if label[0] >6:
                    val_v = 3
                elif label[0] < 4:
                    val_v = 1
                else:
                    val_v = 2
                if label[1] > 6:
                    val_a = 3
                elif label[1] < 4:
                    val_a = 1
                else:
                    val_a = 2
                if i < 39:
                    va_label.write(str(val_v) + ",")
                    ar_label.write(str(val_a) + ",")
                if num==1280:
                    va_label.write(str(val_v) )
                    ar_label.write(str(val_v) )
                eeg_raw = np.reshape(eeg_realtime, (40, 8064))
                eeg_raw = eeg_raw[:32, :]
                eeg_feature_arr = self.get_feature(eeg_raw)
                for f in range(160):
                    if f == 159:
                        fout_data.write(str(eeg_feature_arr[f]))
                    else:
                        fout_data.write(str(eeg_feature_arr[f]) + ",")
                fout_data.write("\n")
                print(file + " Video watched")

然后從計算fft得到所有通道的頻率。輸入:維數為N × M的通道數據,N為通道個(gè)數,M為每個(gè)通道的腦電圖數據個(gè)數。輸出:每個(gè)通道的頻帶:Delta, Theta, Alpha, Beta和Gamma。

# Length data channel
L = len(all_channel_data[0])
# Sampling frequency
Fs = 128
# Get fft data
data_fft = self.do_fft(all_channel_data)
# Compute frequencymotio
frequency = map(lambda x: abs(x // L), data_fft)
frequency = map(lambda x: x[: L // 2 + 1] * 2, frequency)
f1, f2, f3, f4, f5 = itertools.tee(frequency, 5)
# List frequency
delta = np.array(list(map(lambda x: x[L * 1 // Fs - 1: L * 4 // Fs], f1)))
theta = np.array(list(map(lambda x: x[L * 4 // Fs - 1: L * 8 // Fs], f2)))
alpha = np.array(list(map(lambda x: x[L * 5 // Fs - 1: L * 13 // Fs], f3)))
beta = np.array(list(map(lambda x: x[L * 13 // Fs - 1: L * 30 // Fs], f4)))
gamma = np.array(list(map(lambda x: x[L * 30 // Fs - 1:L * 50 // Fs], f5)))

模型預測

從特征得到arousal值和valence值。其中Valence-Arousal分別代表情緒的正負向程度和激動(dòng)程度,基于這兩個(gè)維度可以構成一個(gè)情感平面空間,任何一個(gè)情感狀態(tài)可以通過(guò)具體的Valence-Arousal數值,映射到VA平面空間中具體的一個(gè)點(diǎn)。

輸入:來(lái)自所有頻帶和信道的特征(標準差和平均值),維度為1 × M(特征數量)。

輸出:由每一種arousal和valence產(chǎn)生的1至3級情緒。1表示低,2表示中性,3表示高。

self.train_arousal = self.train_arousal[40*(index-1):40*index]
self.train_valence = self.train_valence[40*(index-1):40*index]
self.class_arousal = np.array([self.class_arousal[0][40*(index-1):40*index]])
self.class_valence = np.array([self.class_valence [0][40*(index-1):40*index]])
distance_ar = list(map(lambda x: ss.distance.canberra(x, feature), self.train_arousal))
# Compute canberra with valence training data
distance_va = list(map(lambda x: ss.distance.canberra(x, feature), self.train_valence))
# Compute 3 nearest index and distance value from arousal
idx_nearest_ar = np.array(np.argsort(distance_ar)[:3])
val_nearest_ar = np.array(np.sort(distance_ar)[:3])
# Compute 3 nearest index and distance value from arousal
idx_nearest_va = np.array(np.argsort(distance_va)[:3])
val_nearest_va = np.array(np.sort(distance_va)[:3])
# Compute comparation from first nearest and second nearest distance.、If comparation less or equal than 0.7, then take class from the first nearest distance. Else take frequently class.
# Arousal
comp_ar = val_nearest_ar[0] / val_nearest_ar[1]

然后從特征中獲取情感類(lèi)。

輸入:來(lái)自所有頻段和信道的特征(標準偏差),尺寸為1 × M(特征數量)。

輸出:根據plex模型,情緒在1到5之間的類(lèi)別。

class_ar, class_va = self.predict_emotion(feature,fname)

print(class_ar, class_va)

if class_ar == 2.0 or class_va == 2.0:

    emotion_class = 5

if class_ar == 3.0 and class_va == 1.0:

    emotion_class = 1

elif class_ar == 3.0 and class_va == 3.0:

    emotion_class = 2

elif class_ar == 1.0 and class_va == 3.0:

    emotion_class = 3

elif class_ar == 1.0 and class_va == 1.0:

    emotion_class = 4

預測結果如下圖可見(jiàn):

5.png

完整代碼

鏈接:

https://pan.baidu.com/s/1BX58sJv037eIJx9A8jWt3g

提取碼:rwpe

*博客內容為網(wǎng)友個(gè)人發(fā)布,僅代表博主個(gè)人觀(guān)點(diǎn),如有侵權請聯(lián)系工作人員刪除。



關(guān)鍵詞: Python

相關(guān)推薦

技術(shù)專(zhuān)區

關(guān)閉
国产精品自在自线亚洲|国产精品无圣光一区二区|国产日产欧洲无码视频|久久久一本精品99久久K精品66|欧美人与动牲交片免费播放
<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>