[ConnectEx]
Socket 통신에서 동시다발적으로 여러 클라이언트에게 오는 요청을 효율적으로 처리하기 위해 많은 방법들이 고안되었다.
그 중 대표적으로 리눅스에는 Epoll, 윈도우에는 IOCP가 있다. 
IOCP에는 기존에 있던 소켓 함수들의 확장 모델이 있는데 그 중 하나가 ConnectEx 함수이다.
실제로 국내 모 메신져 회사의 Windows Application도 초기에 connect API를 사용하다가 
패치 후 ConnectEx API를 사용하는것을 보면 상용프로그램에서도 IOCP가 성능을 좌지우지 하는듯 보인다.
 
[ConnectEx 사용법]
ConnectEx API는 기존 API들 처럼 DLL에 정의되어있지 않다. 
런타임시 WSAIoctl API를 통해 ConnectEx를 메모리에 생성한 후 함수 포인터를 가져와 사용해야한다.
 
[ConnectEx 함수 포맷]
 
[ConnectEx 후킹]
먼저 ConnectEx API를 생성해준다.
이후 WSAIoctl을 후킹하고 인자를 검사하여 함수 포인터를 얻겠다는 의미로 dwIoControlCode 인자가 SIO_GET_EXTENSION_FUNCTION_POINTER 인지 확인하고
어떠한 API를 생성할것인지를 알 수 있는 lpvInBuffer 인자가 WSA_CONNECTEX 인지 검사한 후에 조건에 만족한다면 
ConnectEx를 생성하겠다는 것으로 판단할 수 있으므로 실제 ConnectEx의 주소가 들어갈 포인터에 MyConnectEx 함수의 주소를 대입해주면 된다.
LPFN_CONNECTEX Org_Connectex;//해당 함수포인터는 ConnectEx의 원본함수가 된다.
 
BOOl GetConnectEx(){
    GUID guid = WSAID_CONNECTEX;
    DWORD dwBytes;
    WSADATA wsaData;
    WSAStartup(MAKEWORD(2, 2), &wsaData);
    SOCKET sock = socket(AF_INET, SOCK_STREAM, 0);
    if (sock == INVALID_SOCKET) {
        TCHAR buf[100];
        _stprintf(buf, TEXT("error : %x"), GetLastError());
        OutputDebugString(buf);
        return FALSE;
    }
    int rc = WSAIoctl(sock, SIO_GET_EXTENSION_FUNCTION_POINTER,
        &guid, sizeof(guid),
        &Org_Connectex, sizeof(Org_Connectex),
        &dwBytes, NULL, NULL);
    if(!rc){
        return TRUE;
    }
    else{
        return FALSE;
    }
}
int WSAAPI MyWSAIoctl(
    SOCKET s,
    DWORD dwIoControlCode,
    LPVOID lpvInBuffer,
    DWORD cbInBuffer,
    LPVOID lpvOutBuffer,
    DWORD cbOutBuffer,
    LPDWORD lpcbBytesReturned,
    LPWSAOVERLAPPED lpOverlapped,
    LPWSAOVERLAPPED_COMPLETION_ROUTINE lpCompletionRoutine
) {
    int ret = ((PFMyWSAIoctl)OrgWinAPI[WSAIOCTL])(
        s, 
        dwIoControlCode, 
        lpvInBuffer, 
        cbInBuffer, 
        lpvOutBuffer, 
        cbOutBuffer, 
        lpcbBytesReturned, 
        lpOverlapped, 
        lpCompletionRoutine
    );
    if (dwIoControlCode == SIO_GET_EXTENSION_FUNCTION_POINTER && CheckGUID(lpvInBuffer, WSAID_CONNECTEX)) {
        // WSAIoctl의 인자를 검사해 ConnectEx API를 생성하려는 호출인지 판단한다.
        *(DWORD*)lpvOutBuffer = (DWORD)MyConnectex;
        //정상적으로 생성된 ConnectEx의 주소가 들어있는 포인터를 MyConnectEx함수의 주소로 지정해준다.
    }
    return ret;
}
BOOL WSAAPI MyConnectex(
    SOCKET s,
    const sockaddr* name,
    int namelen,
    PVOID lpSendBuffer,
    DWORD dwSendDataLength,
    LPDWORD lpdwBytesSent,
    LPOVERLAPPED lpOverlapped
) {
    BOOL ret = ((PFMyConnectex)Org_Connectex)(
        s, 
        name,
        namelen, 
        lpSendBuffer, 
        dwSendDataLength, 
        lpdwBytesSent, 
        lpOverlapped
    );
    OutputDebugString(TEXT("ConnectEx API Hooking Success"));
 
    return ret;
}
int main(){
    if(GetConnectEx()){//ConnectEx API를 생성한다.
        hook_code(TEXT("Ws2_32.dll"), TEXT("WSAIoctl"), MyWSAIoctl);
        //WSAIoctl API를 후킹한다.
    }
}

