Python學習——Python進程

来源:http://www.cnblogs.com/huan-ge/archive/2017/06/28/7087961.html
-Advertisement-
Play Games

python中的多線程其實並不是真正的多線程,如果想要充分地使用多核CPU的資源,在python中大部分情況需要使用多進程。Python提供了非常好用的多進程包multiprocessing,只需要定義一個函數,Python會完成其他所有事情。藉助這個包,可以輕鬆完成從單進程到併發執行的轉換。mul ...


  python中的多線程其實並不是真正的多線程,如果想要充分地使用多核CPU的資源,在python中大部分情況需要使用多進程。Python提供了非常好用的多進程包multiprocessing,只需要定義一個函數,Python會完成其他所有事情。藉助這個包,可以輕鬆完成從單進程到併發執行的轉換。multiprocessing支持子進程、通信和共用數據、執行不同形式的同步,提供了Process、Queue、Pipe、Lock等組件。

1、Process

創建進程的類:Process([group [, target [, name [, args [, kwargs]]]]]),target表示調用對象,args表示調用對象的位置參數元組。kwargs表示調用對象的字典。name為別名。group實質上不使用。

方法:is_alive()、join([timeout])、run()、start()、terminate()。其中,Process以start()啟動某個進程。

屬性:authkey、daemon(要通過start()設置)、exitcode(進程在運行時為None、如果為–N,表示被信號N結束)、name、pid。其中daemon是父進程終止後自動終止,且自己不能產生新進程,必須在start()之前設置。

註:

is_live()用來查看進程的狀態

terminate()用來終止進程。

單進程:

 1 import multiprocessing
 2 import time
 3 def worker(interval):
 4     n=5
 5     while n > 0:
 6         print("The time is {0}".format(time.ctime()))
 7         time.sleep(interval)
 8         n -=1
 9 
10 if __name__ == "__main__":
11     p = multiprocessing.Process(target=worker,args=(3,))
12     p.start()
13     print("p.pid:",p.pid)
14     print("p.name:",p.name)
15     print("p.is_alive:",p.is_alive())

多進程:

 1 import multiprocessing
 2 import time
 3 
 4 def worker_1(interval):
 5     print ("worker_1")
 6     time.sleep(interval)
 7     print ("end worker_1")
 8 
 9 def worker_2(interval):
10     print ("worker_2")
11     time.sleep(interval)
12     print ("end worker_2")
13 
14 def worker_3(interval):
15     print ("worker_3")
16     time.sleep(interval)
17     print ("end worker_3")
18 
19 if __name__ == "__main__":
20     p1 = multiprocessing.Process(target = worker_1, args = (2,))
21     p2 = multiprocessing.Process(target = worker_2, args = (3,))
22     p3 = multiprocessing.Process(target = worker_3, args = (4,))
23 
24     p1.start()
25     p2.start()
26     p3.start()
27     # 用來獲得當前的CPU的核數,可以用來設置接下來子進程的個數。
28     # 用來獲得當前所有的子進程,包括daemon和非daemon子進程。
29     # p.name,p.pid分別表示進程的名字,進程id。 
30     print("The number of CPU is:" + str(multiprocessing.cpu_count()))
31     for p in multiprocessing.active_children():
32         print("child   p.name:" + p.name + "\tp.id" + str(p.pid))
33     print ("END!!!!!!!!!!!!!!!!!")

將進程定義為類:

 1 import multiprocessing
 2 import time
 3 
 4 class ClockProcess(multiprocessing.Process):
 5     def __init__(self, interval):
 6         multiprocessing.Process.__init__(self)
 7         self.interval = interval
 8 
 9     def run(self):
10         n = 5
11         while n > 0:
12             print("the time is {0}".format(time.ctime()))
13             time.sleep(self.interval)
14             n -= 1
15 
16 if __name__ == '__main__':
17     p = ClockProcess(3)
18     p.start()

daemon程式對比結果:
1.不加daemon

 1 import multiprocessing
 2 import time
 3 
 4 def worker(interval):
 5     print("work start:{0}".format(time.ctime()));
 6     time.sleep(interval)
 7     print("work end:{0}".format(time.ctime()));
 8 
 9 if __name__ == "__main__":
10     p = multiprocessing.Process(target = worker, args = (3,))
11     p.start()
12     print ("end!")
13 
14 #程式運行結果
15 '''
16 end!
17 work start:Wed Jun 28 00:07:57 2017
18 work end:Wed Jun 28 00:08:00 2017
19 '''

