神經網絡 之 感知器的概念和實現

本文結構:

  1. 什么是感知器
  2. 有什么用
  3. 代碼實現

1. 什么是感知器

如下圖,這個神經網絡中,每個圓圈都是一個神經元,神經元也叫做感知器

只有一個隱藏層的神經網絡就能擬合任何一個函數,但是它需要很多很多的神經元。
而深層網絡用相對少的神經元就能擬合同樣的函數,但是層數增加了,不太容易訓練,需要大量的數據。
為了擬合一個函數,可以使用一個淺而寬的網絡,也可以使用一個深而窄的網絡,后者更節約資源。

下圖單挑出一個感知器來看:
向它輸入 inputs,經過 加權 求和,再作用上激活函數后,得到一個輸出值

感知器的激活函數可以有很多選擇,關于激活函數可以看 常用激活函數比較


2. 有什么用

用感知器可以實現 and 函數,or 函數,還可以擬合任何的線性函數,任何線性分類或線性回歸問題都可以用感知器來解決。

但是,感知器卻不能實現異或運算,如下圖所示,異或運算不是線性的,無法用一條直線把 0 和 1 分開。

xor

訓練權重和偏置的算法如下:

其中,t 是訓練樣本的實際值,y 是感知器的輸出值,即由 f 計算出來的。eta 稱為學習速率,是個常數,作用是控制每一步調整權的幅度。


3. 代碼實現

[main]

先訓練and感知器

and_perception = train_and_perceptron()

得到訓練后獲得的權重和偏置

print and_perception    
weights :[0.1, 0.2]
bias    :-0.200000

再去測試,看結果是否正確

print '1 and 1 = %d' % and_perception.predict([1, 1])

其中

[train_and_perceptron]

先創建感知器,輸入參數個數為2(因為and是二元函數),激活函數為f

    p = Perceptron(2, f)

f 為

def f(x):
    return 1 if x > 0 else 0

輸入訓練data,迭代10次, 學習速率為0.1

    input_vecs, labels = get_training_dataset()
    p.train(input_vecs, labels, 10, 0.1)

訓練data為

    input_vecs = [[1,1], [0,0], [1,0], [0,1]]
    labels = [1, 0, 0, 0]

關于

[train]

一共迭代 10 次,每次迭代時,
先計算感知器在當前權重下的輸出,然后更新weights

            output = self.predict(input_vec)
            self._update_weights(input_vec, output, label, rate)

其中

[_update_weights]

就是用訓練算法里面的兩個公式

        delta = label - output
        self.weights = map(
            lambda (x, w): w + rate * delta * x,
            zip(input_vec, self.weights) )
        self.bias += rate * delta

[predict]

就用感知器的函數 f:

        return self.activator(
            reduce(lambda a, b: a + b,
                   map(lambda (x, w): x * w,  
                       zip(input_vec, self.weights))
                , 0.0) + self.bias)

完整代碼:

#!/usr/bin/python
#-*-coding:utf-8 -*-

class Perceptron(object):
    def __init__(self, input_num, activator):
        '''
        初始化感知器,設置輸入參數的個數,以及激活函數。
        激活函數的類型為double -> double
        '''
        self.activator = activator
        # 權重向量初始化為0
        self.weights = [0.0 for _ in range(input_num)]
        # 偏置項初始化為0
        self.bias = 0.0
        
    def __str__(self):
        '''
        打印學習到的權重、偏置項
        '''
        return 'weights\t:%s\nbias\t:%f\n' % (self.weights, self.bias)
        
    def predict(self, input_vec):
        '''
        輸入向量,輸出感知器的計算結果
        '''
        # 把input_vec[x1,x2,x3...]和weights[w1,w2,w3,...]打包在一起
        # 變成[(x1,w1),(x2,w2),(x3,w3),...]
        # 然后利用map函數計算[x1*w1, x2*w2, x3*w3]
        # 最后利用reduce求和
        return self.activator(
            reduce(lambda a, b: a + b,
                   map(lambda (x, w): x * w,  
                       zip(input_vec, self.weights))
                , 0.0) + self.bias)
                
    def train(self, input_vecs, labels, iteration, rate):
        '''
        輸入訓練數據:一組向量、與每個向量對應的label;以及訓練輪數、學習率
        '''
        for i in range(iteration):
            self._one_iteration(input_vecs, labels, rate)
            
    def _one_iteration(self, input_vecs, labels, rate):
        '''
        一次迭代,把所有的訓練數據過一遍
        '''
        # 把輸入和輸出打包在一起,成為樣本的列表[(input_vec, label), ...]
        # 而每個訓練樣本是(input_vec, label)
        samples = zip(input_vecs, labels)
        # 對每個樣本,按照感知器規則更新權重
        for (input_vec, label) in samples:
            # 計算感知器在當前權重下的輸出
            output = self.predict(input_vec)
            # 更新權重
            self._update_weights(input_vec, output, label, rate)
            
    def _update_weights(self, input_vec, output, label, rate):
        '''
        按照感知器規則更新權重
        '''
        # 把input_vec[x1,x2,x3,...]和weights[w1,w2,w3,...]打包在一起
        # 變成[(x1,w1),(x2,w2),(x3,w3),...]
        # 然后利用感知器規則更新權重
        delta = label - output
        self.weights = map(
            lambda (x, w): w + rate * delta * x,
            zip(input_vec, self.weights) )
        # 更新bias
        self.bias += rate * delta

def f(x):
    '''
    定義激活函數f
    '''
    return 1 if x > 0 else 0
    
def get_training_dataset():
    '''
    基于and真值表構建訓練數據
    '''
    # 構建訓練數據
    # 輸入向量列表
    input_vecs = [[1,1], [0,0], [1,0], [0,1]]
    # 期望的輸出列表,注意要與輸入一一對應
    # [1,1] -> 1, [0,0] -> 0, [1,0] -> 0, [0,1] -> 0
    labels = [1, 0, 0, 0]
    return input_vecs, labels   
     
def train_and_perceptron():
    '''
    使用and真值表訓練感知器
    '''
    # 創建感知器,輸入參數個數為2(因為and是二元函數),激活函數為f
    p = Perceptron(2, f)
    # 訓練,迭代10輪, 學習速率為0.1
    input_vecs, labels = get_training_dataset()
    p.train(input_vecs, labels, 10, 0.1)
    #返回訓練好的感知器
    return p
    
if __name__ == '__main__': 
    # 訓練and感知器
    and_perception = train_and_perceptron()
    # 打印訓練獲得的權重
    print and_perception
    # 測試
    print '1 and 1 = %d' % and_perception.predict([1, 1])
    print '0 and 0 = %d' % and_perception.predict([0, 0])
    print '1 and 0 = %d' % and_perception.predict([1, 0])
    print '0 and 1 = %d' % and_perception.predict([0, 1])

參考資料:
https://www.zybuluo.com/hanbingtao/note/433855


推薦閱讀 歷史技術博文鏈接匯總
也許可以找到你想要的

我是 不會停的蝸牛 Alice
85后全職主婦
喜歡人工智能,行動派
創造力,思考力,學習力提升修煉進行中
歡迎您的喜歡,關注和評論!

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,428評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,024評論 3 413
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,285評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,548評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,328評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,878評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,971評論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,098評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,616評論 1 331
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,554評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,725評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,243評論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 43,971評論 3 345
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,361評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,613評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,339評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,695評論 2 370

推薦閱讀更多精彩內容