# 2023 年「[資訊科技產業專案設計](https://hackmd.io/@sysprog/info2023)」作業 1 > 貢獻者: 鮪魚-maguro > interviewer: 問 > interviewee: 答 > [模擬面試錄影1(漢)](https://www.youtube.com/watch?v=ntYhICGs6-o) > [模擬面試錄影2(漢)](https://www.youtube.com/watch?v=LRuRTilYY24) > [模擬面試錄影(英)](https://www.youtube.com/watch?v=jh_uQVzuHtI) ### 改進方案&初步檢討 * Repeat: 重複提問,確認自己理解原本的要求 * 重複提問的部分我只有用類似自言自語的方式,應該向面試官盡量去詢問,避免理解錯誤。 * Examples: 針對題目條件,舉出符合的案例,不限於特例 * 這部分沒有做到,應該要在一開始就先考慮幾個範例,並且試著提出edge case * Approach: 說明自己打算採用的方案 * 說明後可以和 interviewer 討論 * Code: 撰寫程式 * 在寫程式時應盡量精簡講重點,避免冗言贅字, * Test: 若不能實際測試,那說明驗證程式的方案 * 如同Examples ## [1046. Last Stone Weight](https://leetcode.com/problems/last-stone-weight/) ### 過程的討論 *問: 挑選出這串石頭裡面最重的兩顆 讓兩顆石頭去互相的撞擊 如果這兩顆石頭重量一樣的話 那兩顆石頭都會消失 不是的話大的會留下來小的會消失 一路進行下去直到最後只剩下一顆或者是沒有石頭為止* 答: maximum他就是負責去找出這個array裡面最大跟第二大的 所以我的參數要給他我這個array 那因為是c語言 所以同時也要給它這個array這size 然後我這邊定了兩個變數 這邊跟正常的想法比較不一樣的是 它其實代表的不是最大值的那個值 而是它會在這個array的第幾個位置 我會從最開頭一路找找到這個正列的最結束 那一旦發現有人比IMAX還要大那這個時候他就會去取代IMAX 根據c語言的特性 array帶進來之後裡面的值做的變動 他的參數也是會跟著改變的 最開始的這個數時值陣列stones它的 裡面的數值會會有越來越多的0 那一旦最後剩只剩下一個數值 其他都是0或全部都是0的時候 這個就是我們的終點 *問: 好你說用C++ 寫的方式會比較快 那能不能請你一樣 的題目用C++的方式再寫一次呢* 答: 主要原因是因為他用的是C++能使用到的一些 function功能會比c來得多 那這邊想要用到priority queue 它的top就會是裡面的最大值 把它拿掉之後 這時候就可以找出第二大的紙 大的最大值跟第二大的值他們會相撞得到一個新的值 有可能是0那也有可能是有實際的值 那不管怎麼樣我們都可以把它加進去 直接做push就好了 然後就會一路去做兩顆石頭的相撞 那我們這邊是砍掉兩個加了一個 所以他的size會越來越小 我必須確保說這個priority queue裡面是還有東西的 當他等於一剩下最後一個的時候 或者是說等於0的時候 他就會離開這個迴圈 *問: c以及C++兩個回答的方式 出來的結果 可以請你分析這兩個結果 有什麼差異* 答: 時間複雜度會是一樣的 它的時間複雜就是一個單純的常數 空間複雜度的話 在C++ 這邊我們額外新增一個priority queue 所以這邊的空間複雜度會大一點 不過它的差異也都是只有常數的差異而已 所以這兩個做法我覺得在這個地方是C++略勝一籌 ### 程式碼 ```cpp int lastStoneWeight(int *stones, int stonesSize) { while (true) { int num = 0, idx; for (int i = 0; i < stonesSize; i++) { if (stones[i] > 0) { num++; idx = i; } if (num == 2) break; } if (num == 1) { return stones[idx]; } else if (num == 0) { return 0; } } } void maximum(int *arr, int arrSize) { int i_max1 = 0, i_max2 = 1; for (int i = 0; i < arrSize; i++) { if (arr[i] > arr[i_max1]) { i_max2 = i_max1; i_max1 = i; } else if (arr[i] > arr[i_max2]) { i_max2 = i; } } arr[i_max1] = arr[i_max1] - arr[i_max2]; arr[i_max2] = 0; } ``` ```cpp class Solution { public: int lastStoneWeight(vector<int> &stones) { priority_queue<int> pq(begin(stones), end(stones)); int ans = 0; while (pq.size() > 1) { int max1 = pq.top(); pq.pop(); int max2 = pq.top(); pq.pop(); pq.push(max1 - max2); } return pq.empty() ? 0 : pq.top(); } }; ``` ## [1282. Group the People Given the Group Size They Belong To](https://leetcode.com/problems/group-the-people-given-the-group-size-they-belong-to/) ### 過程的討論 *問: 陣列代表著每一位運動員他在這項競技所獲得的分數 那請你回傳他的結果 分別是所有運動員裡面的第幾名* 答: 題目既要我們把每位運動員的名次給決定出來 同時又要求說必須要保留原本運動員的順序 這邊用priority queue去做 那其實c的二維陣列也可以做 我這邊做的做法是 就單純沒有按照大小去存 但是我在取的時候有照大小去取 因為priority queue在取的時候 我們就可以很輕易的用top的函式去呼叫它的最大 ### 程式碼 ```cpp class Solution { public: vector<int> findRelativeRanks(vector<int> &score) { priority_queue<pair<int, int>> pq; for (int i = 0; i < score.size(); i++) { pq.push({score[i], i}); } vector<int> ans(pq.size()); for (int i = 1; i < size(score); i++) { ans[pq.top().second] = i; pq.pop(); } return ans; } }; ``` ## [338. Counting Bits](https://leetcode.com/problems/counting-bits/) ### 程式碼 ```cpp int *countBits(int n, int *returnSize) { int *ans = (int *)malloc((n + 1) * sizeof(n + 1)); *returnSize = n + 1; ans[0] = 0; for (int i = 0; i < n + 1; i++) { ans[i] = ans[i] / 2 + i % 2; } return ans; } ``` ## 第二次作業-他評01 ### 關於interviewer - [ ] 優點 1.題目講解清晰 - [ ] 可改進之處 1.Last Stone Weight影片[14:30](https://youtu.be/ntYhICGs6-o?t=869)裡面面試者好像沒有先提到用C++ 寫會比較快,反而是在interviewer主動說"你說用C++寫的方式會比較快"後才提出,可能剪輯順序有錯誤,不然看起來很怪。 2.程式碼錯誤沒有指出。 ### 關於interviewee - [ ] 優點 1.有先說明要用哪種語言實作 - [ ] 可改進之處 1.依照REACTO的步驟的話,有點太快進到撰寫程式碼的步驟,應該先舉例和大概說明實作方法比較好理解。後續也缺乏測試程式正確性的步驟。 2.寫Last Stone Weight這題的C語言程式的時候可能需要邊打邊說,影片裡好像蠻常是先打完一小段再說,像是[6:37](https://youtu.be/ntYhICGs6-o?t=397)但是等到打完一小段程式碼再說明,中間的時間容易讓人想放空。後面寫C++反而沒有這個問題。 3.C程式碼好像有問題,至少我放進leetcode後沒有通過。Last Stone Weight這題,C語言版本寫了兩個function但是lastStoneWeight卻沒呼叫maximum,也沒有說明要怎麼搭配使用這兩個function。 4.沒有討論到特例,當輸入的stones陣列長度只有1的時候。 參考修改程式碼: ```C void maximum(int *arr, int arrSize) { int i_max1 = 0; //假設第一大是stones[0] int i_max2 = 1; //假設第二大是stones[1] if(arr[i_max1]<arr[i_max2]){//如果stones[0]<stones[1]要對調 i_max1=1; i_max2=0; } for (int i = 2; i < arrSize; i++) { if (arr[i] > arr[i_max1]) { i_max2 = i_max1; i_max1 = i; } else if (arr[i] > arr[i_max2]) i_max2 = i; } arr[i_max1] = arr[i_max1] - arr[i_max2]; arr[i_max2] = 0; } int lastStoneWeight(int *stones, int stonesSize) { if(stonesSize==1) return stones[0]; while(true){ maximum(stones,stonesSize);//smash int num = 0; int idx=0; for (int i = 0; i < stonesSize; i++) { if (stones[i] > 0) { num++; idx = i; } if (num == 2) break; } if (num == 1) return stones[idx]; else if (num == 0) return 0; } } ``` 5.英文版本影片中要使用C語言動態記憶體配置,我想應該不用特別跟面試官講解為什麼,而且你直接講中文名稱XD,後面也很容易不小心講出中文名詞。 6.英文版本中,malloc語法錯誤應該是sizeof(int),應該先提出要用DP實作的想法是什麼,再開始寫程式,雙方比較好討論,也比較好找出錯誤。程式的部分for迴圈可以修改成這樣,應該就會對了。 ```C ans[0]=0; for (int i = 1; i < n+1; i++) { if((i-1)%2) ans[i] = ans[i/2]; else ans[i]=ans[i-1]+1; } ``` ## 第二次作業-他評02 ### 關於intervier #### 優點: * 有些題目有經過變化,而不是直接放leetcode原題 * 能夠清楚表達題目 #### 可改善的地方: * 與interviewee的互動不夠多,這是interview不是考官和考生的關係,應該多和interviewee溝通,來增進互相的理解。 ### 關於interviewee #### 優點: * 在寫程式的時候有邊寫邊解釋寫法跟邏輯,避免寫程式的對話空白期。 * 講話邏輯清楚且語調穩定 #### 可改進的地方: * 沒有完全遵照REACTO的方式 * 可以在開始寫程式之前,解釋自己的邏輯給interviewer聽,確認邏輯是否錯誤,如有錯誤想不出來可適時提問。 * 也可以先舉幾個簡單的例子來映證自己的想法。 * 在寫程式時可加快打字速度,畢竟時間有限 * 可以再加強英文的表達能力,避免過多的冗詞贅字。 ## 第二次作業-他評03 #### 優點: * 在開始coding前有先用例子重新解釋題目 #### 可改進的地方: * 邊界條件應該不會是面試官直接說出來(一開始就打在螢幕上)而是要透過一問一答方式來回確認所有要求