2.加daemon

 1 import multiprocessing
 2 import time
 3 
 4 def worker(interval):
 5     print("work start:{0}".format(time.ctime()));
 6     time.sleep(interval)
 7     print("work end:{0}".format(time.ctime()));
 8 
 9 if __name__ == "__main__":
10     p = multiprocessing.Process(target = worker, args = (3,))
11     p.daemon = True
12     p.start()
13     print ("end!")
14 
15 #程式運行結果
16 '''
17 end!
18 
19 '''

PS:因數進程設置了daemon屬性,主進程結束,它們就隨著結束了。
3.設置daemon執行完結束的方法

 1 import multiprocessing
 2 import time
 3 
 4 def worker(interval):
 5     print("work start:{0}".format(time.ctime()));
 6     time.sleep(interval)
 7     print("work end:{0}".format(time.ctime()));
 8 
 9 if __name__ == "__main__":
10     p = multiprocessing.Process(target = worker, args = (3,))
11     p.daemon = True
12     p.start()
13     p.join()
14     print "end!"
15 
16 # 結果
17 '''
18 work start:Tue Apr 21 22:16:32 2015
19 work end:Tue Apr 21 22:16:35 2015
20 end!
21 '''

2、Lock
當多個進程需要訪問共用資源的時候,Lock可以用來避免訪問的衝突。

 1 import multiprocessing
 2 import sys
 3 
 4 def worker_with(lock, f):
 5     with lock:
 6         fs = open(f, 'a+')
 7         n = 10
 8         while n > 1:
 9             fs.write("Lockd acquired via with\n")
10             n -= 1
11         fs.close()
12         
13 def worker_no_with(lock, f):
14     lock.acquire()
15     try:
16         fs = open(f, 'a+')
17         n = 10
18         while n > 1:
19             fs.write("Lock acquired directly\n")
20             n -= 1
21         fs.close()
22     finally:
23         lock.release()
24     
25 if __name__ == "__main__":
26     lock = multiprocessing.Lock()
27     f = "file.txt"
28     w = multiprocessing.Process(target = worker_with, args=(lock, f))
29     nw = multiprocessing.Process(target = worker_no_with, args=(lock, f))
30     w.start()
31     nw.start()
32     print ("end")

3、Semaphore
Semaphore用來控制對共用資源的訪問數量,例如池的最大連接數。

 1 import multiprocessing
 2 import time
 3 
 4 def worker(s, i):
 5     s.acquire()
 6     print(multiprocessing.current_process().name + "acquire")
 7     time.sleep(i)
 8     print(multiprocessing.current_process().name + "release\n")
 9     s.release()
10 
11 if __name__ == "__main__":
12     s = multiprocessing.Semaphore(2)   # 限制最多有兩個進程同時執行
13     for i in range(5):
14         p = multiprocessing.Process(target = worker, args=(s, i*2))
15         p.start()

運行結果:

 1 Process-4acquire
 2 Process-2acquire
 3 Process-2release
 4 
 5 Process-1acquire
 6 Process-1release
 7 
 8 Process-3acquire
 9 Process-4release
10 
11 Process-5acquire
12 Process-3release
13 
14 Process-5release

4、Event
Event實現進程間同步通信

 1 import multiprocessing
 2 import time
 3 
 4 def wait_for_event(e):
 5     print("wait_for_event: starting")
 6     e.wait()
 7     print("wairt_for_event: e.is_set()->" + str(e.is_set()))
 8 
 9 def wait_for_event_timeout(e, t):
10     print("wait_for_event_timeout:starting")
11     e.wait(t)
12     print("wait_for_event_timeout:e.is_set->" + str(e.is_set()))
13 
14 if __name__ == "__main__":
15     e = multiprocessing.Event()
16     w1 = multiprocessing.Process(name = "block",
17             target = wait_for_event,
18             args = (e,))
19 
20     w2 = multiprocessing.Process(name = "non-block",
21             target = wait_for_event_timeout,
22             args = (e, 2))
23     w1.start()
24     w2.start()
25 
26     time.sleep(3)
27 
28     e.set()
29     print("main: event is set")
30 
31 # 運行結果
32 '''
33 
34 wait_for_event: starting
35 
36 wait_for_event_timeout:starting
37 
38 wait_for_event_timeout:e.is_set->False
39 
40 main: event is set
41 
42 wairt_for_event: e.is_set()->True
43 
44 '''

