2023. 1. 27. 15:03ใProgramming/Swift
Xcode Project
An Xcode project is a repository for all the files, resources, and information required to build one or more software products. A project contains all the elements used to build your products and maintains the relationships between those elements. It contains one or more targets, which specify how to build products.
A project defines default build settings for all the targets in the project (each target can also specify its own build settings, which override the project build settings).
An Xcode project file contains the following information:
- References to source files:
- Source code, including header files and implementation files
- Libraries and frameworks, internal and external
- Resource files
- Image files
- Interface Builder (nib) files
- Groups used to organize the source files in the structure navigator (๊ณ์ธต์ผ๋ก ๋ณด์ด๋ ํด๋๋ค(๊ทธ๋ฃน)๋ ํฌํจ๋๋ค)
- Project-level build configurations. You can specify more than one build configuration for a project;
for example, you might have debug and release build settings for a project. - Targets, where each target specifies:
- A reference to one product built by the project
- References to the source files needed to build that product
- The build configurations that can be used to build that product, including dependencies on other targets and other settings; the project-level build settings are used when the targets’ build configurations do not override them
- The executable environments(์คํ ํ๊ฒฝ) that can be used to debug or test the program, where each executable environment specifies:
- What executable to launch when you run or debug from Xcode
- Command-line arguments to be passed to the executable, if any
- Environmental variables to be set when the program runs, if any
A project can stand alone or can be included in a workspace.
You use Xcode schemes to specify which target, build configuration, and executable configuration is active at a given time.
- Project๋ Application์ ๋น๋ํ๊ธฐ ์ํ ํ์ผ, ๋ฆฌ์์ค, ์ ๋ณด๋ฅผ ๋ด์ repository
- ์ฒ์ Xcode๋ฅผ ์ผ๊ณ Single View Application์ ์์ฑํ๋ฉด Project๋ฅผ ์์ฑํ๊ฒ ๋ฉ๋๋ค. ์ด๋, ํ๋ก์ ํธ์ ๋๋ ํ ๋ฆฌ๋ฅผ ์ดํด๋ณด๊ฒ ๋๋ฉด ํ๋ก์ ํธ๋ช .xcodeproj๋ผ๋ ํ์ผ์ด ์๊ธด ๊ฒ์ ํ์ธํ ์ ์์ต๋๋ค.
- ์ ํํ ์๊ธฐํ๋ฉด ์ด๋ ํ์ผ์ด ์๋๋ผ ๋๋ ํ ๋ฆฌ์
๋๋ค.
ํฐ๋ฏธ๋์ ํตํด ํด๋น ํ์ผ๋ก ๋ค์ด๊ฐ๊ฒ ๋๋ฉด project.pbxproj์ด๋ผ๋ ํ์ผ๊ณผ xcuserdata๋ผ๋ ๋๋ ํ ๋ฆฌ๊ฐ ์กด์ฌํฉ๋๋ค.
- project.pbxproj๊ฐ ์ค์ ์ค์ ํ์ผ์ด๋ค.
- ํ๋ก์ ํธ ๋ด๋ถ์์ ์์ฑ๋ ํ์ผ์, ํ์ผ ์ ํ์ ๋ฐ๋ผ reference๋ฅผ ์ ์ฅํ๋ ์ญํ
- .xcodeproj : ํ์ผ์ด ์๋ ๋๋ ํ ๋ฆฌ์์ ์ฃผ์!
ํด๋น ๋๋ ํ ๋ฆฌ์ project.pbxproj ํ์ผ, project.xcworkspace ํ์ผ, xcuserdata ๋๋ ํ ๋ฆฌ๊ฐ ์กด์ฌ
- ์ด project.pbxproj๋ ์ฌ์ค git์ ์ฌ์ฉํ ๊ฒฝ์ฐ ์ถฉ๋์ด ์ผ์ด๋๋ ์ฃผ์ ํ์ผ ์ค ํ๋!
- xcode์ ์ต์์ ํ๋ก์ ํธ ๊ด๋ฆฌ ํจํค์ง์ธ .pbxproj์ ๊ฒฝ์ฐ git ๋ณํฉ์ ์ด๋ป๊ฒ ๋ณํฉํ ๊ฒ์ธ๊ฐ์ ๋ํด ๋๊ฐํ ๊ฒฝ์ฐ๊ฐ ๋ง์
- 2๋ช
์ ํ์์ด ์์
์ ์ํด ๊ฐ์ A ํ์ผ, B ํ์ผ์ ์์ฑํ์ฌ ๊ฐ์ ์ปค๋ฐํ ๊ฒฝ์ฐ ํ ์ชฝ์๋ B ํ์ผ์ reference, ๋ค๋ฅธ ์ชฝ์ A ํ์ผ์ reference๊ฐ ์์ต๋๋ค. ๊ทธ๋์ ๋ ์์
์์ ์ฌ์ฉ๋ project.pbxproj ํ์ผ์ ์๋ก ๋ค๋ฅธ reference๋ฅผ ๊ฐ์ง๊ณ ์์ด์ conflict๊ฐ ์ผ์ด๋ฉ๋๋ค.
๊ทธ๋ฆฌ๊ณ ์ด ์ถฉ๋์ด ๊ณ ์น๊ธฐ ์ฝ์ง ์์ ๊ฒ์ด project.pbxproj๊ฐ ๊นจ์ง๋ฉด ํ๋ก์ ํธ ์์ฒด๊ฐ ์ด๋ฆฌ์ง ์์ต๋๋ค. ๊ทธ๋ฌ๋ฏ๋ก ์ด๋ด ๋๋ ์๋ํฐ๋ ์์คํธ๋ฆฌ ๊ฐ์ ํ์ ๊ด๋ฆฌ ํด๋ก ํด๋น ์ถฉ๋์ ์์ ํด์ผ ํฉ๋๋ค. (dev - dev - mybranch) - xcuserdata๋ ํ๋ก์ ํธ์ ๊ฐ์ธ ์ค์ ์ ๋ด์ ๋๋ ํ ๋ฆฌ: breakpoint, UI layout ์ค์ ๋ฑ์ ๋ด์, ํ๋ก์ ํธ ์์ฒด์ ํฌ๊ฒ ์ํฅ์ ์ฃผ์ง ์๋ ๋๋ ํ ๋ฆฌ (.xcuserdata ์ถ์ ๋ฐฉ์ง ์ถ๊ฐ ์ค์ ๊ฐ๋ฅ)
- Workspace๋ ์ฌ๋ฌ ๊ฐ์ Project๋ฅผ ๋ด์ ๊ด๋ฆฌํ ์ ์๋๋ก ํด์ฃผ๋ ๊ฐ๋
- CocoaPod ์ฌ์ฉ ์ ์ ํ๊ฒ ๋ ๊ฒฝ์ฐ ๋ง๋ค. (๋๋!)
- Target์ ํ๋ก์ ํธ๋ฅผ ํตํด ์์ฑ๋๋ Application์ ์ง์นญํจ. ์ผ๋ฐ์ ์ผ๋ก ํ๋์ ๋ชจ๋์ ์๋ฏธ
- Target์ Xcode์ ๋น๋๋ฅผ ํตํด ์์ฑ๋ ์ต์ข product
- Target์ ์ด๋ป๊ฒ ํ๋ก์ ํธ๋ฅผ ๋น๋ํ ๊ฒ์ธ์ง๋ฅผ ๋ด๋น
- ๊ธฐ๋ณธ์ ์ผ๋ก Target์ ํ๋ก์ ํธ ์์ฑ์ 1๊ฐ๋ง ์์ฑ๋์ง๋ง, ๋ชฉ์ ์ ๋ฐ๋ผ์ ํ๋์ ํ๋ก์ ํธ์ ์ฌ๋ฌ ๊ฐ์ Target์ ์์ฑํ ์ ์์
- ๊ฐ๊ฐ์ target์ ํ๋ก์ ํธ์ build setting์ ์ค์ ํ ์ ์๋ค.
- ๊ฐ๊ฐ์ target์ ํ๋ก์ ํธ์์ ํฌํจ๋ ๊ฐ์ฒด, ๋ฆฌ์์ค, ํน์ ๋ณ๋์ ์คํฌ๋ฆฝํธ๋ฅผ ๋ฐ๋ก ์ค์ ํ๋๋ก ํจ
- Xcode์์๋ target์ ํตํด์ ํ๋์ ํ๋ก์ ํธ๋ฅผ ์ฌ๋ฌ ๊ฐ์ ๋ฐฐํฌํ์ผ๋ก ์ฌ์ฉํ ์ ์๋๋ก ํจ
(i.e. iPhone์ฉ target, iPad์ฉ target, ํน์ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๊ฐ ํฌํจ๋ target ๋ฑ)
- Scheme์ Target์ด ํ๋ก์ ํธ๋ฅผ Build, Profile, Test๋ฑ์ ํ ๋ ์ผ์ด๋ ์ผ๋ค์ ์ ์ํ ์ ์๋๋ก ํด์ฃผ๋ ํญ๋ชฉ
- Scheme์์๋ ํ๋ก์ ํธ ๋น๋์ ์ฌ์ฉ๋๋ ํ๊ฒฝ๋ณ์๋ ์ธ์๋ฅผ ๋๊ฒจ์ค ์ ์์
- Scheme์ ํ๋ก์ ํธ์ ๋ฐํ์์์ ์ผ์ด๋ ์ผ์ ์ค์ ํ ์ ์์
- pbxproj is an important file in the Xcode configuration bundle. It is responsible for maintaining references to all of the linked files and their groupings, linked frameworks, and most importantly, the project's build settings
- pbxproj ํ์ผ์ Build Setting(์ค์ ํ๋ก์ ํธ์ ์ค์ ) ์ ๋ด์ ํ์ผ
ํ๋ก์ ํธ ๋ด๋ถ์์ ์์ฑ๋ ํ์ผ๋ค์ ํ์ผ ์ ํ์ ๋ฐ๋ผ reference๋ฅผ ์ ์ฅํ๊ณ ์์
conflict๋ ํ์ผ์ ๋ํ reference๊ฐ ์์ ๋ ์ผ์ด๋๋ค. - ์ค์ํ ํ์ผ์ด๋ฏ๋ก .gitignore์ ์ถ๊ฐํ ์ ์๋ ํ์ผ์ด๋ค.
- pbxproj ํ์ผ์ ๋ณด๊ธฐ๊ฐ ๋๋ฌด ๋ถํธํ๋ค. Pro Git Book ์์๋ ์ฌ์ค ํ ์คํธ ํ์ผ์ด์ง๋ง ๋ง๋ ๋ชฉ์ ๊ณผ ์๋๋ฅผ ๋ณด๋ฉด ๋ฐ์ด๋๋ฆฌ ํ์ผ์ด๋ผ๊ณ ๋งํ๋ฉฐ, ์ฌ๋ฌ ๋ช ์ด ์ด ํ์ผ์ ๋์์ ์์ ํ๊ณ Merge ํ ๋ diff๊ฐ ๋์์ด ๋์ง ์๊ณ ํ๋ก๊ทธ๋จ์ด ์ฝ๊ณ ์ฐ๋ ํ์ผ์ด๊ธฐ ๋๋ฌธ์ ๋ฐ์ด๋๋ฆฌ ํ์ผ์ฒ๋ผ ์ทจ๊ธํ๋ ๊ฒ์ด ์ณ๋ค๊ณ ํ๋ค. (๋๋ ์ด ํ์ผ๋ก conflict ํด๊ฒฐ์ ์ง์ ํ์๋๋ฐ.. ๋ค๋ฅธ ๋ฐฉ๋๋ก๋ pbxproj ํ์ผ์ binary ํ์ผ๋ก ์ทจ๊ธํ๊ธฐ ์ํด ํ๋ก์ ํธ ์ต์์ ํด๋(README.md๊ฐ ์๋ ๊ณณ)์์ .gitattributes ํ์ผ์ ๋ง๋ค์ด, conflict ๋ฐ์์ ํด๊ฒฐํ ์ ์๋ค๊ณ ํ๋ค.)
- ์ด๋ฒ ํ๋ก์ ํธ์์ ํ์ผ ์ค๋ฅ๊ฐ ๋ฐ์ํ ์ด์ (๋นจ๊ฐ ๊ธ์จ ๋จ๋ ์ค๋ฅ -> ์ข
์ข
๋ฐ์ํ๋ค)
- ์ถ๊ฐ๋ ํ์ผ์ ๋ํ์ฌ project file์ ํตํ reference๊ฐ ์ ๋๋ก ๋ฐ๋ผ์ค์ง ์์์
- ๋ก์ปฌ์๋ ์ ๋๋ก ๋ค์ด๊ฐ ์๋ ํ์ผ๋ค์, ์ง์ ์ฌ์ฐ๊ฒฐํด์ฃผ๋ฉด ํด๊ฒฐ๋๋ค
- + build Phase์์ .gitkeep ์ง์ฐ๊ธฐ
- ํ์ผ์ ๋ํ ์ฐธ์กฐ๋ xcodeproj/project.pbxproj ํ์ผ์ ์ ์ฅํ๊ณ ์๋ค.
๋ฐ๋ผ์, ํ์ผ์ ์ถ๊ฐํ๊ฑฐ๋ ์ญ์ ํ ๊ฒฝ์ฐ project.pbxproj๋ ํจ๊ป ์ปค๋ฐํด์ค์ผ ํ๋ค. - ํ๋ก์ ํธ ํ์ผ๋ค์ ๋ณ๊ฒฝ์ ์ ํ์ธํ ๋ค, ํจ๊ป ์ปค๋ฐํด์ฃผ๊ธฐ! :)
Xcode์ ํ์ผ ์ถ๊ฐํ๊ธฐ
1. Destination
- Copy items if needed
ํ์ผ์ ๋ณต์ฌํ์ฌ ์ถ๊ฐํ๋ ์ต์ - ์ฒดํฌํ๋ ๊ฒ์ด ์ข๋ค. (Copy by Value๊ฐ์ ๊ฐ๋ )
์ด ์ต์ ์์ด ์ถ๊ฐํ๋ค๋ฉด ํ๋ก์ ํธ ํด๋๋ก ๋ณต์ฌ๋์ง ์๊ณ ์๋ณธ์ ๋ ํผ๋ฐ์ค๋ฅผ ์ฐธ์กฐ -> ์๋ณธ์ ํด๋๊ฐ ๋ณ๊ฒฝ๋๊ฑฐ๋ ์ญ์ ๋ ๋ ๋ ์ด์ ํด๋น ํ์ผ์ ์ฌ์ฉํ ์ ์๊ฒ ๋๋ค.
+ ๋ด project.pbxproj ํ์ผ์๋ ๋ฐ์์ด ๋์ด ์์ง๋ง ์ด๋๋ก git์ ์ปค๋ฐํ๊ฒ ๋๋ค๋ฉด, ๋ค๋ฅธ ์ฌ๋๋ค์๊ฒ๋ ํด๋น ํ์ผ์ด ์กด์ฌํ์ง ์์ ์ค๋ฅ๊ฐ ๋ฐ์ํ ์ ์๋ค.
2. Added folders
- ํ์ผ ์ถ๊ฐ
ํด๋น ์ต์ ์ 'Added folders' ์ต์ ์ผ๋ก, ํ์ผ ์ถ๊ฐ์ ๋ํด์๋ ๋ ์ต์ ์ ๋ํ ์ฐจ์ด๊ฐ ์๋ค.
- ํด๋ ์ถ๊ฐ
1) Create groups
- ๋ ธ๋ ํด๋๊ฐ ์์ฑ๋๋ค.
- ๊ฐ์ ํด๋์ ๊ฐ๋ ์ผ๋ก ํ์ผ ์์คํ (Finder)์ ์๋ฏธํ์ง ์๋๋ค.
- Finder์์ ํด๋ ๋ด์ ํ์ผ์ ์ถ๊ฐํด๋ ํ๋ก์ ํธ ํด๋ ๋ด์๋ ๋ฐ์๋์ง ์๋๋ค.
- Finder์์ ํด๋์ ์ด๋ฆ์ ๋ณ๊ฒฝํด๋ ํ๋ก์ ํธ ๋ด์์ ํด๋๊ฐ ๋จ์ ์๋ค. (๋ด๋ถ ํ์ผ์ ์ฌ์ฉํ ์ ์๋ค.)
- Finder์์ ํด๋๋ฅผ ์ญ์ ํด๋ ํ๋ก์ ํธ ๋ด์ ํด๋๋ ๋จ์ ์๋ค. (๋ด๋ถ ํ์ผ์ ์ฌ์ฉํ ์ ์๋ค.)
2) Create folder references
- ํ๋ ํด๋๊ฐ ์์ฑ๋๋ค.
- ํ์ผ ์์คํ (Finder)์ ํด๋์ ์ผ๋์ผ ๋งตํ๋๋ค. ๋ฐ๋ผ์, ๋ณ๊ฒฝ ๋ด์ญ์ด ๋ฐ์๋๋ค.
- Finder์์ ํด๋ ๋ด์ ํ์ผ์ ์ถ๊ฐํ๋ฉด ํ๋ก์ ํธ ํด๋ ๋ด์๋ ํ์ผ์ด ์ถ๊ฐ๋๋ค.
- Finder์์ ํด๋์ ์ด๋ฆ์ ๋ณ๊ฒฝํ๋ฉด ํ๋ก์ ํธ ๋ด์์ ํด๋๊ฐ ์ญ์ ๋๋ค.
- Finder์์ ํด๋๋ฅผ ์ญ์ ํ๋ฉด ํ๋ก์ ํธ ๋ด์์๋ ํด๋๊ฐ ์ญ์ ๋๋ค.
- ๋๋ ์ฃผ๋ก ํ๋ ํด๋๋ก ์ฌ์ฉํ์๋ค.
Xcode์์ ํ์ผ ์ญ์ ํ๊ธฐ
- Remove Reference
ํ์ผ์ ๋ํ ์ฐธ์กฐ๋ง ์ ๊ฑฐ๋๋ค. ๋ฐ๋ผ์ ํ์ผ ์ถ๊ฐ ์ Copy items if needed ์ต์ ์ ์ ํํ์ง ์์๋ค๋ฉด, ์ฐธ์กฐ๋ง ์ ๊ฑฐํ์ฌ๋ ์๊ด์๋ค.
๋ง์ฝ Copy items if needed ์ต์ ์ ์ ํํ์ฌ ์ถ๊ฐํ๋๋ฐ, Remove Reference๋ก ์ ๊ฑฐํ๋ค๋ฉด ์ค์ ํ๋ก์ ํธ ํด๋์ ํ์ผ์ ๋จ์์์ง๋ง, ์ฐธ์กฐ๋ง ์ ๊ฑฐ๊ฐ ๋ ์ํ์ด๋ค. ๋จ์ ํ์ผ์ Finder์์ ์ง์ ์ญ์ ํ ์ ์๋ค.
- Move to Trash
ํ์ผ์ ๋ํ ์ฐธ์กฐ์ ํจ๊ป Finder์์ ํ์ธํ ์ ์๋ ๋ฌผ๋ฆฌ์ ์ธ ํ์ผ๊น์ง ํจ๊ป ์ ๊ฑฐ๋๋ค.
Understand why merge conflicts happen
Version control systems, like git, auto-magically manage code contributions. It identifies the change, when it occurred, who made it, and on what line so that developers can easily track the history of their codebase.
However, git sometimes gets confused in the following situations:
- When more than one person changes the same line in a file and tries to merge the change to the same branch
- When a developer deletes a file, but another developer edits it, and they both try to merge their changes to the same branch.
- When a developer deletes a line, but another developer edits it, and they both try to merge their changes to the same branch
- When a developer is cherry-picking a commit, which is the act of picking a commit from a branch and applying it to another
- When a developer is rebasing a branch, which is the process of moving a sequence of commits to a base commit
Git is unsure which change to apply, so it leans on the developer for help and notifies them of a merge conflict. Your job is to help git determine which proposed change is most accurate and up to date.
The benefits of preventing merge conflicts
Sometimes, doing that upfront work for your team or yourself to avoid merge conflicts may seem tedious, but it's worthwhile. Using guard rails to prevent merge conflicts will save time and increase developer happiness. Solving a bug, writing a new feature, or scripting automation is time-consuming. Adding the barrier of debugging and resolving a merge conflict means it takes longer to merge to 'main' and longer to deploy to production. Also, if your team is constantly experiencing conflicts, they'll eventually feel disenchanted with their work.
- Standardize formatting rules.
- Make small commits and frequently review pull requests.
- Rebase, rebase, rebase (early and often)
- Pay attention and communicate.
โณ๏ธ Four ways to prevent merge conflicts
Standardize formatting rules
Many times conflicts occur because of formatting discrepancies. For example, a technologist inadvertently added extra white space on the same line where another developer added new code. Also, people have different coding styles. For instance, some JavaScript developers use semicolons, but some don't. If you're working on a team, enforce code formatters and linting rules. Make sure everyone on the team is aware of these rules and tools to reduce the number of merge conflicts you experience due to formatting.
Make small commits and frequently review pull requests
I'm going to admit something super embarrassing. Back in the day (~3 years ago), I would make pull requests changing over 50 files, and then I would get annoyed that I had so many merge conflicts. (์์ฐ) In hindsight, I was wrong. Changing over 50 files in less than two weeks increased the chances that other developers also made updates to those files.
Also, creating large pull requests discouraged my teammates from thoroughly and quickly reviewing my code. My pull request would sit longer than necessary because it was a behemoth my teammates preferred to avoid.
Take my mistakes as a lesson to make small changes, commit them, and have folks review the pull request as soon as they are available. This way, you'll have fewer chances to change files that other teammates are simultaneously working on.
Rebase, rebase, rebase (early and often)
I learned about rebasing in 2019 from a coworker that my team dubbed the "Git-tator" (Git and dictator combined). He was earnest about improving our startup's git history and reducing the number of blockers we experienced due to merge conflicts. Arguments ensued because many developers were often working on extensive features simultaneously. We even argued over simple conflicts when people imported new dependencies for different features in the same file on the same line. Thus, he introduced us to rebasing, which significantly improved our software development workflow.
What is rebasing?
The git rebase command reapplies changes from one branch into another, which is very similar to the git merge command. However, in this case, git rebase rewrites the commit history to produce a straight, linear succession of commits.
How rebasing helps prevent merge conflicts
Rebasing is not going to magically remove all merge conflicts. In fact, you may encounter conflicts while rebasing. Sometimes, you will have to repeatedly resolve the same conflict while rebasing. However, merge conflicts happen because multiple changes happen to the same chunk of code simultaneously. If you rebase your local working branch with the default branch (main or master), you're rewriting your local commit history with the default branch's history and then reapplying your changes. In many situations, rebasing first and then merging can make teamwork easier. Rebasing is an option, but not the only solution.(dev-dev-my branch๋ ๋ค๋ฅธ๊ฑธ๊น? ๋ญ์ง?)
Be careful with rebasing
Be warned – there are moments when rebasing is not recommended. Rebasing is dangerous because you're rewriting history. When rebasing, pay attention to the changes you're accepting because you risk overwriting teammates' changes or including files your teammates intended to delete. You also probably wouldn't want to rebase a public repository on the main branch because that would inaccurately rewrite the history.
If you're new to rebasing or nervous about rebasing, pair with a teammate to sanity check.
How I rebase to avoid merge conflicts
There's no correct workflow – just preferred ones.
Here's the workflow I follow as determined by teams and codebases I’ve worked in:
- If I'm working with more than one person, I'll create a feature branch from the default branch.
This way, we can contribute to the feature branch in tandem. - I'll make a new local branch from the default or feature branch
- I'll add and commit new changes to my local branch
- I'll rebase updates from the default or feature branch
- I'll merge changes from my local branch to the default or feature branch
Pay attention and communicate
No git command or software tool can replace the need for communication in engineering teams. Being a good software developer and collaborator goes beyond writing code. Good software developers communicate with teammates. Keep your team aware of what files you will be touching and coordinate with your Product Manager and SCRUM Master to avoid working on features that conflict with other features.
If you're working alone, pretend you're working on a team by:
- creating branches
- creating pull requests
- Avoid allowing pull requests to become stale
- Make sure you're not changing the same lines of code before merging a prior change
- Establish and follow formatting rules
Working alone shouldn't stop you from practicing healthy coding habits, so try to keep your codebase clean!
Your codebase may turn into an open source project or a project that you show off in interviews.
์ถ์ฒ
- https://developer.apple.com/library/archive/featuredarticles/XcodeConcepts/Concept-Projects.html
- https://hcn1519.github.io/articles/2018-06/xcodeconfiguration
- https://isjang98.github.io/blog/Merge-conflicts-project-pbxproj
- https://velog.io/@honghoker/git-Xcode-pbxproj-merge-conflict-๋ฌธ์ -ํด๊ฒฐ-๋ฐฉ๋ฒ
- https://ios-development.tistory.com/406
- https://dev.to/github/how-to-prevent-merge-conflicts-or-at-least-have-less-of-them-109p
- https://nazan9.tistory.com/7
- https://git-scm.com/book/ko/v2/Git-๋ธ๋์น-Rebase-ํ๊ธฐ
'Programming > Swift' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
[Swift] RXSwift Refactoring in iOS (0) | 2024.02.13 |
---|---|
[Swift] Alamofire vs Moya & URLSession์ ๋ํ์ฌ (0) | 2023.01.27 |
[Swift] SnapKit vs AutoLayout & Pros and cons of using 3rd-Party (0) | 2023.01.26 |
[Swift] CocoaPod vs Swift Package Manager (0) | 2023.01.26 |
[Swift] tap gesture (0) | 2022.11.21 |