亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

基于Python實現(xiàn)剪切板實時監(jiān)控方法解析

系統(tǒng) 1852 0

前言

上網(wǎng)瀏覽網(wǎng)頁的時候,看見好的內(nèi)容免不了要使用復制粘貼,但是我們看到的內(nèi)容、心里想要的內(nèi)容和實際粘貼后的內(nèi)容往往不一致。數(shù)據(jù)的獲取始于復制,終于粘貼,那么問題來了,在這中間系統(tǒng)做了哪些操作,我們怎么能控制它呢?

人生苦短,我用python,查閱相關資料之后發(fā)現(xiàn)有很多不一樣的實現(xiàn)方式,如利用內(nèi)置ctypes模塊、tk模塊,第三方模塊如跨平臺的pyperclip模塊、clipboard模塊、pywin.win32clipboard模塊等等,大部分都封裝好了簡潔易用的高級接口,方便我們直接使用。

基于強迫癥的心理,本文分析比較了幾種主流的方式,對他們逐一進行源碼分析、讀寫性能實測,最后選擇了讀寫速度最快的一種做出一個實時剪切板監(jiān)控小案例,以供大家參考。

小案例實現(xiàn)的功能如下:

**實時監(jiān)測ctrl+c剪切板寫入事件,去除剪切板中指定字符或文本,如某些文字的后綴 (?V_?V)瞄。**

使用正則對某些文本進行智能替換,如將python2格式的代碼轉(zhuǎn)換為python3格式。

方式一:調(diào)用第三方pyperclip模塊

            
In [1]: import pyperclip
In [2]: data = pyperclip.paste()
In [3]: data
Out[3]: "print 'Hello World'\r\n――――――――――――――――\r\n版權(quán)聲明:本文為CSDN博主「...」的原創(chuàng)文章,遵循 CC 4.0 BY-SA 版權(quán)協(xié)議,轉(zhuǎn)載請附上原文出處鏈接及本聲明。\r\n原文鏈接:https://blog.csdn.net/.../article/details/..."
In [4]: data = data[7:12]
In [5]: pyperclip.copy(data)
In [6]: pyperclip.paste()
Out[6]: 'Hello'
          

源碼調(diào)用: 內(nèi)置ctypes模塊中的ctypes.windll.user32接口編寫,和pandas包的代碼一致,代碼位置:pandas.io.clipboard.windows,代碼引用如下

            
import ctypes
windll = ctypes.windll
safeGetClipboardData = CheckedCall(windll.user32.GetClipboardData)
safeGetClipboardData.argtypes = [UINT]
safeGetClipboardData.restype = HANDLE
safeSetClipboardData = CheckedCall(windll.user32.SetClipboardData)
safeSetClipboardData.argtypes = [UINT, HANDLE]
safeSetClipboardData.restype = HANDLE
          

優(yōu)點: 跨平臺,接口調(diào)用方便簡潔

缺點: 剪切板的數(shù)據(jù)格式只支持utf-8文本,頻繁讀寫速度較慢

方式二:調(diào)用第三方win32clipboard模塊

            
In [1]: import win32clipboard
 ...:
 ...: def clipboard_get():
 ...: """獲取剪貼板數(shù)據(jù)"""
 ...: win32clipboard.OpenClipboard()
 ...: data = win32clipboard.GetClipboardData()
 ...: win32clipboard.CloseClipboard()
 ...: return data
 ...:
 ...: def clipboard_set(data):
 ...: """設置剪貼板數(shù)據(jù)"""
 ...: win32clipboard.OpenClipboard()
 ...: win32clipboard.SetClipboardData(13, data)
 ...: win32clipboard.CloseClipboard()
 ...: return True
 ...:
In [2]: data = clipboard_get()
In [3]: data
Out[3]: "print 'Hello World'\r\n――――――――――――――――\r\n版權(quán)聲明:本文為CSDN博主「...」的原創(chuàng)文章,遵循 CC 4.0 BY-SA 版權(quán)協(xié)議,轉(zhuǎn)載請附上 原文出處鏈接及本聲明。\r\n原文鏈接:https://blog.csdn.net/.../article/details/..."

In [4]: clipboard_set(data[7:12])
Out[4]: True

In [5]: clipboard_get()
Out[5]: 'Hello'
          

源碼調(diào)用: C源碼封裝,python接口調(diào)用如下

            
def GetClipboardData(*args, **kwargs): # real signature unknown 
 pass