5、Queue

Queue是多進程安全的隊列,可以使用Queue實現多進程之間的數據傳遞。put方法用以插入數據到隊列中,put方法還有兩個可選參數:blocked和timeout。如果blocked為True(預設值),並且timeout為正值,該方法會阻塞timeout指定的時間,直到該隊列有剩餘的空間。如果超時,會拋出Queue.Full異常。如果blocked為False,但該Queue已滿,會立即拋出Queue.Full異常。   get方法可以從隊列讀取並且刪除一個元素。同樣,get方法有兩個可選參數:blocked和timeout。如果blocked為True(預設值),並且timeout為正值,那麼在等待時間內沒有取到任何元素,會拋出Queue.Empty異常。如果blocked為False,有兩種情況存在,如果Queue有一個值可用,則立即返回該值,否則,如果隊列為空,則立即拋出Queue.Empty異常。
 1 import multiprocessing
 2 def writer_proc(q):
 3     try:
 4         q.put(1, block = False)
 5     except:
 6         pass
 7 
 8 def reader_proc(q):
 9     try:
10         print (q.get(block = False))
11     except:
12         pass
13 
14 if __name__ == "__main__":
15     q = multiprocessing.Queue()
16     writer = multiprocessing.Process(target=writer_proc, args=(q,))
17     writer.start()
18 
19     reader = multiprocessing.Process(target=reader_proc, args=(q,))
20     reader.start()
21 
22     reader.join()
23     writer.join()
24 
25 # 運行結果
26 # 1

6、Pipe

Pipe方法返回(conn1, conn2)代表一個管道的兩個端。Pipe方法有duplex參數,如果duplex參數為True(預設值),那麼這個管道是全雙工模式,也就是說conn1和conn2均可收發。duplex為False,conn1只負責接受消息,conn2只負責發送消息。   send和recv方法分別是發送和接受消息的方法。例如,在全雙工模式下,可以調用conn1.send發送消息,conn1.recv接收消息。如果沒有消息可接收,recv方法會一直阻塞。如果管道已經被關閉,那麼recv方法會拋出EOFError。

Pipe可以是單向(half-duplex),也可以是雙向(duplex)。我們通過mutiprocessing.Pipe(duplex=False)創建單向管道 (預設為雙向)。一個進程從PIPE一端輸入對象,然後被PIPE另一端的進程接收,單向管道只允許管道一端的進程輸入,而雙向管道則允許從兩端輸入。

 1 # proc1 發送消息,proc2,proc3輪流接收消息
 2 import multiprocessing
 3 import time
 4 
 5 def proc1(pipe):
 6     while True:
 7         for i in range(100):
 8             print ("send: %s" %(i))
 9             pipe.send(i)
10             time.sleep(1)
11 
12 def proc2(pipe):
13     while True:
14         print ("proc2 rev:", pipe.recv())
15         time.sleep(1)
16 
17 def proc3(pipe):
18     while True:
19         print ("proc3 rev:", pipe.recv())
20         time.sleep(1)
21 
22 if __name__ == "__main__":
23     pipe = multiprocessing.Pipe()
24     p1 = multiprocessing.Process(target=proc1, args=(pipe[0],))
25     p2 = multiprocessing.Process(target=proc2, args=(pipe[1],))
26     p3 = multiprocessing.Process(target=proc3, args=(pipe[1],))
27 
28     p1.start()
29     p2.start()
30     p3.start()
31 
32     p1.join()
33     p2.join()
34     p3.join()
35 
36 # 運行結果
37 '''
38 send: 0
39 proc2 rev: 0
40 send: 1
41 proc3 rev: 1
42 send: 2
43 proc2 rev: 2
44 send: 3
45 proc3 rev: 3
46 send: 4
47 proc2 rev: 4
48 send: 5
49 proc3 rev: 5
50 send: 6
51 proc2 rev: 6
52 send: 7
53 proc3 rev: 7
54 send: 8
55 proc2 rev: 8
56 send: 9
57 proc3 rev: 9
58 send: 10
59 proc2 rev: 10
60 ......
61 '''