'C&C++' 카테고리의 다른 글

CMake Syntax  (0) 2020.01.21
How To Use Unit Test In C++  (0) 2020.01.13
WINAPI DLL_PROCESS_DETACH  (1) 2019.08.08
WINAPI Options Using Bit Flag  (0) 2019.08.07
WINAPI CreateProcess Suspend State  (0) 2019.07.11
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved){
    switch (fdwReason)  {
    case DLL_PROCESS_ATTACH:
        LogFileOpen();
        break;
    case DLL_PROCESS_DETACH:
        LogFileClose();
        break;
    }
    return TRUE;
}
 
나는 해당 DLL을 로드한 프로세스가 종료될때 까지 다른 프로세스가 로그파일에 접근하지 못하도록 DLL이 로드될때 파일을 열고 DLL이 언로드될때 파일 핸들을 닫아주도록 구현을 하려고 했다.
그러나 위 방식대로 구현을 하니 DOS 명령 kill의 경우나 Procexp로 프로세스를 죽이는 경우에는 로그파일이 제대로 생성되지 않았다.
즉 외부 프로세스가 강제로 해당 DLL을 로딩중인 프로세스를 끄려고 할 경우 DLL_PROCESS_DETACH로 들어가지 않는것을 확인했다.
 
MSDN에서는 FreeLibrary API를 통해 가상메모리에서 언로드 될때 그리고 프로세스가 종료될때 DLL_PROCESS_DETACH에 들어온다고 정의한다.
그래서 당연히 나는 어떤식으로든 프로세스가 종료되면 DLL_PROCESS_DETACH로 들어갈거라고 생각했다.
그러나 MSDN에서 정의하는 프로세스가 종료되는 순간은 지극히 정상종료를 의미하는것 같다.
 
프로세스가 정상적으로 종료될때 메인스레드에 진입하는 메인함수들(main, winmain, dllmain…)이 반환되면 CRT(C/C++ RunTime Library)에 제어권이 돌아가고 OS에서 메인스레드 종료작업 후 메인함수의 리턴값을 가지고 ExitProcess API를 호출한다.
이 ExitProcess API에서 해당 프로세스에 로드된 DLL들에 DLL_PROCESS_DETACH를 인자로 dllmain을 호출한다고 한다. 
그러나 TerminateProcess API를 통해 프로세스를 죽이면 CRT로 제어권이 안돌아갈 뿐더러 어떠한 메인스레드 종료작업도 하지못하고 그냥 강제로 종료된다.
 
 
MSDN에서 또한 TerminateProcess를 사용한다면 DLL에서 사용되는 데이터를 보장받지 못할것이라고 말한다.
 
결론은 타 프로세스가 TerminateProcess를 통해 프로세스를 종료시키면 DLL_PROCESS_DETACH로 안들어간다.
 
VOID WaitThreadStart() {
    hEvent = OpenEvent(EVENT_ALL_ACCESS, TRUE, TEXT("finish"));
    hThread = (HANDLE)_beginthreadex(NULL, 0, WaitForExit, NULL, 0, (unsigned*)& dwThreadID);
}
UINT WINAPI WaitForExit(LPVOID lpParam){
    WaitForSingleObject(hEvent, INFINITE);
 
    ExitProcess(1);
}
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved){
    switch (fdwReason)  {
    case DLL_PROCESS_ATTACH:
        LogFileOpen();
        WaitThreadStart();
        break;
    case DLL_PROCESS_DETACH:
        LogFileClose();
        OutputDebugString(TEXT("finish"));
        break;
    }
    return TRUE;
}
 
