기본 콘텐츠로 건너뛰기

[python] python에서 namespace를 써야하나??

파이썬에서는 @staticmethod를 이용하여 네임스페이스처럼 구현을 할 수가 있다

근데 파이썬에서 꼭 네임스페이스를 써야하나라는 생각이 들기시작 했다(자고 일어나다가 왜 이런생각이 난거지 참....)

namespace란?
 - namespace란 함수 이름이 같아도 쓸수 있게 함수마다 공간을 할당을 해주는 것을 의미한다.

네임스페이스를 쓰는 목적만 보면 오 좋은거네? 라고 생각을 할 수가 있다.

그런데 네임스페이스를 쓰는 c++과 python의 모듈 혹은 헤더파일이 작동 방식이 매우 다르다.

c++ 헤더파일 작동 방식 그리고 namespace
(IDE를 설치 되어있지 않아 우분투에서 vim + gcc로 테스트 진행 - 도커를 이러려고 설치한건 아닌데 아무쪼로 잘 쓰는중 ㅋㅋㅋㅋㅋㅋㅋ)

우선 c, c++에서 헤더파일이 어떻게 작동하는지 알아보겠다.

1
2
3
4
5
6
#include <stdio.h>
 int main(void){
     printf("hello world\n");
     return 0;
 }
cs


프로그래밍을 배웠으면 누구나 다 쳐봤다는 hello world!

해당 print부분을 헤더로 분할을 해보겠다.

test.cpp
1
2
3
4
5
6
7
8
9
#include <stdio.h>
#include "pjt.h"
int main(void){
        printf("hello world\n");
        show();
        return 0;
}
cs


pjt.h
1
void show();
cs

pjt.cpp
1
2
3
4
5
6
#include "pjt.h"
#include <stdio.h>
void show(){
        printf("pjt \n");
}
cs


root@a60be428cf16:/home/c_test# ./a.out
hello world
pjt
정상적으로 출력이 된다.(컴파일 과정에서도 에러가 없었다.)

헤더를 더 추가를 해보자.

pht2.h
1
void show();
cs

pjt2.cpp
1
2
3
4
5
6
#include "pjt2.h"
#include <stdio.h>
void show(){
        printf("pjt2 \n");
}
cs

root@a60be428cf16:/home/c_test# ls
pjt.cpp pjt.h pjt2.cpp pjt2.h test.cpp
root@a60be428cf16:/home/c_test# gcc *.cpp
pjt2.cpp:(.text+0x0): multiple definition of `show()'
/tmp/ccu1eXix.o: In function `show()':
collect2: error: ld returned 1 exit status
/tmp/cc9AmFCX.o:pjt.cpp:(.text+0x0): first defined here

pjt2cpp의 show가 여러번 정의 됬다고 컴파일 에러가 발생을하였다.
또한 첫번째 정의된 부분을 pjt.cpp라고 까지 나온다.

헤더가 나눠줬음에도 이미 정의가 내렸다고 에러가 뜨는 이유는 
전처리 과정에서 헤더파일을 하나의 파일로 합치는 과정이 있기 때문이다. 그때문에 함수 명이 곂친다고 에러가 발생을 하는 것이다.
하지만 c++에서는 네임스페이스를 사용하여 해당 에러를 피할 수 있다.

1
2
3
4
5
6
7
8
9
10
11
12
13
#include <stdio.h>
#include "pjt.h"
#include "pjt2.h"
using namespace pjt;
using namespace pjt2 ;
int main(void){
        printf("hello world\n");
        pjt::show();
        pjt2::show();
        return 0;
}
cs

root@a60be428cf16:/home/c_test# gcc *.cpp root@a60be428cf16:/home/c_test# ./a.out hello world pjt pjt2

정상적으로 실행이 된다.

하지만 파이썬에서 import를 하면 해당 파일을 객체화 시켜서 가지고 있게 된다. 즉 cpp처럼 파일이 분리 됬을 때 같은 이름에 의해서 충돌이 일어나지 않는다.

t.py
1
2
def show():
    print("hello world")
cs



a.py
1
2
def show():
    print("hello world")
cs

p.py
1
2
3
4
5
import t
import a
print(id(t), t)
print(id(a), a)
cs


C:\Users\mung\Desktop\etc\study\python>python p.py 12023592 12023752

해당 파일 자체를 객체로들고 있게 된다. 파일 자체가 객체화 되어서 접근을 하기 때문에 이미 공간 부여자체가 다르게 된다. 즉 네임스페이스를 쓸 필요가 없어보인다.

이것은 컴파일 언어에서의 파일을 불러오는것과 스크립트에서 파일을 불러오는 것에 의한 차이라고 보면 된다.
(스크립트에서는 파일을 불러오게 되면 객체화 시킨다 하지만 c나 c++에서는 파일을 객체화 시키지 않는다)