7、Pool
在利用Python進行系統管理的時候,特別是同時操作多個文件目錄,或者遠程式控制制多台主機,並行操作可以節約大量的時間。當被操作對象數目不大時,可以直接利用multiprocessing中的Process動態成生多個進程,十幾個還好,但如果是上百個,上千個目標,手動的去限制進程數量卻又太過繁瑣,此時可以發揮進程池的功效。
Pool可以提供指定數量的進程,供用戶調用,當有新的請求提交到pool中時,如果池還沒有滿,那麼就會創建一個新的進程用來執行該請求;但如果池中的進程數已經達到規定最大值,那麼該請求就會等待,直到池中有進程結束,才會創建新的進程來執行它。

使用進程池(非阻塞)

 1 import multiprocessing
 2 import time
 3 
 4 def func(msg):
 5     print ("msg:", msg)
 6     time.sleep(3)
 7     print ("end")
 8 
 9 if __name__ == "__main__":
10     pool = multiprocessing.Pool(processes = 3)   # 池中最大進程數為3
11     for i in range(10):
12         msg = "hello %d" %(i)
13         pool.apply_async(func, (msg, ))   #維持執行的進程總數為processes,當一個進程執行完畢後會添加新的進程進去
14 
15     print ("Mark~ Mark~ Mark~~~~~~~~~~~~~~~~~~~~~~")
16     pool.close()
17     pool.join()   #調用join之前,先調用close函數,否則會出錯。執行完close後不會有新的進程加入到pool,join函數等待所有子進程結束
18     print ("Sub-process(es) done.")

運行結果:

Mark~ Mark~ Mark~~~~~~~~~~~~~~~~~~~~~~
msg: hello 0
msg: hello 1
msg: hello 2
end
msg: hello 3
end
msg: hello 4
end
msg: hello 5
end
msg: hello 6
end
msg: hello 7
end
msg: hello 8
end
msg: hello 9
end
end
end
Sub-process(es) done.

函數解釋:

  • apply_async(func[, args[, kwds[, callback]]]) 它是非阻塞,apply(func[, args[, kwds]])是阻塞的(理解區別,看例1例2結果區別)
  • close()    關閉pool,使其不在接受新的任務。
  • terminate()    結束工作進程,不在處理未完成的任務。
  • join()    主進程阻塞,等待子進程的退出, join方法要在close或terminate之後使用。

執行說明:創建一個進程池pool,並設定進程的數量為3,range(4)會相繼產生四個對象[0, 1, 2, 3,4,5,6,7,8,9],十個對象被提交到pool中,因pool指定進程數為3,所以0、1、2會直接送到進程中執行,當其中一個執行完事後才空出一個進程處理對象3,所以會出現輸出“msg: hello 3”出現在"end"後。因為為非阻塞,主函數會自己執行自個的,不搭理進程的執行,所以運行完for迴圈後直接輸出“mMsg: hark~ Mark~ Mark~~~~~~~~~~~~~~~~~~~~~~”,主程式在pool.join()處等待各個進程的結束。

 

使用線程池(阻塞)

 1 import multiprocessing
 2 import time
 3 
 4 def func(msg):
 5     print ("msg:", msg)
 6     time.sleep(3)
 7     print ("end")
 8 
 9 if __name__ == "__main__":
10     pool = multiprocessing.Pool(processes = 3)   # 池中最大進程數為3
11     for i in range(10):
12         msg = "hello %d" %(i)
13         pool.apply(func, (msg, ))   #維持執行的進程總數為processes,當一個進程執行完畢後會添加新的進程進去
14 
15     print ("Mark~ Mark~ Mark~~~~~~~~~~~~~~~~~~~~~~")
16     pool.close()
17     pool.join()   #調用join之前,先調用close函數,否則會出錯。執行完close後不會有新的進程加入到pool,join函數等待所有子進程結束
18     print ("Sub-process(es) done.")
19 
20 # 運行結果
21 '''
22 msg: hello 0
23 end
24 msg: hello 1
25 end
26 msg: hello 2
27 end
28 msg: hello 3
29 end
30 msg: hello 4
31 end
32 msg: hello 5
33 end
34 msg: hello 6
35 end
36 msg: hello 7
37 end
38 msg: hello 8
39 end
40 msg: hello 9
41 end
42 Mark~ Mark~ Mark~~~~~~~~~~~~~~~~~~~~~~
43 Sub-process(es) done.
44 '''

使用多個進程池

 1 import multiprocessing
 2 import os, time, random
 3 
 4 
 5 def Lee():
 6     print("\nRun task Lee-%s" % (os.getpid()))  # os.getpid()獲取當前的進程的ID
 7     start = time.time()
 8     time.sleep(random.random() * 10)  # random.random()隨機生成0-1之間的小數
 9     end = time.time()