내 문제에서는 DLL을 인젝션 해주는 에이전트 프로세스에서 CreateEvent로 이벤트를 만들고 종료를 원하는 순간 SetEvent를 통해 해당 DLL을 로드한 모든 프로세스가 종료되도록 함으로써 
DLL을 로드한 모든 프로세스가 실행중인 동안에는 타 프로세스의 로그파일 접근을 막았지만 이건 내 문제에서 이벤트를 통한 해결방법이다.
 
DLL_PROCESS_DETACH를 너무 믿지 말자.
 

'C&C++' 카테고리의 다른 글

How To Use Unit Test In C++  (0) 2020.01.13
WINAPI How To Hook ConnectEx  (0) 2019.11.06
WINAPI Options Using Bit Flag  (0) 2019.08.07
WINAPI CreateProcess Suspend State  (0) 2019.07.11
Incremental Linking  (0) 2019.07.10
비트 플래그 (Bit Flag)
비트 플래그란 한 비트를 플래그로 한 비트에 하나의 상태가 참이냐 거짓이냐를 따지는 것이다.
메모리는 바이트 단위이므로 최소 8비트를 가진다. 이 말은 8개의 조건이 참이냐 거짓이냐를 가질수 있다는 말이다.
0000 0000 에서 한 비트를 플래그로 생각했을때 1이면 on, 0이면off이 된다.
EX)
0000 0001 = 0x01 = 사과
0000 0010 = 0x02 = 딸기
0000 0100 = 0x04 = 참외
0000 1000 = 0x08 = 자두
0001 0000 = 0x10 = 수박
0010 0000 = 0x20 = 복숭아
0100 0000 = 0x40 = 바나나
1000 0000 = 0x80 = 배
 
옵션이 이렇게 있을때 사과와 자두 그리고 복숭아를 골랐다고 표현한다면 
(0000 0001) | (0000 1000) | (0100 0000) = (0100 1001) 
이런식으로 표현할 수 있다.
 
Windows API에서 제공하는 다양한 옵션들 또한 대부분 비트 플래그를 통해 지정된다.
 
NtCreateFile에서 DesiredAccess 파라미터를 예로 들어보겠다.
<winnt.h> 헤더의 정의를 보면 ACCESS_MASK는 DWORD 타입이고 DWORD는 4바이트 자료형이기 때문에 최대 32가지의 옵션을 지정할 수 있다.
위 목록은 MSDN에서 제공하는 ACCESS_MASK flag 목록의 일부이고
아래의 매크로는 <winnt.h> 헤더에서 실제로 옵션들을 정의하는 코드의 일부다.
NtCreateFile 호출시 보통 여러개의 옵션을 OR 연산으로 지정해주기 때문에 위 비트플래그 설명을 본다면 이해가 된다.
나는 NtCreateFile 후킹시 인자로 들어온 DesiredAccess의 값에 FILE_WRITE_ATTRIBUTES 옵션이 들어왔는지를 알아야 했다.
WinAPI에서 옵션이 어떠한 원리로 지정되는지 몰라서 처음에는 이 기능을 제공하는 WinAPI가 있나보다 하고 엄청 찾아봤다.
하지만 비트플래그 원리를 통해 옵션이 지정된다는 것을 알고 간단히 비트마스크 연산을 통해 해결했다.

'C&C++' 카테고리의 다른 글

WINAPI How To Hook ConnectEx  (0) 2019.11.06
WINAPI DLL_PROCESS_DETACH  (1) 2019.08.08
WINAPI CreateProcess Suspend State  (0) 2019.07.11
Incremental Linking  (0) 2019.07.10
WINAPI VirtualAlloc and VirtualAllocEx Difference  (0) 2019.07.09
[프로세스 시작전에 DLL Injection을 하는 방법]
 
CreateProcess API를 suspend 상태로 실행시키면 메모리에 프로세스가 올라가지만
suspend 상태이기 때문에 메인스레드가 실행은 되지 않는다.
이때 프로세스에 DLL Injection을 해주고 완료가 되면 ResumeThread API를 통해 대기중인 메인스레드를 실행시킨다.
 