def SetClipboardData(*args, **kwargs): # real signature unknown
 pass
          

優(yōu)點: 原生C封裝讀寫速度最快,支持多種剪切板數(shù)據(jù)格式

缺點: 只適用于windows平臺,高頻率讀寫會報錯需要小心處理,utf-8格式之外的數(shù)據(jù)格式需要熟悉winuser.h庫自行設計編寫

方法三:調(diào)用內(nèi)置tkinter模塊

            
In [1]: from tkinter import *
 ...:
 ...: r = Tk()

In [2]: data = r.clipboard_get()

In [3]: data
Out[3]: "print 'Hello World'\n――――――――――――――――\n版權(quán)聲明:本文為CSDN博主「...」的原創(chuàng)文章,遵循 CC 4.0 BY-SA 版權(quán)協(xié)議,轉(zhuǎn)載請附上原文 出處鏈接及本聲明。\n原文鏈接:https://blog.csdn.net/.../article/details/..."

In [4]: r.clipboard_append(data[7:12])

In [5]: r.clipboard_get()
Out[6]: 'Hello'
          

注意: 在win10系統(tǒng)測試后發(fā)現(xiàn),使用tkinter模塊只能獲取剪切板數(shù)據(jù),不能將數(shù)據(jù)寫入剪切板,外部調(diào)用clipboard_board方法時,系統(tǒng)剪切板進程會被tk接管鎖死,此時在其他的應用按ctrl+v,粘貼的應用會直接處于卡死的狀態(tài),或者粘貼后內(nèi)容為空。

如果還是通過Tk()對象將數(shù)據(jù)寫入剪切板,只能采取下面的方法,設置延遲銷毀Tk對象,系統(tǒng)剪切板數(shù)據(jù)才會被更新,否則內(nèi)容還是為空(實測如果設置0.2秒以內(nèi)的頻率讀取,剪切板還是為空,這就很雞肋了):

            
from tkinter import *
import time
r = Tk()
r.withdraw()
r.clipboard_clear()
r.clipboard_append('some string')
r.update()
time.sleep(.2)
r.update()
r.destroy()
          

源碼調(diào)用: C源碼封裝,python接口調(diào)用如下

            
# 讀取剪切板數(shù)據(jù):
_tkinter.tkapp('clipboard', 'get')
# 寫入剪切板數(shù)據(jù):
_tkinter.tkapp('clipboard', 'append')
          

剪切板讀寫速度測試結(jié)果

基于Python實現(xiàn)剪切板實時監(jiān)控方法解析_第1張圖片

實時監(jiān)控小案例:

            
import win32clipboard
import re
import time
def clipboard_get():
  """獲取剪貼板數(shù)據(jù)"""
  win32clipboard.OpenClipboard()
  data = win32clipboard.GetClipboardData(win32clipboard.CF_UNICODETEXT)
  win32clipboard.CloseClipboard()
  return data
def clipboard_set(data):
  """設置剪貼板數(shù)據(jù)"""
  win32clipboard.OpenClipboard()
  win32clipboard.EmptyClipboard()
  win32clipboard.SetClipboardData(win32clipboard.CF_UNICODETEXT, data)
  win32clipboard.CloseClipboard()
