###### tags: `資訊科技` # 程式設計(使用Python、ZeroJudge) ## 零、評量、學習歷程 ### 1. 評量方式 + 上課基本練習(遲交期限:1週) **40%** + 作業題數 **20%** (期中10%+期末10%) - 作業登記表:[103](https://docs.google.com/spreadsheets/d/1Ftcil7r6wpAnPucz3zF9pd7iC_5VP63wykgbAJPIj9w/edit?usp=sharing)、[104](https://docs.google.com/spreadsheets/d/1QLfSVBIxS9xOFxaog1N5YDTIsofhI_-OhHallVQt590/edit?usp=sharing)、[106](https://docs.google.com/spreadsheets/d/1rtZrk_LzjXbvl84E1Ge-YFCWcXB7tnOstrU2xlsjEpE/edit?usp=sharing) - [歷年作業登記表](https://hackmd.io/@cube/r1Cw2JgDT) - 每寫完一題作業,請將程式碼連結登錄於作業登記表,以統計作業題數。請小心不要改到別人的資料。惡意編刪別人的資料,視情節扣分或校規處理。 - 作業原始分數:解題數 $*$ 6 (最高120分) ``` if 上機考分數>=60 作業實得分數=作業原始分數 else 作業實得分數=作業原始分數*(上機考成績/60) ``` - 不要為了衝題數,做複製貼上、不求甚解這種沒意義的事。上機考0分,全部都寫也是0分。 - 不寫作業,除了無作業成績,上機考時網路會斷線,沒有任何參考資料,也必定寫不出來,請每個單元至少完成 3 題作業,熟練基本語法。**(不寫作業被當的機會很高,除非你有本事上機考2次都及格)** - 不要考前才寫作業,初學程式會有很多錯誤,要花時間除錯。請自我要求每週至少要寫 1 題作業,上機考才會比較保險。 - 不會的可以去參考網路上的程式碼、解法,同學也可以互相討論,而且是鼓勵的。不論如何,都要在了解解法後,不看別人的程式碼,自己親自寫一遍。 - 有做作業,考差了才有補救的機會,作業題為期末解題報告的題目。 + 期中上機考(迴圈教完後 1 週) **20%** + 期末上機考(倒數第 2 週) **20%** + 期末解題報告 **10%** - 非必要,想高分或不及格的同學請保握最後機會。 - 繳交期限:最後一週上課前 1 天。 - 分數:每一題 10 分。每個單元至多可放 2 題 (不可以都寫同一單元的題目) - 報告格式: * 第1頁為解題目錄(單元、題號、題目、頁碼)。 * 每題的子標題為 (1) 單元、題號、題目、題目簡述 (2) 解題動態、評分結果截圖 (3) 解題思路 (4) 程式碼 (5) 反思(遇到的錯誤、修正的地方、更好的方法…) (6) 錯誤程式碼 - 檢核方式: * 有交解題報告者,最後一週會請你在沒有網路的狀態下重寫一題當做檢核(題目由老師挑選),寫不出來作業直接作廢不算分,請務必自行思考完成。 * 不要為了分數直接複製貼上別人的程式碼,那是無意義的行為,檢核時你就會現出原形,就不用做這種白工了。 - 可整合上課內容後完成課程學習成果,上傳至學習歷程檔案平台。請儘量挑選一開始有錯,然後修正的題目,將修正的想法寫出來更能顯示出反思的能力。 + 如果對寫程式有興趣並要參加APCS檢定,下學期多元選修可以選修APCS(大學程式設計先修檢測)程式實作,挑選標準主要為ZJ的解題數(請寫在多元選修意向書裏)。 ### 2. [TOI推廣計畫-線上練習賽](https://tpmso.org/toi/index.php/reg/) + 成績計算:分數/100直接加在總成績。 + 上學期10、11、12月,下學期3、4、5月,最後一週。 + 星期一08:00 ~ 星期五20:00。90分鐘。 + [證書](https://drive.google.com/open?id=1DcgDKn1R33kOPc_HazdIyQDgyYxSCPlZ) ### 3. [APCS](https://apcs.csie.ntnu.edu.tw/) + 實作級分*2,直接加在總成績。 + 4級分免上機考,上機考成績為100。 + 5級分以上者,一切皆免,學期成績直接算100。 ### 4. 學習歷程檔案(修課記錄、課程學習成果) + 最後一週上課結束前如果認證完成,學期總成績加分。 + [撰寫「課程學習成果」參考資源](https://hackmd.io/@cube/HkCv90e19)。 ### 5. 基礎教學網站 + [Teach Python 3 and web design with 200+ exercises Learn Python 3 - Snakify](https://snakify.org/en/) + [tutorialpoint](https://www.tutorialspoint.com/python/index.htm) + [w3cschool](https://www.w3schools.com/python/) ### 6. Python APCS(中正大學 吳邦一教授) + [PythAPCS123講義](https://drive.google.com/drive/folders/1mnVdO2LHq7e4vesn6pt_R0-S6YWtz4Q4?fbclid=IwAR2Xo7LP8V3lWyhJEbFx3F8JLF9AIEI9t9snla9vwwtI4qlGc0mDwJVuf1o)、[PythAPCS123 YouTube撥放清單](https://youtube.com/playlist?list=PLpmg1QLbgMuSIDOgOcwf0Fbbn2ZDR7s-X ) + [AP325(Python)](https://hackmd.io/@bangyewu/Hy2kbYLI6/%2Fg2kqHh5_Q4eQnz-mfNu3Kw)、[範例程式](https://drive.google.com/drive/folders/1JziPJ39tANRokjx6nFzZSRXtym0p5_kg?fbclid=IwAR2Xo7LP8V3lWyhJEbFx3F8JLF9AIEI9t9snla9vwwtI4qlGc0mDwJVuf1o) + [PyAP45_v202201題解講義](https://drive.google.com/drive/u/0/folders/10hZCMHH0YgsfguVZCHU7EYiG8qJE5f-m)、[PyApcs45 YouTube撥放清單](https://www.youtube.com/playlist?list=PLpmg1QLbgMuRQXHRkX9iDHyAVIW1D6OJF ) ### 7. 軟體 + [Google Colaboratory](https://colab.research.google.com/) - [Google Colab相關設定](https://hackmd.io/@wiimax/HJuUPnPQr) + [The collaborative browser based IDE - Replit](https://replit.com/) - 「+ Create Repl」 可新增一個 Repl (專案) - Template 欄選擇 「C++」 或 「Python (with Prybar)」,Title 欄填入專案標題如 a001, 再按「+ Create Repl」鈕,就會進入專案編輯的頁面。 - 如果 Template 沒有看到 Python (with Prybar) ,請自行輸入 prybar 後按「Search community templates」後點找到的「Python (with Prybar)」,按「+ Use Template」/「Use Template」,回首頁重新整理後,「Python (with Prybar)」即會出現在 「Favorites」項目下。 - Repls:專案管理 New folder:建立分類資料夾,如 ch1、ch2 …。 專案.../Move:將專案移至分類資料夾。 - 側邊欄/Tools/User Settings/Indentation Size:4 - 側邊欄/Tools/User Settings/ser Settings/AI Code Completion:關閉 - 側邊欄/Tools/User Settings/Themes:Explore Themes/VS Code Dark/Use Theme + [GDB online Debugger | Compiler - Code, Compile, Run, Debug online C, C++](https://www.onlinegdb.com/) + [Online Python](https://www.online-python.com/) + [Thonny](https://thonny.org/) (安裝套件: Tools/Manage plug-ins) + [Python IDLE 完整安裝教學](https://simplelearn.tw/2022/07/30/python-%E4%BB%8B%E7%B4%B9%E5%8F%8A%E5%AE%89%E8%A3%9D%E6%95%99%E5%AD%B8-2022%E6%9B%B4%E6%96%B0%E7%89%88/) ## 一、輸出與輸入、變數、運算式與運算子 ### 1. 輸出與輸入 + 變數=input(\[提示字串\]) + print(項目1\[,項目2,...\]) - 字串要加 '&nbsp;&nbsp;' 或 "&nbsp;&nbsp;" - 可用 + 將字串連接 - print() 輸出後,預設會自動補上換行符號。 - 印出多個項目,預設以一個空白間隔。 ``` python print('Hello Python') # 印出一段文字 print('Hello', 'Python', 123) # 印出多個項目,預設以一個空白間隔 print('Hello', 'Python', 123, sep=',') # 以『,』為項目間的分隔符號 ``` + print('A=%d B=%d C=%d' %(A, B, C)) - [Python 3 print 函数用法总结](http://www.runoob.com/w3cnote/python3-print-func-b.html) + print('A={} B={} C={}'.format(A, B, C)) - {} 的位置會被後方 format 所帶的參數所取代,預設會依照順序輸出。 - [Python Format String 字串格式化整理](https://blog.jaycetyle.com/2018/01/python-format-string/) + print(f'A={A} B={B} C={C}') - [f-string 是 Python 3.6 才有的方法](https://myapollo.com.tw/blog/python-f-string-formating-syntax/#google_vignette) :::info EX_1_1:[d483: hello, world](https://zerojudge.tw/ShowProblem?problemid=d483) + Colab 快速鍵 - `Ctrl+Enter`:執行目前程式 - `Ctrl+a`:全選 - `Ctrl+c`:複製 - `Ctrl+v`:貼上 - `Ctrl+x`:剪下 - `Ctrl+s`:儲存 - `Ctrl+z`:復原 - `Ctrl+滾輪`:放大、縮小 - `Shift`+方向鍵、`Home`、`End`:選取程式 + Replit 快速鍵 `Ctrl+Shift+v`:Replit 測資貼上。或右鍵/貼上。 + Thonny 快速鍵 `F5`:執行目前程式 ::: :::info EX_1_2:[a001: 哈囉](https://zerojudge.tw/ShowProblem?problemid=a001) ``` python s=input('please input a string:') # 輸出提示訊息後等使用者輸入,但 OJ 不可輸出規定以外的東西。 s=input() print('hello,',s) ``` ::: ### 2. 變數與資料型態 + 變數:存放資料的記憶體空間(名稱自訂) + 數值型態:int(整數)、float(浮點數、小數)、bool(True、False) + 字串型態:'&nbsp;&nbsp;' 或 "&nbsp;&nbsp;" 中的文字 + 容器型態: - list:有順序、可變動的資料集合 - tuple:有順序、不可變動的資料集合 - set:無順序的資料集合 - dict:鍵值對(key-value)的集合 + 等號的意義是『賦值』(assignment),不是『等於』,把等號右邊計算完的值指定給等號左邊的變數(左邊一定是變數(記憶體位置))。 - 錯誤:a+5=7 - a,b=1,2 + 資料型態轉換:int()、float()、str() - int(): 字串或浮點數轉整數 - str(): 數字轉字串 - float(): 字串轉浮點數 :::info EX_1_3:[d049: 中華民國萬歲!](https://zerojudge.tw/ShowProblem?problemid=d049) ``` python s=input() # 讀入的為字串 y=int(s) # 將字串轉換為整數,整數才能做算數運算加減乘除 y=int(input()) # 上兩行整合 ``` ::: :::info EX_1_4:[a002: 簡易加法](https://zerojudge.tw/ShowProblem?problemid=a002) + 需要同一行輸入兩個整數(空白間隔)時,可用 str.split() 將 str 字串分割成數個子字串(以空白區分) ``` python s=input().split() # s=['1','2']。s[0]、s[1] a=s[0] b=s[1] print(a+b) # Q:輸入「1 2」,會印出? ``` + map(某函式名, list) 會把某函式依次作用到list的每個元素上 ``` python s=input().split() # s=['1','2']。s[0]、s[1] a=int(s[0]) b=int(s[1]) a,b=input().split() # a='1', b='2' c=int(a) # c=1 d=int(b) # d=2 a,b=input().split() # a='1', b='2' a=int(a) # a=1。Python 支持動態類型,變數的類型可以在運行時改變,C++不行。 b=int(b) # b=2 a,b=map(int, input().split()) ``` ::: :::info EX_1_5:[d489: 伏林的三角地](https://zerojudge.tw/ShowProblem?problemid=d489) + 以[海龍公式](https://zh.wikipedia.org/zh-tw/%E6%B5%B7%E4%BC%A6%E5%85%AC%E5%BC%8F)求三角形面積。 + 「*」不可省略。 + 除錯第一步,檢查變數的值。 + 先乘除後加減,可用小括號改變運算順序。 + 運算式裏不能用中、大括號,通通都用小括號。2 * [3 + (4 + 5) * 6] → 2 * (3 + (4 + 5) * 6) + 浮點數與整數運算後會變成浮點數,整數做『/』運算後也會變成浮點數。在OJ上,答案36.0與36是不一樣的,要注意輸入、輸出的說明。 + 明明程式都對,但出現錯誤。 - 之前程式碼錯誤造成,如print=(int(area))。如果程式碼已沒有錯誤,『執行階段/重新啟動工作階段』。 ``` python TypeError Traceback (most recent call last) <ipython-input-21-678cc0bd0ae2> in <cell line: 4>() 2 s=(a+b+c)/2 3 area=s*(s-a)*(s-b)*(s-c) ----> 4 print(int(area)) TypeError: 'int' object is not callable ``` ::: :::warning [a861: 1. Secure the Perimeter](https://zerojudge.tw/ShowProblem?problemid=a861) + 「*」不可省略。 + 先乘除後加減,可用小括號改變運算順序。 + 運算式裏不能有中括號、大括號。2 * [3 + (4 + 5) * 6] → 2 * (3 + (4 + 5) * 6) + EOF 結尾的測資 ``` python while True: try: ~ ~ except: break ``` + Python 以冒號「:」及縮排來表示程式區塊。 - if、else、elif、for、while 等為會使用到程式區塊的關鍵字,在其「:」之後的下一行的就必須縮排。 - 以`Tab`鍵縮排,預設為2個空格。Colab縮排寬度調為4。 - `Tab、Shift+Tab`:增、減縮排 ::: :::warning [d461: 班際籃球賽](https://zerojudge.tw/ShowProblem?problemid=d461) ::: :::warning [d053: 10970 - Big Chocolate](https://zerojudge.tw/ShowProblem?problemid=d053) + EOF 結尾的測資 ``` python while True: try: ~ ~ except: break ``` ::: ### 3. 算數運算子 | 算數運算子 | 意義 | 備註| |:-------- |:------ |:-------| | + | 加 | | | - | 減 | | | * | 乘 | | | / | 除 |17/5 為 3.4,15/5 為 3.0 | | // | 取商數 |17//5 為 3</br> 17.0//5 為3.0 | | % | 取餘數 |17%5 為 2 | | ** | 次方 | 5**2 為 25 | :::info EX_1_6:[d050: 妳那裡現在幾點了?](https://zerojudge.tw/ShowProblem?problemid=d050) + 取餘數:% + [常用運算子的優先順序](https://sites.google.com/site/jingprogrampy/python/%E9%81%8B%E7%AE%97%E5%AD%90%E9%81%8B%E7%AE%97%E5%BC%8F),以海龍公式解釋。 ::: :::info EX_1_7:[d485: 我愛偶數](https://zerojudge.tw/ShowProblem?problemid=d485) + 將 a 調整為後一個偶數, b 為前一個偶數,[計算等差數列項數](https://zh.wikihow.com/%E8%AE%A1%E7%AE%97%E7%AD%89%E5%B7%AE%E6%95%B0%E5%88%97%E4%B8%AD%E7%9A%84%E9%A1%B9%E6%95%B0)。 + a 偶數 a=a+0,a 奇數 a=a+1 - a=a+a%2 + 複合指定運算子(a=a+5 → a+=5) + 整數除法:// ::: :::info EX_1_8:[e343: BMI 計算](https://zerojudge.tw/ShowProblem?problemid=e343) + 輸入分兩行讀入。 + 字串轉浮點數。 + 次方:** + 小數點位數控制 - 參考解題報告:round。 試著印出bmi=22.7 print('bmi=', round(bmi,1), sep='') - 參考 一、1. 輸出與輸入 print(f'bmi={bmi:.1f}') print(f'{bmi:.1f}') ::: :::warning [d073: 分組報告](https://zerojudge.tw/ShowProblem?problemid=d073) + 整數除法:// ::: :::warning [c379: 成為出題者](https://zerojudge.tw/ShowProblem?problemid=c379) + 整數除法:// ::: :::warning [d827: 買鉛筆](https://zerojudge.tw/ShowProblem?problemid=d827) + 整數除法:// + 取餘數:% ::: :::warning [b757: 頸美椰子樹](https://zerojudge.tw/ShowProblem?problemid=b757) + 除:/ + print(f'{變數名:g}') 根據數值大小採用科學符號 {變數名:e} 或浮點數 {變數名:f}。 + {變數名:g} 如果小數點為0,不會輸出。 ::: :::warning [a862: 2. My Dear Friend VIR](https://zerojudge.tw/ShowProblem?problemid=a862) + 除:/ + v,r=map(float,input().split()) + 小數點位數控制。 + EOF 結尾的測資。 ``` python while True: try: ~ ~ except: break ``` ::: :::warning [f651: 開關燈](https://zerojudge.tw/ShowProblem?problemid=f651) + 整數除法:// + 觀察n=1~10,找規律。類似d073。 + EOF 結尾的測資。 ``` python while True: try: ~ ~ except: break ``` ::: ### 4. 關係運算子 | 關係運算子 | 意義 | 備註| |:-------- |:------ |:-------| | == | 等於 | =是賦值,==是判斷是否相等 | | != | 不等於 || | > | 大於 || | >= | 大於等於 || | < | 小於 || | <= | 小於等於 || :mega: python 可以用 1 < x < 9 ,其它語言不行,該如何? 1<x and x<9 ### 5. 邏輯運算子 | 邏輯運算子 | 意義 | 備註| |:-------- |:---- |:---| | and | 而且 || | or | 或者 || | not | 否 || ## 二、選擇 ### 1. if 條件式語法 ``` python if 條件: 條件「成立」時的程式碼 ``` :mega: if 後要加「:」 :mega: python 沒有如 C++ 的大括號 { } 來表示程式區塊,沒縮排、縮排錯誤都會有問題。 :::info EX_2_1:[a012: 10055 - Hashmat the Brave Warrior](https://zerojudge.tw/ShowProblem?problemid=a012) + Q:以下的程式會發生什麼問題? ``` python if ans<0: ans=-ans print(ans) ``` ::: :::warning [a799: 正值國](https://zerojudge.tw/ShowProblem?problemid=a799) ::: :::warning [d068: 該減肥了!](https://zerojudge.tw/ShowProblem?problemid=d068) ::: :::warning [o578. 起司 (Cheese)](https://zerojudge.tw/ShowProblem?problemid=o578) + 先把答案的初值設為 0。 + 如果 L、W、H 除以 K 的餘數都為 0,則答案改為 (L//K)\*(W//K)\*(H//K) ::: :::warning [b682: 2. 同學早安](https://zerojudge.tw/ShowProblem?problemid=b682) + [提示](https://zerojudge.tw/ShowThread?postid=16450&reply=0) + 將時間全部轉成分鐘會比較好做,分鐘的時間差以//、%轉回小時、分。 + 因為校長最久只會站23小時59分,所以如果開始時間大於結束時間,表示站隔夜,結束時間要多加1440(24小時)。 ::: ### 2. if~else 語法 ``` python if 條件: 條件「成立」時的程式碼 else: 條件「不成立」時的程式碼 ``` :mega: else:後面不要寫條件。 :::info EX_2_2:[d064: ㄑㄧˊ 數?](https://zerojudge.tw/ShowProblem?problemid=d064) ::: :::info EX_2_3:[d066: 上學去吧!](https://zerojudge.tw/ShowProblem?problemid=d066) + if h>=7 and m>=30 and h<17: → 幾點會造成錯誤? + if (8<=h<17) or (7點的狀況): + and、or + [常用運算子的優先順序](https://sites.google.com/site/jingprogrampy/python/%E9%81%8B%E7%AE%97%E5%AD%90%E9%81%8B%E7%AE%97%E5%BC%8F) ::: :::warning [b877: 我是電視迷](https://zerojudge.tw/ShowProblem?problemid=b877) ::: :::warning [f373: 週年慶 Anniversary](https://zerojudge.tw/ShowProblem?problemid=f373) + 分別計算兩百貨折扣後的金額,比較大小輸出。 ::: :mega: 如果測資的第一行有一個整數t,代表測試的筆數,要用以下的寫法。 ``` python t=int(input()) for _ in range(t): ~ # 程式碼 ``` :::warning [e948: 1. 基礎代謝率 (BMR Calculation)](https://zerojudge.tw/ShowProblem?problemid=e948) + 輸入的第一行有一個整數n,代表測試筆數。 ::: :::warning [f043: 1. 小豪的回家作業 (Homework)](https://zerojudge.tw/ShowProblem?problemid=f043) + 給定兩整數R、A,求出A+B=R的B值後輸出。 + 如果R=A,則先將A-3再求B。 + A、B小的寫在算式的前面。 ::: ### 3. 巢狀 if 語法 ``` python if 條件1: if 條件2: 條件1「成立」,且條件2「成立」時的程式碼 else: 條件1「成立」,且條件2「不成立」時的程式碼 else: 條件1「不成立」時程式碼 ``` :mega: if 或 else 的程式區塊內,都可再增加條件式。 :::info EX_2_4:[d058: BASIC 的 SGN 函數](https://zerojudge.tw/ShowProblem?problemid=d058) + if n>0: vs. if n>=0: ::: :::info EX_2_5:[a006: 一元二次方程式](https://zerojudge.tw/ShowProblem?problemid=a006) + 開根號可用**0.5 + Q:d>0時,先試著自行寫出x1,x2公式。 - 需除錯時,Colab 可+程式碼區塊,詢問變數內容。 - a=2, b=3, c=1, x1=-0.5, x2=-1 + d==0時,還是要有x1的計算過程。執行時,不會去執行不符合條件的程式碼。 a=2, b=4, c=2, x1=-1 + [Python f-string 各種格式使用方法](https://myapollo.com.tw/blog/python-f-string-formating-syntax/#google_vignette) + 練習用 Thonny 「除錯目前程式」看條件式執行流程。 + 註解 - 單行: \# - 多行:「'''」開頭(三個單引號),「'''」結尾。 在 Colab 中,可選取多行,再按 `ctrl + /` 完成多個單行註解。 ::: :mega: 逐行除錯 + Thonny (1) 執行目前程式(F5)後發現錯誤。 (2) 點行號設定中斷點(可跳過沒問題的程式)。 (3) 按「![截圖 2024-09-05 下午2.36.26](https://hackmd.io/_uploads/SJNq-0I3A.png =20x20) 為目前腳本除錯」,準備逐行執行。 (4) 檢視/變數,開啟變數面板,可以看到執行每一步的變數變化。 (5) 逐行執行   跳過(Step Over)(F6):黃色區塊一次執行(big step)。   跳入(Step into)(F7):詳細執行每個小步驟(small step)。 (6) 發現錯誤後按『STOP』停止除錯,修正程式。 + Google Colab (1) 執行目前程式(Ctrl+Enter)後發現錯誤。 (2) 於有問題程式碼前加入breakpoint() (執行到中斷點,可跳過沒問題的程式)。 (3) Pdb 指令 | 指令 | 功能 | | ----| ----- | | n(next) | 執行當前行,並移動到下一行,不進入函數內部| | s(step) | 進入函數內部逐步執行| | c(continue) | 繼續執行到下一個中斷點或程式結束| | p <變數名> | 印出變數的值。例如p score 會印出 score 變數的值| | q(quit) | 離開 pdb| | h(help) | 指令說明| + [Replit](https://docs.replit.com/programming-ide/workspace-features/debugging) (1) 執行目前程式(Ctrl+Enter)後發現錯誤。 (2) 打開除錯視窗(Debugger),並將其拖曳至適當位子(不要和Console在同一窗格)。 (3) 於行號前設定中斷點(可跳過沒問題的程式)。 (4) 按除錯視窗的『RUN』。 (5) 使用『Next Step』或『Skip Step』或『Next Breakpoint』逐行執行,觀察流程和變數的變化。 (6) 發現錯誤按『Stop』停止除錯,修正程式。 :::warning [a003: 兩光法師占卜術](https://zerojudge.tw/ShowProblem?problemid=a003) ::: :::warning [d065: 三人行必有我師](https://zerojudge.tw/ShowProblem?problemid=d065) ::: :::warning [f165: 棒棒糖事件](https://zerojudge.tw/ShowProblem?problemid=f165) + 判斷棒棒糖數除以小朋友人數的餘數r是否為0,當r=0時,輸出「OK!」,反之輸出r。 + 小朋友人數可能為0,不會搶成一團,所以蝸牛老師不需吃下任何棒棒糖,輸出「OK!」。 + 先排除小朋友人數為0的狀況,不然「%」運算會錯誤。 ::: :::warning [b899: 2. 物品探測](https://zerojudge.tw/ShowProblem?problemid=b899) + 分別計算三點間的距離,最長的為正方形的對角線。 + [正方形四點 A、B、C、D,假設 A、C 為對角,則 D = A + C - B。](https://zerojudge.tw/ShowThread?postid=16940&reply=13550#16940) ::: ### 4. 多重選擇 if\~elif\~else 語法 ``` python if 條件1: 條件1「成立」時程式碼區塊 elif 條件2: 條件1「不成立」,且條件2「成立」時的程式碼 elif 條件3: 條件1、2「不成立」,且條件3「成立」時的程式碼 ... else: 上述條件都「不成立」時的程式碼 ``` :mega: else if()可視需要增加。 :::info EX_2_6:[d460: 山六九之旅](https://zerojudge.tw/ShowProblem?problemid=d460) + elif 6<=a<=11: v.s. elif a <= 11: - C++ 需以 a>=6 and a<=11 的概念來寫 + 單用 if,不用 elif 可以嗎? ``` python if a<=5: print(0) if a<=11: print(590) ``` ::: :::warning [f337: 同樂會 (Party)](https://zerojudge.tw/ShowProblem?problemid=f337) ::: :::warning [b676: 63萬勞工苦輪班不像人像機器](https://zerojudge.tw/ShowProblem?problemid=b676) ::: :::warning [a053: Sagit's 計分程式](https://zerojudge.tw/ShowProblem?problemid=a053) ::: :::warning [c382: 加減乘除](https://zerojudge.tw/ShowProblem?problemid=c382) + 先不要使用 eval(),練習多重選擇。 + 輸入使用split()先分割就好,要計算時再轉成int。 ::: ## 三、迴圈 ### 1. range 函式 + 串列變數=range(整數) - 產生0~「整數-1」的串列 + 串列變數=range(起始值, 終止值) - 產生起始值~「終止值-1」的串列 - 左閉右開區間 + 串列變數=range(起始值, 終止值, 間隔值) - 產生:起始值, 起始值+間隔值, 起始值+間隔值*2, ... + [Python range() 函数用法](http://www.runoob.com/python/python-func-range.html) ### 2. for 迴圈語法 ``` python for 變數 in range(重複次數): 程式碼 ``` ``` python for 變數 in 串列: 程式碼 ``` ``` python for 變數 in 字典: # 變數會儲存字典的key 程式碼 ``` ``` python break:強制跳出「整個」迴圈 continue:強制跳出「此次」 迴圈,繼續進入下一次圈 ``` ``` python for 變數 in 串列或字串: 程式碼 else: 迴圈正常結束,執行此區塊程式碼。(break不會執行) ``` :::info EX_3_1:[d498: 我不說髒話](https://zerojudge.tw/ShowProblem?problemid=d498) + 迴圈變數名:i 或 _ + [字串中輸出單引號](https://blog.csdn.net/menghuanshen/article/details/104289192) + 練習用 Thonny 「除錯目前程式」看迴圈執行流程。 + range 3 種參數寫法。 ::: :::info EX_3_2:[d491. 我也愛偶數 (swap 版)](https://zerojudge.tw/ShowProblem?problemid=d491) + Q:先完成 a+...+b,會有錯誤? A:sum 初值要歸零。雖然python變數不需要事先宣告,但它沒厲害到可以沒初值,就直接放在表達式中去進行運算(沒有賦值會不知道該變數是什麼類型,預設為'內建函數或方法',因而無法判斷能否進行運算)。 + 課堂加分:自行完成加偶數的部份。 + python 兩變數交換:a,b=b,a + C不能這樣寫 ``` c t=a; a=b; b=t; ``` ::: :::info EX_3_3:[h658: 捕魚 (Fishing)](https://zerojudge.tw/ShowProblem?problemid=h658) + 先計算魚夫和每個魚群的距離,第2筆測資:7.07 2.23 3.16 - 開根號可用**0.5,d=((x-a)\**2+(y-b)\**2)**0.5 + 課堂加分:輸出最小值的距離。 - min_d(最小距離)的初值可假設為一個比可能最大值還要大的值(如1000)。 - 讀入每個魚群的中心座標(a,b),計算魚夫和魚群的距離,如果比目前的最小距離小,則記錄此時的最小距離和座標(min_a,min_b)。 + [蛇形命名法](https://zh.wikipedia.org/zh-tw/%E8%9B%87%E5%BD%A2%E5%91%BD%E5%90%8D%E6%B3%95) ::: :::warning [b970: 我不說髒話 (續)](https://zerojudge.tw/ShowProblem?problemid=b970) + range範圍 + 數字和『.』中間沒有空白。 + [Python f-string 各種格式使用方法](https://myapollo.com.tw/blog/python-f-string-formating-syntax/#google_vignette) ::: :::warning [a244: 新手訓練 ~ for + if](https://zerojudge.tw/ShowProblem?problemid=a244) ::: :::warning [f338: 后羿射日(Archer)](https://zerojudge.tw/ShowProblem?problemid=f338) + 距離用 R**2,不需開根號。 ::: :::warning [f605: 1. 購買力](https://zerojudge.tw/ShowProblem?problemid=f605) + 找出3物品最大值和最小值,可使用max、min函式。 ::: :::warning [f312: 1.人力分配](https://zerojudge.tw/ShowProblem?problemid=f312) + 以for迴圈枚舉所有可能的工人分配方法。 + 求最大值可用max函式或if。 ::: ### 3. list 簡介 + list(串列) 把很多元素串起來,依照排列的順序給予編號 0, 1, 2,... + 同一行輸入元素個數不定時,需以 list 處理。 ``` python= a=[3,5,7,1,2] print(a) print(a[2]) # 第四章會再詳細介紹 slice for x in a: # 直接遍歷(走訪) a 串列 print(x) # 每印一個元素換行 for i in range(5): # 以索引值存取 print(a[i]) for x in a: print(x, end=' ') # 每行以空白結尾 print(*a) # 印出每個元素。「*」可將序列分解成多個獨立的元素(不在賦值語句中使用*號) print(*a, sep=',') # 可以更改分隔符號 Q:如何用 for 做到? 直接 print(n, end=',') 最後會多出一個「,」 for x in a: print(x, end=',') # 每行以「,」結尾 Hint:以索引值的方式,分段處理(先印出前4個元素以「,」結尾,再印出最後一個元素)。 ``` <font color="#fff"> for i in range(len(a)-1):</br>   print(a[i], end=',')</br> print(a[len(a)-1]) </font> + list常用的計算函式 | 串列函式 | 意義 | |:------------ |:-------------- | | lst=list() 或 lst=[] |宣告空的串列| | len(lst) | 回傳串列個數 | | max(lst) | 回傳串列中的最大值 | | min(lst) | 回傳串列中的最小值 | | sum(lst) | 將串列元素加總 | :::info EX_3_4:[j178: 手遊廣告 (Advertisement)](https://zerojudge.tw/ShowProblem?problemid=j178) + 同一行輸入兩個整數(空白間隔) - m, a = map(int, input().split()) + 同一行的多個元素轉成 list (元素個數可不定) - val = [*map(int,input().split())] - val = list(map(int,input().split())) - (四.1再說明)val = [int(x) for x in input().split()] # 使用列表生成式(list comprehension),將讀進來的字串轉成整數後放入一個list。 + Q:如果程式寫成 ``` python for x in val: if a>x: a+=x ``` 以下測資的正確答案為25,但會輸出? 4 10 5 10 30 10 + 迴圈中斷 - break:強制跳出「整個」迴圈。 - continue:強制跳出「此次」 迴圈,繼續進入下一次圈。 ``` python for i in range(10): if i==5: break # continue print(i) ``` ::: :::info EX_3_5:[f708: 蟲蟲危機 (Insect)](https://zerojudge.tw/ShowProblem?problemid=f708) + 串列元素加總 ``` python val=[1,2,3] sum=0 for x in val: sum+=x print(sum) ``` + sum(lst) 將串列元素加總 ::: :::warning [d046: 文文採西瓜](https://zerojudge.tw/ShowProblem?problemid=d046) ``` python watermelon = [*map(int,input().split())] watermelon = list(map(int,input().split())) watermelon = [int(x) for x in input().split()] ``` ::: :::warning [b138: NOIP2005 1.陶陶摘苹果](https://zerojudge.tw/ShowProblem?problemid=b138) ::: ### 4. 巢狀迴圈語法 + Q:在日常生活中,有什麼事像巢狀迴圈的概念? ``` python for i in range(外圈次數): for j in range(內圈次數): 程式碼 ``` + 九九乘法表 ``` python for i in range(1, 10): for j in range(1, 10): print(f'{i}x{j}={i*j}', end=' ') # end=' ' 表示每行以空白結尾 print() # 內層迴圈執行結束後才換行 ``` :::info EX_3_6:[e621. 1. 免費停車 (Free Parking)](https://zerojudge.tw/ShowProblem?problemid=e621) + 先完成一天的狀態。 + 免費停車位以空白隔開,輸出一筆之後才以print()換行。 + 布林型態(True、False):free = False + not ::: :::info EX_3_7:[d660: 11764 - Jumping Mario](https://zerojudge.tw/ShowProblem?problemid=d660) + w = [*map(int,input().split())] + 用 now 表示目前高度( 初值為 wall[0] ),每次和右牆比較完,調整其為右牆的高度。 - 以索引值存取較麻煩 ``` python for i in range(1,n): if now < w[i]: ~ else: ~ ``` - 直接存取串列 ``` python for x in w[1:n]: # w[1:]索引值留白表示取到最後一個 if now < x: ~ else: ~ ``` + 先用if else 且漏調目前高度,練習除錯。(if else 說明 F7 慢慢按,觀察變數變化) ::: :::warning [d786: 三、平均值](https://zerojudge.tw/ShowProblem?problemid=d786) + 每筆測資sum要歸零。 + 每筆輸入 num=[*map(int,input().split())] + n=num[0]為數列長度。 + 加總num[1]~num[n]後平均。 - for x in num[1:]: - print(f'{sum/n:.2f}') ::: :::warning [c005: 10300 - Ecological Premium](https://zerojudge.tw/ShowProblem?problemid=c005) + 每筆測資獎金要歸零。 + 運算式可先化簡避免除法誤差。 ::: :::warning [c013: 00488-Triangle Wave](https://zerojudge.tw/ShowProblem?problemid=c013) + _ = input() # 第一列和測資間有一空白行要讀掉 + 下半部 for j in range(A-1, 0, -1): + 印出數字不換行 print(j, end='') + 換行 print() ::: ### 5. while 迴圈語法 ``` python while 條件判斷: # 條件為「真」的時候繼續執行 程式碼 ``` :::info EX_3_8:[d189: 11150 - Cola](https://zerojudge.tw/ShowProblem?problemid=d189) + 請用 while 模擬換瓶過程,不可以直接套公式。 + 先寫 n>3 以除錯看換瓶過程。最後剩 2 空瓶,可借 1 空瓶多換 1 瓶。 + 整數除法:// ``` python= sum = ~ # sum 表示喝的可樂,一開始可以喝幾瓶? n 可以表示空瓶 while ~: # 還有 3 個以上的空瓶 ~ # 更新喝的可樂數量 (加上利用空瓶換的可樂) ~ # 更新空瓶數量 (新換 + 剛換剩的) ~ # 最後剩 2 空瓶時 ``` ::: :::warning [c079: 10346 - Peter's Smokes](https://zerojudge.tw/ShowProblem?problemid=c079) + 類似 d189 ::: :::warning [d570: 神龍見首不見尾](https://zerojudge.tw/ShowProblem?problemid=d570) ::: :::warning [d356: NOIP2002 1.级数求和](https://zerojudge.tw/ShowProblem?problemid=d356) ::: :::warning [f070: 1. 韓信點兵 (HanXin)](https://zerojudge.tw/ShowProblem?problemid=f070) + 因為答案很小,暴力解即可。 + 以 while 迴圈,從1開始測,輸出符合條件的值。 ::: :::warning [c350: “綠白黃” 四校聯課](https://zerojudge.tw/ShowProblem?problemid=c350) + ans+=(n/k)*w # 每次新換的電話號碼 + n-=(n/k)*(k-w) # 因為 k > w,所以可以看成每一次的交換(k換w),新號碼會減少 (k-w) 個 ::: ## 四、複合式資料型態 ### 1. [list(串列)](https://www.twblogs.net/a/5cd7d860bd9eee67a77f8884) + 串列變數 = [ 元素1, 元素2, .... ] # 類似 C 的「陣列」。 + 串列b = 串列a,不同於變數,不是將 a 複製到 b,兩者會有相同的記憶體位置。c = a.copy() 才會將 a 複製給 c。 ``` python a=1 b=a print(a,b) b=5 print(a,b) a = [1, 2, 3] b = a print(a,b) b[2] = 5 print(a,b) c = a.copy() c[2]=7 print(a,c) ``` + 列表對「+」和「$*$」的操作與字串相似。「+」用予組合列表,「$*$」用於重複列表。 | 串列運算 | 範列 | |:-------- |:-------------- | | + |[1, 2]+[3, 4] 為 [1, 2, 3, 4]| | $*$ |[1, 2]\*3 為 [1, 2, 1, 2, 1, 2]| | x in lst |判斷x是否在串列, 3 in [1, 2, 3] 為 True| | == |判斷兩串列是否相等, [1, 2]==[1, 2, 3] 為 False| + 運用 [] 取值,可以使用 index(索引值從0開始)、slice 來存取陣列裡的資料。假設lst,lst2為串列變數,常用的運算和方法如下表 - 編號由0開始。 - 區間定義都是左閉右開(不含右端)。 - 負的編號是倒數,最後一個是-1。 - 第一個編號不寫代表開頭,第二個編號不寫表示一直到結尾都包含。 - 完整的 slice 包含 list[start, stop, step],step 表示每次增加幾個位置,step=-1 會反轉 list (start 要大於 stop)。 | slice運算 | 意義 | |:------------ |:-------------- | | lst[i] | 取出索引值i的元素 | | lst[-1] | 取出最後一個元素 | | lst[i:j] | 取出索引值 i ~ j-1 的元素 | | lst[i:j:k] | 取出索引值 i ~ j-1 的,間隔為 k 的元素 | | lst[i:j]=lst2| 把索引值 i ~ j-1 的元素換成 lst2 的元素 | | del lst[i:j] | 刪除索引值 i ~ j-1 的元素 | ``` python lst = [0, 1, 2, 3, 4, 5, 6] lst[0] lst[-1] lst[0:4] lst[0:4:2] lst[ :4:2] lst[ : :2] lst[ : :-1] # 串列反轉 lst[2: :-1] # 從 idx 2 開始反向走到頭 lst[1:4]=[7,7,7] lst[1:4]=[8]*3 del lst[1:4] ``` + 常用串列函式 | 串列函式 | 意義 | |:------------ |:-------------- | | lst=list() 或 lst=[] |宣告空的串列| | len(lst) | 回傳串列個數 | | max(lst) | 回傳串列中的最大值 | | min(lst) | 回傳串列中的最小值 | | sum(lst) | 將串列元素加總 | | list('abc') | 將'abc'拆成'a','b','c'加入串列 | | 串列方法(函式) | 意義 | |:---------------|:-------------- | | lst.append(x) | 將x附加到串列後面 | | lst.insert(i,x)| 將x插入到索引值i的位置 | | lst.extend(x) | 將串列x中的所有元素,附加到串列後面。<br />a=[1,2] b=[3,4] <br /> a.append(b) vs. a.extend(b)| | lst.remove(x) | 刪除串列中的第一個x | | lst.pop(i) | 回傳索引值i的元素,並將其刪除;<br/>如果沒有i,則傳回最後一個元素並刪除 | | lst.index(x) | 回傳第一次出現x的索引值 | | lst.count(x) | 計算出現x的次數 | | lst.sort() | 將串列中的元素小->大排序,大->小則加入參數reverse=True | | lst.reverse() | 將串列中的元素反轉 | | lst2=lst.copy()| copy串列 | | lst.clear() | 清除串列內所有元素 | ``` python lst = [0, 1, 2, 3, 4] lst[5] = 5 # IndexError: list assignment index out of range lst.append(5) n=lst.pop(0) ``` + [列表生成式(List Comprehensions)](https://www.liaoxuefeng.com/wiki/1016959663602400/1017317609699776), 類似數學集合的表示法。$\{ 2n+1 \mid n \in \{0, 1, 2, 3, 4\} \} \rightarrow \{1,3,5,7,9\}$ - 以說明 list 內容的方式建構 list,把 for 迴圈包含在 list 建構中,可以是巢狀迴圈,甚至加 if 來過濾。 ``` python a = [] for i in range(5): a.append(2*i+1) a = [2*i+1 for i in range(5)] b = [x for x in a if x%3==0] # [3, 9] val = [int(x) for x in input().split()] # EX_3_4 輸入,使用列表生成式(list comprehension),將讀進來的字串轉成整數後放入一個list。 # 作業 c290 ``` :::info EX_4_1:[g595. 1. 修補圍籬](https://zerojudge.tw/ShowProblem?problemid=g595) + 直接遍歷串列元素(for x in h:)可以嗎? 只能循序取出圍籬高度,無法取得前一個、後一個。 + 對於每一個 0,把它左右較小的數字取出加總。求2數的最小值可以用 min 函式。 + 先完成不管左右邊界是 0 的程式,for i in range(n): 讓第 1 筆測資對。IndexError: - Q:第 2 筆測資會出現 list index out of range,如何修改 range 讓第 2 筆測資答案為 4。(除錯說明) + 左右邊界如果是 0,要特別處理,因為左邊界只有右邊鄰居而右邊界只有左邊鄰居。右邊界可以用負的索引值寫,會更好理解。 ``` python if h[n-1]==0: sum+=h[n-2] if h[-1]==0: sum+=h[-2] ``` ::: :::info EX_4_2:[b558: 求數列第 n 項](https://zerojudge.tw/ShowProblem?problemid=b558) + [1-based indexing](https://xlinux.nist.gov/dads/HTML/oneBasedIndexing.html) ``` python a=[0,1] for i in range(2,501): a[i]=a[i-1]+(i-1) # list assignment index out of range 該如何改? 先把串列空間造出來。 a=[0]*501 a[1]=1 ``` + append() ::: :::info EX_4_3:[c299: 1. 連號或不連號](https://zerojudge.tw/ShowProblem?problemid=c299) + 刪除最前面的串列長度。 - slice 取值 n=a[0] s=a[1:] - pop 刪除 n=a[0] a.pop(0) # 或 del a[0] - pop 取值並刪除 n=a.pop(0) + 輸入數列時,找出其最大值和最小值,可使用 if 或 max、min 函式。 ``` python a=[2,1,4,3] mx=0 # 初值設一個極小值 for x in a: if x > mx: mx=x print(mx) ``` + 最大-最小+1==n,表示這個數列是連續的。 + 三元運算子 ``` python if a>b: ans=a else: ans=b ans=a if a>b else b ans='yes' if mx-mn+1==n else 'no' ``` ::: :::info EX_4_4:[b139: NOIP2005 2.校门外的树](https://zerojudge.tw/ShowProblem?problemid=b139) + 可以宣告一個整數串列tree(預設為1),索引值代表位置,值 0、1 表示有沒有樹。基本作法: ``` python tree=[] for _ in range(L+1): tree.append(1) ``` tree=[1]*(L+1) + 最後計算有多少個 1。基本作法: ``` python ans=0 for i in range(L+1): if tree[i]==1: ans+=1 ``` print(sum(tree)) + Bonus:以 slice 運算,將索引 b ~ e 間的值換成0。 lst[i:j]=lst2 # 把索引值 i ~ j-1 的元素換成 lst2 的元素 ::: <font color="#fff">tree[b:e+1]=[0]*(e-b+1)</font> :::info EX_4_5:[i071: 風景 (Landscape)](https://zerojudge.tw/ShowProblem?problemid=i071) + [1-based indexing](https://xlinux.nist.gov/dads/HTML/oneBasedIndexing.html) h=[*map(int,input().split())] h=[0]+h h=[0]+[*map(int,input().split())] + 以 slice 運算,從小明家往左、往右檢查>小明家樓高的數量。輸入範例 2 答案為 5,錯誤的原因為? ``` python for x in h[m+1:]: # 右半 h[m+1:n+1] if x > h[m]: ans+=1 for x in h[m-1::-1]: # 左半 if x > h[m]: ans+=1 ``` + 以變數(l_max、r_max)記錄小明家左邊、右邊最高樓高(初值為小明家樓高)。從小明家往左、往右檢查,如果檢查的樓高 > l_max、r_max,則答案+1,並調整 l_max、r_max。(就好像發現較高的樓,小明就搬過去) + 或往右發現有比小明家高的樓高,就讓小明家長高。(小明家高要先保留,往左檢查前要恢復原高) + 變數名儘量小寫, r_max 和 r_Max是不同的,且因為變數不用宣告,迴圈內如果打錯也不知道。 ::: :::info (進階)EX_4_6:[a693: 吞食天地](https://zerojudge.tw/ShowProblem?problemid=a693) + [1-based indexing](https://xlinux.nist.gov/dads/HTML/oneBasedIndexing.html) food=[0]+[*map(int,input().split())] + 先試每次都重算 ➝ TLE + [時間複雜度](https://jason-chen-1992.weebly.com/home/time-space-complexity) + [前綴和](https://www.youtube.com/watch?v=VfL7dzFkW30),但不要用爆力法。 ``` python for i in range(1,n+1): # 對每個idx for j in range(1,i+1): # 從頭開始加到此idx pfx[i]+=food[j] ``` + 1-based indexing 也可避免算前綴和 i=0 時,pfx[i]=pfx[i-1]+food[i] 會有 pfx[-1] 的錯誤。 ::: :::warning [b127: 會議中心(Room)](https://zerojudge.tw/ShowProblem?problemid=b127) + [費式數列](https://zh.wikipedia.org/wiki/%E6%96%90%E6%B3%A2%E9%82%A3%E5%A5%91%E6%95%B0%E5%88%97) + [0]*50 ::: :::warning [f818: 物競天擇 (Survival)](https://zerojudge.tw/ShowProblem?problemid=f818) + 設一個變數存身高*體重的最小值,初值為1000005。 + 進階可以使用 zip - [python多個變數的for迴圈](https://www.itread01.com/content/1544266142.html) - [Python zip()函數](http://www.runoob.com/python3/python3-func-zip.html) - for h,w in zip(height, weight) ::: :::warning [f063: The Strongest Chain](https://zerojudge.tw/ShowProblem?problemid=f063) + 求每條鏈子最小值中,找出最大值。 + max()、min() + [] slice取值 + abs() 取絕對值 ::: :::warning [c290. APCS 2017-0304-1秘密差](https://zerojudge.tw/ShowProblem?problemid=c290) + line = [int(x) for x in input()] # 把每一個數字字元轉換成整數後存入list + [] slice取值 ::: :::warning [n631. 撲克 (Poker)](https://zerojudge.tw/ShowProblem?problemid=n631) + 因為牌只有52張,因此可以開一個大小為52的陣列(初始值皆為0),陣列的索引值代表牌的編號,值為牌的數量。poker=[0]*52 + 讀入每個牌的編號,統計每張牌的數量(放在編號-1的位子) + 在不增加任何牌的情況下,以目前的牌最多能湊出幾副完整牌組:poker串列裏的最小值。 + 若要用上所有現有的牌去湊出完整牌組,最少還需要補幾張牌:?(和串列最大值有關) + [解題參考](https://hackmd.io/@apcser/r16Vuk5ma?utm_source=preview-mode&utm_medium=rec#n631-%E6%92%B2%E5%85%8B-Poker) ::: :::warning [b374: [福州19中]众数](https://zerojudge.tw/ShowProblem?problemid=b374) + [想法參考](https://zerojudge.tw/ShowThread?postid=16300&reply=0) ::: :::warning [c199: 爬山去(Hiking)-TOI練習賽y7m5-1](https://zerojudge.tw/ShowProblem?problemid=c199) + 輸入時可先將重覆的數字去掉。 ::: :::warning [g308: pB. 跳跳布朗尼(Brownie)](https://zerojudge.tw/ShowProblem?problemid=g308) + 使用布林陣列來記錄每個格子是否普經走過。 + 亦可直接使用記錄「傳送點資訊」的陣列,走過就設為「-1」。 ::: :::warning [n686. pA. 訊號傳遞](https://zerojudge.tw/ShowProblem?problemid=n686) + 依序檢查每個基地台訊號能達到的範圍,更新能達到的最遠位置。 + [解題參考](https://hackmd.io/@cyk/2024-hgsh) ::: :::warning [e339: 前綴和練習](https://zerojudge.tw/ShowProblem?problemid=e339) + 小心不要出現pfx[-1]的狀況。 ::: ### 2. list of list + list 中元素可以是不同型態,也可以放list。所以二維矩陣由 list of list 實作。 ``` python lst = [1, 'apple', [1,2,3]] a = [[1, 2, 3], [4, 5, 6]] a[0] # [1, 2, 3] a[1][2] # 6 for i in range(2): # 列 for j in range(3): # 欄 print(a[i][j], end=' ') print() # 不用索引值,直接遍歷的寫法 for row in a: # 遍歷每一列 for x in row: # 遍歷每一列中的每一個元素 print(x, end=' ') print() ``` | a[0][0] | a[0][1] | a[0][2] | | -------- | -------- | -------- | |**a[1][0]**|**a[1][1]**|**a[1][2]**| + 使用列表生成式產生2維以上串列 ``` python a = [[0 for i in range(3)] for j in range(2)] # a=[[0, 0, 0], [0, 0, 0]] a = [[0]*3 for i in range(2)] # 初始化一個 2*3 矩陣,初值均為 0 # 上一行不能簡寫為 b = [[0]*3]*2 # 試試 b[1][1] = 5 , 印出 b 會看到什麼錯誤? # 因為使用 * 來複製可變物件(如list、字典等)時,複製的是對同一個列表的引用,而不是獨立的副本。 # c = [[1, 2, 3]]*2,可更清楚看出其運作方式為 # c[0] = [1,2,3] # c[1] = c[0] # 所以 c[1][1] = 5 印出 c 會看到 [[1, 5, 3], [1, 5, 3]] ``` :::info EX_4_7:[e798: p5. 卷積神經網路](https://zerojudge.tw/ShowProblem?problemid=e798) + 多行輸入 ``` python # 如何把多行輸入,變成二維串列的結構? for i in range(n): a = [*map(int, input().split())] print(a) print(a[1][1]) ``` ``` python a = [] for _ in range(n): a.append([*map(int, input().split())]) # print(a) a = [[] for i in range(n)] for i in range(n): a[i] = [*map(int, input().split())] # print(a) a = [[*map(int, input().split())] for _ in range(n)] ``` + 求最大值可用 max 函式或 if。 + 小心不要存取到串列外的空間。 IndexError: list index out of range ::: :::info EX_4_8:[f513: 舉旗遊戲 (Flag)](https://zerojudge.tw/ShowProblem?problemid=f513) + m=[input().split() for _ in range( r )] # 不用特別用 map 轉成 int 也可以,串列裏的元素為 '1'、'2'。 + 如果自己的顏色和相鄰 8 點都不同,則淘汰人數+1。 + 相鄰點的座標 ![](https://hackmd.io/_uploads/BJ7fNfpNke.png =250x) + 直接檢查每個人相鄰的 8 個點,這樣會發生什麼問題? IndexError: list index out of range Q:要如何改? ``` python cnt=0 for i in range(r): for j in range(c): out=True # 先假設被淘汰 if m[i][j]==m[i-1][j-1]: # i>0 and j>0 and ... out=False if m[i][j]== m[i-1][j]: out=False if m[i][j]==m[i-1][j+1]: # i>0 and j<c-1 and ... out=False if m[i][j]==m[i][j-1]: out=False if m[i][j]==m[i][j+1]: out=False if m[i][j]==m[i+1][j-1]: out=False if m[i][j]==m[i+1][j]: out=False if m[i][j]==[i+1][j+1]: out=False if out: cnt+=1 ``` + 先定義相鄰點每一個方向列、欄座標跟 (i,j) 的差值,然後以 for 迴圈產生相鄰點的座標,並檢查是否出界。這種以位移的角度處理(列與行的變化量)的方式,是方格圖走訪常用的技巧。 dr=[-1, -1, -1, 0, 0, 1, 1 ,1] dc=~ nr=i+dr[] + (進階)for ~: else: ::: :::info (進階)EX_4_9:[a694: 吞食天地二](https://zerojudge.tw/ShowProblem?problemid=a694) + 前綴和 ![](https://hackmd.io/_uploads/r1vpzDlSn.png) **➜** ![](https://hackmd.io/_uploads/BJMRGvlr3.png) + [如何以a693的方式加速?](https://sites.google.com/a/mail.hpsh.tp.edu.tw/pc/jiao-cai/a694-tun-shi-tian-de-er) |[0][0] |[0][1]|[0][2] |[0][3] |[0][4]|[0][5] | |:----: |:----:|:----: |:----: |:----:|:----: | |**[1][0]**| | | | | | |**[2][0]**| |[r1-1][c1-1]| | |[r1-1][c2]| |**[3][0]**| | |[r1][c1]| | | |**[4][0]**| | | | | | |**[5][0]**| |[r2][c1-1] | | |[r2][c2] | ::: :::warning [o077. 2. 電子畫布](https://zerojudge.tw/ShowProblem?problemid=o077) + a=[[0]\*w for i in range(h)] # 初始化一個h*w矩陣,初值均為 0 + 遍歷陣列所有點,如果此點跟座標(r,c)距離在 t 內就把那點+=x + 印出陣列 ``` python for row in a: # 取出每一列 print(*row) # 印出此列的每個元素,以空白隔開 ``` ::: :::warning [b367: 翻轉世界](https://zerojudge.tw/ShowProblem?problemid=b367) + 假設 a180 為 a 轉 180 度後的陣列,a[i][j] 轉180度後會放在 a180[r-i-1][c-j-1]; + [題意轉180度的意思](https://zerojudge.tw/ShowThread?postid=21462&reply=9472#21462) ::: :::warning [f418: Word Search Puzzle](https://zerojudge.tw/ShowProblem?problemid=f418) + 不會有起始點在下,終點在上面的單字。 + 總共只有3種情況 - r1==r2 橫著印 - c1==c2 直著印 - 其它斜著印,從(r1+1,c1+1),(r1+2,c1+2)......印到(r1+(r2-r1-1),c1+(c2-c1-1)),(r2,c2) + [解題參考](https://www.youtube.com/watch?v=BDHhgndIP-s) ::: :::warning [e787: b2.尋寶地圖(Map)](https://zerojudge.tw/ShowProblem?problemid=e787) + 轉換圖計算行、列和時,重複算的要扣除。 + [題目](https://tpmso.org/toi/wp-content/uploads/question/201912/B2-Map.pdf) + [解題參考(1)](https://tpmso.org/toi/wp-content/uploads/question/201912/B2-Map.pptx)、[(2)](https://www.youtube.com/watch?v=lGYJUGmPzCk) + 兩張地圖資料之間有一個空白行。 ::: ## 五、字串 ### 1. 字串相關操作 + 與 list 操作類似,相加是串接,乘整數是重複。 + 字串與 list 一樣可以 slice,但字串內容不可更改。 ``` python s = 'Hello World' s[1] s[1] = 'x' # error a = s[0:5]*2 a = s[:5] + ' Python ' + s[6:] a = s[::-1] # 反轉字串 ``` + 字串與 list 都可以用 in 檢查元素(字元、子字串)是否在裏面。 ``` python s = 'Hello World' 'H' in s # True 'E' in s # False 'He' in s # True 'She' in s # False ``` ### 2. 字串(string)相關函式 + [Python 字串操作(string替換、刪除、擷取、複製、連線、比較、查詢、包含、大小寫轉換、分割等)](https://codertw.com/%E7%A8%8B%E5%BC%8F%E8%AA%9E%E8%A8%80/358746/) + [RUNOOB Python 字符串](http://www.runoob.com/python/python-strings.html) | 字串相關函式 | 意義 | |:--------------|:------ | | str(n) |將整數變數 n 轉為字串型態| | len(s) |計算字串 s 的長度| | s.lower() |將字串中的字母轉成小寫| | s.upper() |將字串中的字母轉成大寫| | s.islower() |判斷字串中的字母是否皆為小寫| | s.isupper() |判斷字串中的字母是否皆為大寫| | s.find('abc') |尋找字串中 'abc' 的位置| | s.count('abc')|計算字串中 'abc' 出現的次數| | s.replace(舊子字串,新子字串)|將字串中的舊子字串用新子字串取代| | s.split()|將字串分割成數個子字串(以空白區分)| | s.strip()|去掉字串左右空白| | s.join(sequence)| 指定字符(s)連接串列(sequence)中的元素後,生成新字串 | | 'ab' in 'abcd' | 判斷 'ab' 是否在 'abcd' | | s[::-1] | 將字串反轉 | | [] | 操作同 list | ``` python s = 'Hello' len(s) a = s + 5 # can only concatenate str (not "int") to str a = s + str(5) ``` :::info EX_5_1:[b428: 凱薩加密](https://zerojudge.tw/ShowProblem?problemid=b428) + ord(): 字元的 ASCII code + chr(): ASCII 轉字元 + if 或 % + Q:可以寫b[1]-a[1]嗎? ::: :::info EX_5_2:[d235: Q10929:You can say 11](https://zerojudge.tw/ShowProblem?problemid=d235) + 11 的倍數判別法:奇數位數字和與偶數位數字和相差為11的倍數。 - 串列編號由 0 開始,所以奇數位的索引值%2 為 0。 + [1~13 的倍數判別法](https://leestar2013.pixnet.net/blog/post/45638266) + 字元轉成整數 - ord(s[i])-48 - int(s[i]) + 使用列表生成式收進奇、偶項。 odd=sum([int(x) for x in n[0::2]]) ::: :::info EX_5_3:[h083: 3. 數位占卜](https://zerojudge.tw/ShowProblem?problemid=h083) + 將不同行的字串加入到串列 ``` python a=[] for _ in range(m): a.append(input()) a=[input() for _ in range(m)] ``` + 先用暴力法產生所有組合方式 $\rightarrow$ NA (score:20%) - [a, b, c, d] $\rightarrow$ ab、ac、ad、bc、bd、cd + 字串取前半、後半 x='abc'+'def' 前半:x[0:3],後半:x[3:6] x[:len(x)//2] + [100%解題參考](https://www.youtube.com/watch?v=VNSmO1y1W-w ) ::: :::warning [a009: 解碼器](https://zerojudge.tw/ShowProblem?problemid=a009) + 字串遍歷 ``` python str=input() for x in str: print(x) ``` ::: :::warning [f035. 最佳隊名獎](https://zerojudge.tw/ShowProblem?problemid=f035) + ord(): 字元的 ASCII code ::: :::warning [a065: 提款卡密碼](https://zerojudge.tw/ShowProblem?problemid=a065) + ord(): 字元的 ASCII code ::: :::warning [c760. 蝸牛老師的點名單 (懶憜篇)](https://zerojudge.tw/ShowProblem?problemid=c760) + s=input().split() + for x in s: # 遍歷所有名字 + 用 upper() 將名字 x 的第一個字母轉成大寫後,串接 x 其餘的字母。 ::: :::warning [e505: 12602 - Nice Licence Plates](https://zerojudge.tw/ShowProblem?problemid=e505) + 取絕對值函式:abs() ::: :::warning [g006: 密碼備忘錄 (Password)](https://zerojudge.tw/ShowProblem?problemid=g006) + 以陣列儲存字母出現的次數,cnt[ord(ch)-65]+=1。 + 因為最多100個字母,表示次數最大為100,由100->1檢查陣列中是否有這個次數,如果有則輸出 print(chr(i+65), end='') + [解題參考](https://tpmso.org/toi/wp-content/uploads/question/202105/Password.odp) ::: :::warning [c276: 沒有手機的下課時間](https://zerojudge.tw/ShowProblem?problemid=c276) ::: ## 六、函式(遞迴) ### 1. 自定函式宣告語法 ``` python def 函式名稱([參數1, 參數2, ....]): 程式碼 [return 回傳值1, 回傳值2, ....] # 函數可以同時返回多個值(一個tuple) ``` ``` python def 函式名稱([參數1=預設值, 參數2, ....]): 程式碼 [return 回傳值1, 回傳值2, ....] # 函數可以同時返回多個值(一個tuple) ``` ``` python def 函式名稱(*參數): # 參數數量不定,以tuple儲存 程式碼 [return 回傳值1, 回傳值2, ....] ``` :::info EX_6_1:[a623: 3. Combination](https://zerojudge.tw/ShowProblem?problemid=a623) + 將n!寫成函式,在主程式呼叫3次。 + $C(n,k)=\frac{n!}{(n-k)! * k!}$,例如C(4,2)=6。 ``` python def ~ ~ return ~ while True: try: ~ ~ except: break ``` ::: :::warning [f327: 刪除欄位](https://zerojudge.tw/ShowProblem?problemid=f327)。 + 寫一函式將輸入的字串轉成整數(26進位),A為1,B為2,…,AA為27,AB為28,…。 ``` python for x in s: n=n*26+ord(x)-64 ``` ::: :::warning [b112: 5. 高中運動會](https://zerojudge.tw/ShowProblem?problemid=b112)。 + 寫一函式求兩數的最大公因數(輾轉相除法)。 + a=b*q+r → gcd(a,b)=gcd(b,r) ::: ### 2. 遞迴 + 在函式之中呼叫函式自己本身(數學上為函數自己定義自己) + 問題的答案,可從同類的子問題(規模更小)得到。 + 要有終止條件,不然會無窮遞迴下去。 :::info EX_6_2:[95數學學測填充題G](http://cs.cysh.cy.edu.tw/computer_concept_108/高中數學遞迴.pdf) 用黑、白兩種顏色的正方形地磚依照如下的規律拼成若干圖形: ![](https://i.imgur.com/PrDcedl.jpg) 拼第95個圖需用到幾塊白色地磚。(478) ::: ``` python= def f(n): if ~: ~ else: return ~ n=int(input()) print(f(n)) # 5->28 ``` :::info EX_6_3:[a024. 最大公因數(GCD)](https://zerojudge.tw/ShowProblem?problemid=a024) + [原理](https://www.youtube.com/watch?v=fGesPF3QA1U) + [輾轉相除法](http://www.mathland.idv.tw/fun/euclidean.htm) (a=b\*q+r,則gcd(a, b) = gcd(b, r)) + Thonny debug ``` python= def gcd(a,b): r=a%b return gcd(b,r) ``` ::: :::warning [e357: 遞迴函數練習](https://zerojudge.tw/ShowProblem?problemid=e357) ::: :::warning [c002: 10696 - f91](https://zerojudge.tw/ShowProblem?problemid=c002) ::: ### 3. 區域變數 vs. 全域變數 - 在函式中的變數為區域變數,在函式外的變數為全域變數。 ``` python= def fun(): a = 10 # 此處的 a 為區域變數 print('函式內', a) a = 5 # 此處的 a 為全域變數 fun() print('主程式', a) # a 印出5 ``` ``` python= def fun(): global a # 使用全域變數 a a = 10 print('函式內', a) a = 5 # 此處的 a 為全域變數 fun() print('主程式', a) # a 被函式改變,印出 10 ``` ``` python= def fun(): print('函式內', a) # 可以使用全域變數 a,印出 5 a = 5 # 此處的 a 為全域變數 fun() ``` ```python= def fun(a): # 更改參數 a 不會影響函式外的變數 a = 10 print('函式內', a) a = 5 fun(a) print('主程式', a) ```