프로세스 대기 상태
suspend 상태에서 메모리를 보면 아무것도 없다.
위 코드를 보면 suspend상태에서 5초후 실행된다.
이때 notepad.exe에서 필요한 dll이 로딩되지 않는 이유는 프로세스가 올라갈 메모리만 할당되고
프로세스 내 메인 스레드가 실행되지 않기 때문에 notepad.exe의 IAT에 있는 dll들이 올라가지 않지만
Suspend 상태가 해제되면 메인스레드가 실행되고 notepad.exe가 정상적으로 실행되며 DLL 또한 로딩된다.
 

프로세스 실행 후

 

프로세스 suspend 상태에서 DLL Injection을 통해 IAT를 동적으로 후킹하면 프로세스 시작전 후킹을 할 수 있다.

'C&C++' 카테고리의 다른 글

WINAPI DLL_PROCESS_DETACH  (1) 2019.08.08
WINAPI Options Using Bit Flag  (0) 2019.08.07
Incremental Linking  (0) 2019.07.10
WINAPI VirtualAlloc and VirtualAllocEx Difference  (0) 2019.07.09
Features And Uses Of STL Container In C++  (0) 2018.11.04
 

해당 코드를 실행하면 함수 포인터를 출력한다.
하지만 해당 주소에 가보면 실제 함수 프롤로그가 아닌 실제 주소로 점프하도록 하는 코드가 있다.
MyFunc 함수 이외에도 위 주소 근처를 보면 해당 프로그램에서 사용하는 모든 함수들로 점프하는 코드들이 있다.
원래는 함수포인터는 함수의 시작주소를 갖는게 일반적이지만 위 모습은 일반적이지 않은 모습이다.
이 문제에 대한 원인은 (문제는 아니지만…) 증분 링크 옵션때문이다.
증분링크(Incremental Link)란 
 
매번 컴파일할 때 마다 링킹과정이 계속 반복 되는데 처음 컴파일 할 때 함수 주소들을 테이블에 보관하고 
다음 컴파일시 수정된것만 다시 계산하여 좀더 빠르게 링킹을 하도록 하는 기능이다.
증분링킹이 진행되면 초기 ilk파일이 생기고 저 ilk 파일을 토대로 빠른 링킹과정이 진행된다.
저 파일에 문제가 생기면 Visual Studio는 경고를 하고 비증분 링킹으로 전환한다.
 
warning LNK4076: invalid incremental status file '*.ilk'; linking nonincrementally
프로젝트 속성에 들어가 증분링크 옵션을 꺼두고 다시 컴파일한다.
증분링크 옵션을 꺼두고 다시 컴파일하니 함수 프롤로그가 잘 나타나는것을 확인할 수 있다.

 

'C&C++' 카테고리의 다른 글

WINAPI Options Using Bit Flag  (0) 2019.08.07
WINAPI CreateProcess Suspend State  (0) 2019.07.11
WINAPI VirtualAlloc and VirtualAllocEx Difference  (0) 2019.07.09
Features And Uses Of STL Container In C++  (0) 2018.11.04
Modern C++  (1) 2018.10.02
x86 아키텍쳐에서 사용하는 jmp 명령어는 다음과 같다.
이중 상대주소로 점프하는 Opcode는 0xEB, 0xE9이다.
메모리 패치를 위해 어셈블리 하드코딩을 할때 필요하기 때문에 operand에 들어갈 상대주소를 계산해보겠다.
 

 

[0xEB 0xXX]
0xEB는 1Byte operand를 받는다. 
현재 명령어 주소에  이 1Byte를 더한 후  2를 더해주면 jmp 명령이 행해질 주소가 나온다. (Opcode(1byte) + operand(1byte))
operand에 들어갈 값은 1Byte인데 양수는 0x00 ~ 0x7F (0 ~ 127), 음수는 0x80 ~ 0xFF( -128 ~ -1)로 해석한다.
127 or -128 범위에 벗어나는 만큼 점프하려면 0xE9 Opcode를 사용한다.
0xAB2489 주소에서 0xEB 0x84 명령이 실행되면 0xAB2489 + (0x84 => -124 => -0x7C) + 2 = 0xAB240F의 주소로 점프한다.
 

 