# 初始化替換字符列表,相比于正則使用replace函數(shù)進行單字符替換更快
char_list = [('(', '('),
       (')', ')'),
       ('“', '"'),
       ('”', '"'),
       ('‘', '\''),
       (''', '\''),
       ('print ', 'print '),
       ('版權(quán)聲明:本文為CSDN', '版權(quán)聲明:本文為CSDN'),
       ]
# 預編譯正則替換匹配表達式
# 匹配python2格式的 print函數(shù)文本
sub_print = re.compile(r'\bprint\s+(.+)')
# 匹配csdn復制自帶的版權(quán)聲明后綴文本
sub_csdn = re.compile(r'―+\s+版權(quán)聲明:本文為CSDN.*\s+原文鏈接.*')
# 指定場景 sub替換函數(shù):python2格式的 print函數(shù) 替換為python3格式
def sub_fn(s):
  return 'print(' + s.group(1).strip() + ')\r\n'
# 判斷如果沒有要替換的字符則返回None,有則執(zhí)行替換操作,先進行字符列表replace,再執(zhí)行reg.sub(sub_fn, txt)
def char_replace_reg_sub(txt):
  new_txt = txt
  # 對字符列表中字符 逐一判斷,如果字符在文本中 則replace替換,如果都不在 則return None,不用再進行替換操作
  i = 0
  for old_char, new_char in char_list:
    if old_char in new_txt:
      i += 1
      new_txt = new_txt.replace(old_char, new_char)
  if i == 0:
    return None

  print('-' * 150, '\n【After char replace】:', new_txt)
  # 對指定場景替換 使用正則re.sub
  new_txt = sub_print.sub(sub_fn, new_txt)
  new_txt = sub_csdn.sub('', new_txt)
  print('【After sub replace:】', new_txt)
  return new_txt


def main():
  """后臺腳本:每隔0.2秒,讀取剪切板文本,檢查有無指定字符或字符串,如果有則執(zhí)行替換"""
  # recent_txt 存放最近一次剪切板文本,初始化值只多執(zhí)行一次paste函數(shù)讀取和替換
  recent_txt = clipboard_get()
  replaced_txt = char_replace_reg_sub(recent_txt)
  clipboard_set(recent_txt if replaced_txt is None else replaced_txt)

  while True:
    # txt 存放當前剪切板文本
    txt = clipboard_get()

    # 剪切板內(nèi)容和上一次對比如有變動,再進行內(nèi)容判斷,判斷后如果發(fā)現(xiàn)有指定字符在其中的話,再執(zhí)行替換
    if txt != recent_txt:
      # print(f'txt:{txt}')
      new_txt = char_replace_reg_sub(txt) # 沒查到要替換的子串,返回None

      if new_txt is not None:
        clipboard_set(new_txt)
        # 更新 recent_txt 為替換之后的文本,便于下次與 txt 剪切板文本對比,判斷內(nèi)容有無更新
        recent_txt = new_txt
    # 檢測間隔(延遲0.2秒)
    time.sleep(0.2)
if __name__ == '__main__':
  main()
          

運行效果:

            
-----------------------------------------------------------------------------------------
【Copy text】:

print 'Hello World' \r\n――――――――――――――――\r\n版權(quán)聲明:本文為CSDN博主「...」的原創(chuàng)文章,遵循 CC 4.0 BY-SA 版權(quán)協(xié)議,轉(zhuǎn)載請附上 原文出處鏈接及本聲明。\r\n原文鏈接:https://blog.csdn.net/.../article/details/...)
-----------------------------------------------------------------------------------------
【After replace:】:

print('Hello World')
-----------------------------------------------------------------------------------------
          

參考鏈接:

微軟開發(fā)文檔:https://docs.microsoft.com/zh-cn/windows/win32/dataxchg/using-the-clipboard?redirectedfrom=MSDN#_win32_Copying_Information_to_the_Clipboard

Stack Overflow:https://stackoverflow.com/questions/579687/how-do-i-copy-a-string-to-the-clipboard-on-windows-using-python

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。


更多文章、技術交流、商務合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發(fā)表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 国产精品久久久久不卡绿巨人 | 91国视频在线观看 | 国产成人精品日本亚洲直接 | 97视频在线免费观看 | 天天色综合2 | 成人性a激情免费视频 | 四虎www| 99精品视频在线成人精彩视频 | 爱唯侦察1024入口地址 | 福利视频91 | 中文字幕日韩精品一区口 | 亚洲欧美中文日韩在线 | 99久久国产亚洲综合精品 | 一级a毛片免费观看 | 国产亚洲欧美日韩综合另类 | 免费一级a毛片免费观看欧美大片 | 一级特黄女人生活片 | 久久国产美女免费观看精品 | 他也色在线 | 美女久草 | 日韩欧美精品综合一区二区三区 | 国产成人综合精品 | 日本三级日本三级人妇三级四 | 天天视频国产免费入口 | 久久综合久 | 九九这里只有精品视频 | japanese护士奶水 | 精品一区二区三区影片 | 欧美另类黑人巨大videos | 最新亚洲国产有精品 | 亚洲综合精品一二三区在线 | 国产精品一区二区三区四区五区 | 97久久精品人人澡人人爽 | 波多野结衣一区二区在线 | 日韩一级不卡 | 色综合a怡红院怡红院首页 色综合h | 亚洲va欧美va国产va天堂 | 性做久久久久久久久25的美女 | 四虎影院视频在线观看 | 黄色高清网站 | 一级欧美一级日韩毛片99 |