10     print( 'Task Lee, runs %0.2f seconds.' % (end - start))
11 
12 
13 def Marlon():
14     print("\nRun task Marlon-%s" % (os.getpid()))
15     start = time.time()
16     time.sleep(random.random() * 40)
17     end = time.time()
18     print('Task Marlon runs %0.2f seconds.' % (end - start))
19 
20 
21 def Allen():
22     print("\nRun task Allen-%s" % (os.getpid()))
23     start = time.time()
24     time.sleep(random.random() * 30)
25     end = time.time()
26     print('Task Allen runs %0.2f seconds.' % (end - start))
27 
28 
29 def Frank():
30     print( "\nRun task Frank-%s" % (os.getpid()))
31     start = time.time()
32     time.sleep(random.random() * 20)
33     end = time.time()
34     print( 'Task Frank runs %0.2f seconds.' % (end - start))
35 
36 
37 if __name__ == '__main__':
38     function_list = [Lee, Marlon, Allen, Frank]
39     print("parent process %s" % (os.getpid()))
40 
41     pool = multiprocessing.Pool(4)
42     for func in function_list:
43         pool.apply_async(func)  # Pool執行函數,apply執行函數,當有一個進程執行完畢後,會添加一個新的進程到pool中
44 
45     print('Waiting for all subprocesses done...')
46     pool.close()
47     pool.join()  # 調用join之前,一定要先調用close() 函數,否則會出錯, close()執行後不會有新的進程加入到pool,join函數等待素有子進程結束
48     print( 'All subprocesses done.')
49 
50 # 運行結果
51 '''
52 parent process 3256
53 Waiting for all subprocesses done...
54 
55 Run task Lee-2196
56 
57 Run task Marlon-4580
58 
59 Run task Allen-5920
60 
61 Run task Frank-6384
62 Task Allen runs 2.15 seconds.
63 Task Lee, runs 9.99 seconds.
64 Task Frank runs 14.14 seconds.
65 Task Marlon runs 32.74 seconds.
66 All subprocesses done.
67 
68 '''

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Spring 對 JPA 的支持已經非常強大,開發者只需關心核心業務邏輯的實現代碼,無需過多關註 EntityManager 的創建、事務處理等 JPA 相關的處理。Spring Data JPA更是能夠根據方法名字自動實現持久層。 目標 這次我們的目標還是實現前面幾節的功能,即對Category的 ...
  • JPA是Java EE5規範之一,是一個orm規範,由廠商來實現該規範。目前有hibernate,OpenJPA,TopLink和EclipseJPA等實現 spring提供三種方法集成JPA:1、LocalEntityManagerFactoryBean:適用於那些僅使用JPA進行數據訪問的項目。 ...
  • >獲取當前時間: >獲取當天開始、結束時間: >獲取相應時間戳: >時間戳轉為字元串格式: ...
  • 15套java架構師、集群、高可用、高可擴展、高性能、高併發、性能優化、Spring boot、Redis、ActiveMQ、Nginx、Mycat、Netty、Jvm大型分散式項目實戰視頻教程 視頻課程包含: 高級Java架構師包含:Spring boot、Spring cloud、Dubbo、R ...
  • program ex1807;var n,i,j,t,f,l,q,z:integer;s1,s2:longint; a:array[1..20,1..20] of integer;begin readln(n); t:=1; f:=1; l:=n;i:=1;q:=1;z:=n; while t<=n ...
  • #include <bits/stdc++.h>using namespace std;int main(){ string s; int i=1,n,z; cin>>s; z=s[0]-48;n=s.length(); while(i<n) { if(s[i]=='+') z=z+(s[i+1]- ...
  • 在servlet中,轉發和重定向是由request和response完成的。兩者之間的區別請看我之前的文章。那麼在springMVC中是如何完成的呢? /**轉發**/ @RequestMapping("/login.do") public String login(HttpServletReque ...
  • 真是前所未有的大失誤啊,竟然快兩個月忘記寫博客了! 公司的項目組連著解散了倆,挺動蕩的! 終於把欠公司的錢還上了,這次改欠支付寶了! 摩托車也練的可以正常騎行了,拉過幾次高速,最快到111了,油門還沒到底,不知道磨合期過了之後是不是能更快一些! 房貸也開始還了,現在每個月都是收支都是負數,情況不太樂 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...