[0xE9 0xXX 0xXX 0xXX 0xXX]
0xE9는 2 / 4 Byte operand를 받는다.
현재 명령어 주소에 이 4Byte를 더한 후 5를 더해주면 jmp 명령이 행해질 실제주소가 나온다. (Opcode(1byte) + operand(4byte))
0xAB251E에 -0x92(0xFFFFFF6E는 2의보수)를 더하고  5를 더해주면 점프할 실제주소(0xAB2491)를 구할 수 있다.
 
 
위 계산법을 이용하여 후킹코드를 작성하면서 내 후킹 함수의 주소로 점프하도록 할때 
내 후킹 함수 주소 - 현재 명령어 주소 - 5[sizeof(e9 xx xx xx xx)]의 계산이 나온다.

'리버싱' 카테고리의 다른 글

How To See opcode using IDA  (0) 2017.02.10
Z3 SMT Solver for windows using visual studio  (0) 2017.02.08
Remote Linux Debugging using IDA  (0) 2016.08.25
LPVOID VirtualAlloc( LPVOID lpAddress, SIZE_T dwSize, DWORD flAllocationType, DWORD flProtect );
 
LPVOID VirtualAllocEx( HANDLE hProcess, LPVOID lpAddress, SIZE_T dwSize, DWORD flAllocationType, DWORD flProtect );
 
함수 원형에서 확인 가능하듯이 차이점은 프로세스 핸들이 인자로 들어가느냐 안들어가느냐에 있다.
 
다시말해 VirtualAlloc 함수는 내 프로세스에, VirtualAllocEx 함수는 타 프로세스에도 메모리 공간을 할당할 수 있다.
 
[인자 설명]
LPVOID lpAddress
  • 할당할 메모리의 절대주소
  • NULL을 인자로 주면 시스템에서 알아서 할당
 
SIZE_T dwSize
  • 할당할 메모리의 크기
  • NULL을 인자로 주면 하나의 페이지 크기만큼 할당
 
DWORD flAllocationType
  • 할당 방법 지정
  • MEM_COMMIT : 물리 메모리의 할당을 확정
  • MEM_RESERVE : 물리 메모리의 할당 없이 예약만 한다
  • ETC : 나머지 옵션은 MSDN
 
DWORD flProtect
 
Return Value
  • 성공시 할당한 메모리 주소를 반환
  • 실패시 NULL 반환
 

'C&C++' 카테고리의 다른 글

WINAPI CreateProcess Suspend State  (0) 2019.07.11
Incremental Linking  (0) 2019.07.10
Features And Uses Of STL Container In C++  (0) 2018.11.04
Modern C++  (1) 2018.10.02
WINAPI How To Load Windows Tray Icon  (0) 2018.09.08
개념
=> Disjoint Set(서로소 집합) 자료구조를 표현할때 사용하는 알고리즘.
=> 여러 노드가 존재할 때, 두개의 노드를 선택해서 두개의 노드가 하나의 집합에 포함되어 있는지 판별하는 알고리즘
=> Kruskal 알고리즘의 사이클 체크에 사용된다.
=> 다양한 자료구조를 사용해 구현할수 있지만, 가장 편한 트리의 개념을 통해 구현한다.

연산
init()
=> 각 노드를 독립노드로 초기화한다.

find(x)
=> x가 어떤 집합에 포함되어있는지 알려준다.

union(x, y)
=> x가 속한 집합과, y가 속한 집합을 합친다.

if (find(x) == find(y))
std::cout << "같은 집합에 속해있다.";
else
std::cout << "다른 집합에 속해있다";

구현
=>배열의 개념을 통해 구현(각 노드의 대표번호)
//Union Find
#include <iostream>
#include <vector>
#define NUM 7
using namespace std;
vector<int> root;
void Init(int n){
root.resize(n);
for (int i = 1; i < n; i++)
root[i] = i;
}
int Find(int x){
return root[x];
}
void Union(int x, int y){//y를 대표번호로 하는 모든 노드의 대표번호를 모두 x로 설정
int tmp = root[y];
for (int& i : root){
if (i == tmp)
i = root[x];
}
}
int main(){
Init(NUM + 1);
Union(2, 3);
Union(2, 4);
Union(4, 7);
Union(1, 2);
if (Find(1) == Find(7))
cout << "같은 집합에 속해있다." << endl;
else
cout << "다른 집합에 속해있다." << endl;
}



