天天看點

Python的抽象方法實作的簡潔方式-- coding: utf-8 -------------------------重點關注這個全局函數的self參數

一個小的疑問:

全局函數的第一個參數可以為self麼?

全局函數可以作為成員函數使用麼?

下面是程式設計中碰見的一個文法現象,Python的面向對象的底層實作不是那麼簡單的。

一、來自scrapy架構的資料項加載器處理器代碼

-- coding: utf-8 --

import scrapy

import scrapy.loader

from scrapy.loader.processors import Join, MapCompose, TakeFirst

class TencentItemSpider(scrapy.Spider):

name = 'tencent_item'
allowed_domains = ['ke.qq.com']
start_urls = ['https://ke.qq.com/course/list?mt=1001&st=2002&tt=3019&price_min=1&page=1']

def parse(self, response):
    item = JobscrapyItem()
    # loader = scrapy.loader.ItemLoader(item=item,selector=response.selector)
    # loader = scrapy.loader.ItemLoader(item=item, response=response)

    # 測試xpath的代碼
    # price = response.xpath('/html/body//section/div/div/ul/li/div/span[@class="line-cell item-price"]/text()')
    # print('self:', len(price), price.getall())
    # company_name = response.xpath('/html/body//section/div/div/ul/li/div/span/a/text()')
    # print('self:', len(company_name), company_name.getall())

    course_list = response.xpath('/html/body//section[@class="main autoM clearfix"]/div/div/ul/li')
    print(len(course_list))
    for course_ in course_list:
        # loader = scrapy.loader.ItemLoader(item=item, selector=course_)
        loader = CourseItemLoader(item=item, selector=course_)
        loader.add_xpath('company_name', 'div/span/a/text()')
        loader.add_xpath('course_price', 'div/span[@class="line-cell item-price"]/text()')

        re = loader.get_xpath('div/span[@class="line-cell item-price"]/text()')
        # print(re)
        yield loader.load_item()
           

def in_price_processor(value):

print(float(value[0][1:]))
# print(value)
# 删除¥符号,轉換為float類型
return float(value[0][1:])
           

def out_price_processor(value):

print(value)
# 删除¥符号,轉換為float類型
return value[0] + 100
           

class JobscrapyItem(scrapy.Item):

# 教育訓練公司名
company_name = scrapy.Field()
# 課程價格
course_price = scrapy.Field(
    # input_processor=in_price_processor,
    # output_processor = out_price_processor
)
           

-----------------------重點關注這個全局函數的self參數

def price_processor(self, value):

print(self)   # 輸出的類型是CourseItemLoader
print(float(value[0][1:]))
# print(value)
# 删除¥符号,轉換為float類型
return float(value[0][1:])
           

class CourseItemLoader(scrapy.loader.ItemLoader):

course_price_in = price_processor     # 這個函數被調用。
# course_price_out = price_processor           

二、Python的接口程式設計實作技巧

從上面代碼可以得到一種類似抽象接口實作的程式設計技巧;盡管傳統使用的是抛出實作異常機制,我覺得這種程式設計思路也不錯(傳統異常方式穩定性,邏輯性更加好點),可以實作類的定義與成員實作的分離,達到程式設計的低耦合實作,在此備忘一下代碼。

class A:

meth = None
           

def meth(self, param):

print(param)
           

class B(A):

meth = meth
           

b = B()

b.meth('hello')

繼續閱讀