include는 전처리에 의해서 실행이 된다. 즉 컴파일 과정이 아니다.
하지만 import는 프로그램이 실행이 되면 실행이 된다. 해당 파일을객체화 시켜서 가져온다.(아직 import의정확한 개념인지는 잘 모르겠다.)

결론은 파이썬에서는 타 모듈에서 이미 정의가 됬다고 에러를 볼일이 없다.

그래서 파이썬에서 네임스페이스를 꼭 써야하나??라는 의문이 들었고 굳이 안써도 될것 같다는 생각이 든다.

댓글

이 블로그의 인기 게시물

[kali linux] sqlmap - post요청 injection 시도

아래 내용은 직접 테스트 서버를 구축하여 테스트 함을 알립니다.  실 서버에 사용하여 얻는 불이익에는 책임을 지지 않음을 알립니다. sqlmap을 이용하여 get요청이 아닌 post요청에 대해서 injection공격을 시도하자. 뚀한 다양한 플래그를 이용하여 DB 취약점 테스트를 진행을 해보려고 한다. 서버  OS : windows 7 64bit Web server : X Server engine : node.js Framework : expresss Use modules : mysql Address : 172.30.1.30 Open port : 6000번 공격자 OS : kali linux 64bit use tools : sqlmap Address : 172.30.1.57 우선 서버측 부터  1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 var  express  =  require( 'express' ); var  app  =  express(); var  mysql  =  require( 'mysql' ); var  ccc  =  mysql.createConnection({     host: '127.0.0.1' ,     user: 'root' ,     pos...

[git] pull을 하여 최신코드를 내려받자

보면 먼가 로고가 다르게 뜨는것을 확인을 할 수가있다. C:\Users\mung\Desktop\etc\study\python-gene>git checkout remotes/origin/master Note: checking out 'remotes/origin/master'. You are in 'detached HEAD' state. You can look around, make experimental changes and commit them, and you can discard any commits you make in this state without impacting any branches by performing another checkout. If you want to create a new branch to retain commits you create, you may do so (now or later) by using -b with the checkout command again. Example:   git checkout -b HEAD is now at 29e282a... fetch test C:\Users\mung\Desktop\etc\study\python-gene>git branch * (HEAD detached at origin/master)   master   test1   test2 깃이 잘 쓰면 참 좋은놈인데 어지간히 쓰기가 까다롭다. 처음에 깃을 푸시 성공하는데만 한달정도 걸렸던걸로 기억이 난다.. ㅋㅋㅋ 여담으로  깃 프로필을 가면 아래사진 처럼 보인다. 기여도에 따라서 초록색으로 작은 박스가 채워지는데 저걸 잔디라고 표현을 한다고 합니다 ㅎ 저 사진은 제 깃 기여도 사진입니당 ㅋㅋㅋㅋ 다시 본론으로 돌아와서 ㅋㅋ pull을 하면...

[git] git log 확인하기

git log를 통해서 커밋 이력과 해당 커밋에서 어떤 작업이 있었는지에 대해 조회를 할 수 있다. 우선 git에서의 주요 명령어부터 알아보겠다. $ git push [branch name] $ git pull [branch name] 여기서 branch name은 로컬일 경우 해당 브런치 이름만 적으면 되지만 깃허브 원격 저장소로 연결을 원할 경우는 해당 브런치 이름 앞에 꼭 origin을 붙이도록 한다. $ git brnch [branch name] $ git checkout [branch name] branch일경우 해당 브런치를 생성을 한다. 여기서 현재의 브런치를 기준으로 브런치를 따는것이다. checkout은 브런치를 바꾸는 것이다.(HEAD~[숫자]를 이용하면 해당 커밋으로 움직일수 있다.. 아니면 해당 커밋 번호를 통해 직접 옮기는것도 가능하다.) -> 해당 커밋으로 옮기는 것일뿐 실질적으로 바뀌는 것은 없다. 해당 커밋으로 완전히 되돌리려면 reset이라는 명령어를 써야한다. 처음 checkout을 쓰면 매우 신기하게 느껴진다. 막 폴더가 생겼다가 지워졌다가 ㅋㅋㅋㅋㅋ  master 브런치에서는 ht.html파일이 존재하지만 a브런치에서는 존재하지않는다. checkout 으로 변경을 하면 D 로 명시를 해준다.  $ git log 해당 브런치의 커밋 내역을 보여준다. a 브런치의 커밋 내역들이다. (머지 테스트를 하느라 커밋 내용이 거의 비슷하다 ㅋㅋ) master 브런치의 커밋 내역들이다. 커밋 번호, 사용자, 날짜, 내용순으로 등장을 한다. 이건 단순히 지금까지의 내역을 훑어보기 좋다. 좀더 세밀한 내용을 봐보자. $ git log --stat --stat을 붙이면 기존의 로그에서 간략하게...