=> 트리의 개념을 통해 구현(각 노드의 루트노드)
//Union Find
#include <iostream>
#include <vector>
#define NUM 7
using namespace std;
vector<int> root;
void Init(int n){
root.resize(n);
for (int i = 1; i < n; i++)
root[i] = i;
}
int Find(int x){//모든 자식노드의 루트를 x로 설정하면서 찾기
if (x == root[x])
return x;
else
return root[x] = Find(root[x]);//경로압축 최적화
}
void Union(int x, int y){
root[Find(y)] = Find(x);//x노드의 자식노드로 y노드를 추가
}
int main(){
Init(NUM + 1);
Union(2, 3);
Union(2, 4);
Union(4, 7);
Union(1, 2);
if (Find(1) == Find(7))
cout << "같은 집합에 속해있다." << endl;
else
cout << "다른 집합에 속해있다." << endl;
}


'알고리즘' 카테고리의 다른 글

BFS(Breadth first search)  (0) 2019.03.08
DFS(Depth First Search)  (0) 2019.03.08
Pseudo code

BFS(v)
    for(i=0;i<n;i++)
        visited[i] = false
    Queue = creaetQueue()
    Visited[v]  = true
    v 방문
    while(not isEmpty(Queue))
        While(visited[v의 인접정점 w] == false)
            Visited[w] = true
            W 방문
            Queue.push(w)
        v = q.pop()



//BFS(인접리스트) 내가 구현한 코드
#include <iostream>
#include <vector>
#include <queue>
using namespace std;
/*7 7 1
1 2
1 3
2 4
2 5
3 6
3 7
6 7
*/
class graph{
public:
int vertex;//정점의 갯수
vector<vector<int>> edge;//간선 목록
vector<bool> visited;//방문 여부
queue<int> q;
graph(int n){
vertex = n;
edge.resize(n + 1);
visited.resize(n + 1, false);
}
void bfs(int v){
cout << v << " ";
visited[v] = true;
q.push(v);
while (!q.empty()){
v = q.front();
for (auto i : edge[v]){
if (!visited[i]){
cout << i << " ";
visited[i] = true;
q.push(i);
}
}
q.pop();
}
}
};
int main(){
int n, m, v, i, a, b;
graph *g;
cin >> n >> m >> v;
g = new graph(n);
for (i = 0; i < m; i++){
cin >> a >> b;
g->edge[a].push_back(b);
g->edge[b].push_back(a);
}
for (i = 1; i <= n; i++)
sort(g->edge[i].begin(), g->edge[i].end());
g->bfs(v);
}


'알고리즘' 카테고리의 다른 글

Union - Find  (0) 2019.03.08
DFS(Depth First Search)  (0) 2019.03.08
Pseudo code

DFS(v)
    for(i=0;i<n;i++)
        visited[i] = false
    Stack = creaetStack()
    Visited[v]  = true
    v 방문
    while(not isEmpty(stack))
        If(visited[v의 인접정점 w] == false)
            push(stack, v)
            visited[w] = true
            w 방문
            v = w
        Else
            v = pop(stack)



인접 리스트(재귀)


//DFS(인접리스트) 내가 구현한 코드 [재귀]
#include <iostream>
#include <vector>
using namespace std;
/*7 7 1
1 2
1 3
2 4
2 5
3 6
3 7
6 7
*/
class graph{
    public:
        int vertex;              //정점의 갯수
        vector<vector<int>> edge; //간선 목록
        vector<bool> visited;    //방문 여부
        graph(int n){
            vertex = n;
            edge.resize(n + 1);
            visited.resize(n + 1, false);
        }
        void dfs(int v){
            cout << v << " ";
            visited[v] = true;
            for (auto i : edge[v]){
                if (!visited[i])
                    dfs(i);
            }
        }
};
int main(){
    int n, m, v, i, a, b;
    graph* g;
    cin >> n >> m >> v;
    g = new graph(n);
    for (i = 0; i < m; i++){
        cin >> a >> b;
        g->edge[a].push_back(b);
        g->edge[b].push_back(a);
    }
    g->dfs(v);
}




인접리스트(스택)


//DFS(인접리스트) 내가 구현한 코드
//O(v+e)
#include <iostream>
#include <stack>
#include <vector>
#include <algorithm>
using namespace std;
/*
7 7 1
1 2
1 3
2 4
2 5
3 6
3 7
6 7
*/
class graph{
    public:
        int size;                //vertex 갯수
        vector<vector<int>> head; //각 vertex에 연결된 vertex들이 저장된 배열 (인접리스트)
        vector<int> visited;     //방문했는지
        graph(int size);
};
graph::graph(int size){
    this->size = size;
    this->head.resize(size + 1);
    this->visited.resize(size + 1, false);
}
void printGraph(graph* g){
    for (int i = 1; i <= g->size; i++){
        cout << i << " : ";
        for (auto j : g->head[i])
            cout << j << " ";
        cout << endl;
    }
}
void dfs(graph* g, int v){
    stack<int> s;
    bool flag;
    s.push(v);
    cout << v << " ";
    g->visited[v] = true;
    while (1){
        flag = true;
        for (auto i : g->head[v]){
            if (!g->visited[i]){ //해당 vertex를 방문 안했다면
                s.push(i);
                cout << i << " ";
                g->visited[i] = true;
                v = i; //해당 vertex의 head에서부터 다시 탐색하겠다는 의미
                flag = false;
                break;
            }
        }
        if (flag){ //flag가 true가 되는경우는 해당 vertex에 연결된 vertex들을 모두 방문했을때 stack에서 pop
            if (!s.empty()){
                v = s.top();
                s.pop();
            }
            else
                break;
        }
    }
    cout << endl;
}
int main(){
    int i, n, m, v, a, b;
    graph* g;
    cin >> n >> m >> v;
    g = new graph(n);
    for (i = 1; i <= m; i++){
        cin >> a >> b;
        g->head[a].push_back(b);
        g->head[b].push_back(a); //방향 연결리스트시 삭제
    }
    for (i = 1; i <= m; i++)
        sort(g->head[i].begin(), g->head[i].end()); //탐색시 오름차순으로 탐색할때 정렬해야함
    printGraph(g);
    dfs(g, v);
}




인접행렬(스택)


//DFS(인접행렬) 내가 구현한 코드
//O(v^2)
#include <iostream>
#include <stack>
#include <vector>
using namespace std;
/*
7 7 1
1 2
1 3
2 4
2 5
3 6
3 7
6 7
*/
class graph{
public:
int v;//vertex의 갯수
int **adjArr;//인접 행렬
graph(int v);
};
graph::graph(int v){
this->v = v;
this->adjArr = new int *[v + 1];
for (int i = 0; i < v + 1; i++){
this->adjArr[i] = new int[v + 1];
memset(this->adjArr[i], 0, v + 1);
}
}
void dfs(graph* g, int v){
int i;
stack<int> s;
vector<bool> check;
check.resize(g->v + 1, false);
s.push(v);
check[v] = true;
cout << v << " ";
while (!s.empty()){
v = s.top();
for (i = 1; i < g->v + 1; i++){//방문한 vertex의 모든 vertex을 탐색
if (g->adjArr[v][i] == 1){//방문한 vertex에 연결된 vertex일때
if (check[i] == false){//연결된 vertex가 방문되지 않았을때
s.push(i);
check[i] = true;
cout << i << " ";
v = i; //해당 vertex를 탐색
break;
}
}
}
if (i == g->v + 1)s.pop();//stack의 top에 있는 vertex부터 다시 탐색
}
}
int main(){
int n, m, v, i, a, b;
cin >> n >> m >> v;
graph* g = new graph(n);
for (i = 0; i < m; i++){
cin >> a >> b;
g->adjArr[a][b] = 1;
g->adjArr[b][a] = 1;
}
dfs(g, v);
}


'알고리즘' 카테고리의 다른 글

Union - Find  (0) 2019.03.08
BFS(Breadth first search)  (0) 2019.03.08

+ Recent posts