쪽지발송 성공
Click here
재능넷 이용방법
재능넷 이용방법 동영상편
가입인사 이벤트
판매 수수료 안내
안전거래 TIP
재능인 인증서 발급안내

🌲 지식인의 숲 🌲

🌳 디자인
🌳 음악/영상
🌳 문서작성
🌳 번역/외국어
🌳 프로그램개발
🌳 마케팅/비즈니스
🌳 생활서비스
🌳 철학
🌳 과학
🌳 수학
🌳 역사
해당 지식과 관련있는 인기재능

AS규정기본적으로 A/S 는 평생 가능합니다. *. 구매자의 요청으로 수정 및 보완이 필요한 경우 일정 금액의 수고비를 상호 협의하에 요청 할수 있...

안녕하세요!!!고객님이 상상하시는 작업물 그 이상을 작업해 드리려 노력합니다.저는 작업물을 완성하여 고객님에게 보내드리는 것으로 거래 완료...

30년간 직장 생활을 하고 정년 퇴직을 하였습니다.퇴직 후 재능넷 수행 내용은 쇼핑몰/학원/판매점 등 관리 프로그램 및 데이터 ...

판매자 소개- 한국 정보올림피아드(KOI) / 세계대학생프로그래밍 경시대회(ACM) 출신- 해외 프로그래밍 챌린지 (Topcoder, Codeforces, Codechef, ...

JavaScript 오프라인 기능 구현

2024-12-31 13:00:29

재능넷
조회수 95 댓글수 0

JavaScript 오프라인 기능 구현: 웹의 새로운 지평을 열다! 🚀

 

 

안녕, 친구들! 오늘은 정말 흥미진진한 주제로 찾아왔어. 바로 JavaScript를 이용한 오프라인 기능 구현에 대해 함께 알아볼 거야. 이게 뭔 소리냐고? 간단히 말해서, 인터넷이 끊겨도 웹 앱이 계속 작동할 수 있게 만드는 마법 같은 기술이라고 할 수 있지. 😎

우리가 살고 있는 이 디지털 시대에 인터넷은 거의 필수품이 됐잖아. 하지만 가끔 신호가 약하거나 아예 연결이 끊기는 상황이 있지 않아? 그럴 때마다 웹사이트나 앱이 먹통이 되면 얼마나 답답하겠어. 그래서 등장한 게 바로 오프라인 기능이야. 이걸 구현하면 네트워크 연결 없이도 웹 앱을 계속 사용할 수 있다니, 완전 대박 아니야? 🤯

이런 오프라인 기능은 특히 재능넷 같은 재능 공유 플랫폼에서 엄청 유용할 거야. 예를 들어, 사용자가 지하철을 타고 이동 중에 갑자기 인터넷이 끊겼다고 해도 계속해서 재능 정보를 둘러보거나 자신의 프로필을 수정할 수 있게 되는 거지. 이런 식으로 사용자 경험을 한층 업그레이드할 수 있어!

자, 이제 본격적으로 JavaScript로 오프라인 기능을 어떻게 구현하는지 알아보자. 준비됐어? 그럼 출발~! 🏁

1. 오프라인 기능의 기초: Service Worker 🛠️

오프라인 기능을 구현하는 데 있어 가장 중요한 주인공은 바로 Service Worker야. 이 녀석은 뭐하는 애냐고? 간단히 말해서, 웹 브라우저와 네트워크 사이에서 중간다리 역할을 하는 JavaScript 파일이라고 할 수 있어.

Service Worker는 웹 페이지와는 별도로 백그라운드에서 실행되면서 네트워크 요청을 가로채고, 캐시를 관리하고, 푸시 알림을 처리하는 등의 작업을 수행해. 이런 특성 때문에 오프라인 환경에서도 웹 앱이 동작할 수 있게 만드는 데 아주 중요한 역할을 하는 거지.

그럼 이제 Service Worker를 어떻게 등록하고 사용하는지 살펴볼까? 😃

Service Worker 등록하기

먼저, 웹 앱의 메인 JavaScript 파일에서 다음과 같은 코드를 작성해야 해:


if ('serviceWorker' in navigator) {
  window.addEventListener('load', function() {
    navigator.serviceWorker.register('/sw.js').then(function(registration) {
      console.log('ServiceWorker registration successful with scope: ', registration.scope);
    }, function(err) {
      console.log('ServiceWorker registration failed: ', err);
    });
  });
}
    

이 코드는 브라우저가 Service Worker를 지원하는지 확인하고, 지원한다면 '/sw.js' 파일을 Service Worker로 등록하는 역할을 해.

그 다음으로는 'sw.js' 파일을 만들어야 해. 이 파일에서 실제로 Service Worker의 동작을 정의하게 될 거야. 예를 들어, 캐시할 파일들을 지정하고, 네트워크 요청을 어떻게 처리할지 결정하는 등의 작업을 여기서 수행하지.

기본적인 Service Worker 구조


const CACHE_NAME = 'my-site-cache-v1';
const urlsToCache = [
  '/',
  '/styles/main.css',
  '/script/main.js'
];

self.addEventListener('install', function(event) {
  // 설치 단계에서 수행할 작업
  event.waitUntil(
    caches.open(CACHE_NAME)
      .then(function(cache) {
        console.log('Opened cache');
        return cache.addAll(urlsToCache);
      })
  );
});

self.addEventListener('fetch', function(event) {
  // 네트워크 요청을 가로채서 처리하는 작업
  event.respondWith(
    caches.match(event.request)
      .then(function(response) {
        // Cache hit - return response
        if (response) {
          return response;
        }
        return fetch(event.request);
      }
    )
  );
});
    

위 코드에서 볼 수 있듯이, Service Worker는 주로 'install'과 'fetch' 이벤트를 처리해. 'install' 이벤트에서는 필요한 리소스들을 캐시에 저장하고, 'fetch' 이벤트에서는 네트워크 요청을 가로채서 캐시된 리소스로 응답하거나 네트워크로 요청을 전달하는 역할을 해.

이렇게 Service Worker를 사용하면 오프라인 상태에서도 웹 앱이 동작할 수 있는 기반을 마련할 수 있어. 예를 들어, 재능넷 같은 사이트에서 사용자가 이전에 본 재능 정보나 자신의 프로필 정보를 오프라인 상태에서도 볼 수 있게 되는 거지. 완전 편리하지 않아? 😊

하지만 이게 끝이 아니야. Service Worker를 제대로 활용하려면 더 많은 것들을 알아야 해. 다음 섹션에서는 좀 더 심화된 내용을 다뤄볼 거야. 준비됐지? 계속 가보자고! 🚀

2. 캐시 전략: 오프라인 기능의 핵심 🗃️

자, 이제 Service Worker의 기본을 알았으니 좀 더 깊이 들어가 볼까? 오프라인 기능을 제대로 구현하려면 캐시 전략을 잘 세워야 해. 캐시 전략이란 쉽게 말해서 "어떤 데이터를 언제 어떻게 저장하고 사용할 것인가"를 결정하는 방법이야.

캐시 전략은 크게 다섯 가지로 나눌 수 있어. 각각의 전략은 상황에 따라 장단점이 있으니, 잘 골라서 사용해야 해. 그럼 하나씩 살펴볼까? 😎

1. Cache Only

이 전략은 말 그대로 캐시에만 의존해. 네트워크는 전혀 사용하지 않고 오직 캐시에 있는 데이터만 사용하는 거야. 항상 같은 내용을 보여줘야 하는 정적 리소스에 적합해.


self.addEventListener('fetch', function(event) {
  event.respondWith(caches.match(event.request));
});
    

2. Network Only

이건 Cache Only의 반대야. 항상 네트워크를 통해 새로운 데이터를 가져오는 전략이지. 실시간 데이터가 중요한 경우에 사용해.


self.addEventListener('fetch', function(event) {
  event.respondWith(fetch(event.request));
});
    

3. Cache First, Falling Back to Network

이 전략은 먼저 캐시를 확인하고, 캐시에 없으면 네트워크로 요청을 보내는 방식이야. 오프라인 기능을 구현할 때 가장 많이 사용되는 전략이지.


self.addEventListener('fetch', function(event) {
  event.respondWith(
    caches.match(event.request)
      .then(function(response) {
        return response || fetch(event.request);
      })
  );
});
    

4. Network First, Falling Back to Cache

이 전략은 먼저 네트워크로 요청을 보내고, 실패하면 캐시를 사용해. 최신 데이터를 우선으로 하면서도 오프라인 기능을 지원하고 싶을 때 사용하기 좋아.


self.addEventListener('fetch', function(event) {
  event.respondWith(
    fetch(event.request)
      .catch(function() {
        return caches.match(event.request);
      })
  );
});
    

5. Stale-While-Revalidate

이 전략은 캐시된 응답을 즉시 반환하면서 동시에 네트워크 요청을 보내 캐시를 업데이트해. 빠른 응답과 최신 데이터 유지를 동시에 할 수 있는 방법이야.


self.addEventListener('fetch', function(event) {
  event.respondWith(
    caches.open('my-cache').then(function(cache) {
      return cache.match(event.request).then(function(response) {
        var fetchPromise = fetch(event.request).then(function(networkResponse) {
          cache.put(event.request, networkResponse.clone());
          return networkResponse;
        });
        return response || fetchPromise;
      });
    })
  );
});
    

와, 꽤 많은 전략들이 있지? 😅 이 중에서 어떤 전략을 선택할지는 네 앱의 특성과 사용자의 니즈에 따라 달라질 거야. 예를 들어, 재능넷에서는 사용자의 프로필 정보나 자주 보는 재능 목록 같은 건 'Cache First' 전략을 사용할 수 있겠지. 반면에 실시간 채팅이나 알림 같은 기능은 'Network First' 전략이 더 적합할 거야.

그리고 잊지 말아야 할 게 있어. 캐시도 관리가 필요해! 오래된 데이터는 삭제하고 새로운 데이터로 업데이트해야 하거든. 이를 위해 Service Worker의 'activate' 이벤트를 활용할 수 있어. 예를 들면 이렇게:

캐시 정리하기


self.addEventListener('activate', function(event) {
  var cacheWhitelist = ['my-site-cache-v2'];

  event.waitUntil(
    caches.keys().then(function(cacheNames) {
      return Promise.all(
        cacheNames.map(function(cacheName) {
          if (cacheWhitelist.indexOf(cacheName) === -1) {
            return caches.delete(cacheName);
          }
        })
      );
    })
  );
});
    

이 코드는 새로운 버전의 Service Worker가 활성화될 때 오래된 캐시를 삭제하는 역할을 해. 이렇게 하면 항상 최신의 데이터를 유지할 수 있지.

자, 여기까지 캐시 전략에 대해 알아봤어. 이제 좀 감이 잡히니? 😊 하지만 아직 끝이 아니야. 오프라인 기능을 더 멋지게 만들려면 또 다른 중요한 요소들이 있어. 다음 섹션에서 계속해서 알아보자고!

3. IndexedDB: 오프라인 데이터의 보물창고 💎

자, 이제 우리의 웹 앱은 Service Worker와 캐시 전략 덕분에 기본적인 오프라인 기능을 갖추게 됐어. 하지만 여기서 만족하긴 이르지! 더 강력한 오프라인 경험을 제공하려면 IndexedDB라는 녀석을 알아야 해. 😎

IndexedDB는 뭐냐고? 간단히 말해서 브라우저에 내장된 로컬 데이터베이스야. 캐시와는 달리 구조화된 데이터를 저장하고 검색할 수 있어서 복잡한 데이터를 다루기에 아주 좋아. 예를 들어, 재능넷에서 사용자의 프로필 정보, 관심 있는 재능 목록, 최근 본 재능 등을 저장하고 관리하는 데 아주 유용하지.

그럼 이제 IndexedDB를 어떻게 사용하는지 살펴볼까? 🤓

IndexedDB 열기

먼저 데이터베이스를 열어야 해:


let db;
const request = indexedDB.open('MyDatabase', 1);

request.onerror = function(event) {
  console.log('데이터베이스 열기 실패');
};

request.onsuccess = function(event) {
  db = event.target.result;
  console.log('데이터베이스 열기 성공');
};

request.onupgradeneeded = function(event) {
  db = event.target.result;
  const objectStore = db.createObjectStore('talents', { keyPath: 'id' });
};
    

이 코드는 'MyDatabase'라는 이름의 데이터베이스를 열고, 'talents'라는 객체 저장소를 만들어. 여기에 재능 정보를 저장할 수 있겠지?

데이터베이스를 열었으니 이제 데이터를 추가, 읽기, 수정, 삭제하는 방법을 알아보자.

데이터 추가하기


function addTalent(talent) {
  const transaction = db.transaction(['talents'], 'readwrite');
  const objectStore = transaction.objectStore('talents');
  const request = objectStore.add(talent);
  
  request.onerror = function(event) {
    console.log('재능 추가 실패');
  };
  
  request.onsuccess = function(event) {
    console.log('재능 추가 성공');
  };
}

// 사용 예시
addTalent({ id: 1, name: '웹 개발', category: 'IT' });
    

데이터 읽기


function getTalent(id) {
  const transaction = db.transaction(['talents']);
  const objectStore = transaction.objectStore('talents');
  const request = objectStore.get(id);

  request.onerror = function(event) {
    console.log('재능 정보 가져오기 실패');
  };

  request.onsuccess = function(event) {
    if (request.result) {
      console.log('재능:', request.result);
    } else {
      console.log('해당 재능 정보 없음');
    }
  };
}

// 사용 예시
getTalent(1);
    

데이터 수정하기


function updateTalent(talent) {
  const transaction = db.transaction(['talents'], 'readwrite');
  const objectStore = transaction.objectStore('talents');
  const request = objectStore.put(talent);

  request.onerror = function(event) {
    console.log('재능 정보 수정 실패');
  };

  request.onsuccess = function(event) {
    console.log('재능 정보 수정 성공');
  };
}

// 사용 예시
updateTalent({ id: 1, name: '웹 개발', category: 'IT', level: '중급' });
    

데이터 삭제하기


function deleteTalent(id) {
  const transaction = db.transaction(['talents'], 'readwrite');
  const objectStore = transaction.objectStore('talents');
  const request = objectStore.delete(id);

  request.onerror = function(event) {
    console.log('재능 정보 삭제 실패');
  };

  request.onsuccess = function(event) {
    console.log('재능 정보 삭제 성공');
  };
}

// 사용 예시
deleteTalent(1);
    

와, 이렇게 하면 우리의 웹 앱은 진정한 오프라인 파워를 갖추게 되는 거야! 🦸‍♂️ 예를 들어, 재능넷 사용자가 지하철에서 인터넷 연결 없이도 자신이 관심 있는 재능 목록을 볼 수 있고, 새로운 재능을 추가하거나 기존 재능 정보를 수정할 수 있게 되는 거지. 그리고 나중에 인터넷에 연결되면 이 변경사항들을 서버와 동기화할 수 있어.

하지만 주의할 점이 있어. IndexedDB는 강력하지만, 사용자의 민감한 정보를 저장하는 데는 적합하지 않아. 왜냐하면 클라이언트 측 저장소는 보안에 취약할 수 있거든. 그래서 비밀번호나 결제 정보 같은 중요한 데이터는 절대로 여기에 저장하면 안 돼!

자, 여기까지 IndexedDB에 대해 알아봤어. 이제 우리의 웹 앱은 훨씬 더 강력한 오프라인 기능을 갖추게 됐지? 😊 하지만 아직 끝이 아니야. 다음 섹션에서는 이 모든 것들을 어떻게 조화롭게 사용할 수 있는지, 그리고 사용자에게 더 나은 경험을 제공하기 위한 추가적인 팁들을 알아볼 거야. 계속 가보자고! 🚀

4. 오프라인 경험 개선하기: 사용자를 위한 마법 🧙‍♂️

자, 이제 우리는 Service Worker, 캐시 전략, 그리고 IndexedDB까지 알아봤어. 이것들을 잘 조합하면 꽤 괜찮은 오프라인 기능을 구현할 수 있지. 하지만 진정한 마법사라면 여기서 만족하면 안 돼! 사용자에게 더 나은 경험을 제공하기 위해 우리가 할 수 있는 일들이 아직 많이 남아있거든. 어떤 것들이 있는지 한번 살펴볼까? 🤔

1. 오프라인 상태 감지 및 알림

사용자가 오프라인 상태가 됐을 때 이를 알려주는 게 좋아. 그래야 사용자가 현재 상황을 이해하고 그에 맞게 행동할 수 있거든. 다음과 같은 코드를 사용할 수 있어:


window.addEventListener('online', updateOnlineStatus);
window.addEventListener('offline', updateOnlineStatus);

function updateOnlineStatus(event) {
  const condition = navigator.onLine ? "온라인" : "오프라인";
  const status = document.getElementById("status");
  status.textContent = condition;
  status.className = condition;

  if (condition === "오프라인") {
    showNotification("인터넷 연결이 끊겼습니다. 오프라인 모드로 전환합니다.");
  } else {
    showNotification("인터넷에 다시 연결되었습니다.");
  }
}

function showNotification(message) {
  const notification = document.createElement('div');
  notification.textContent = message;
  notification.style.position = 'fixed';
  notification.style.top = '10px';
  notification.style.right = '10px';
  notification.style.padding = '10px';
  notification.style.backgroundColor = '#333';
  notification.style.color = '#fff';
  notification.style.borderRadius = '5px';
  document.body.appendChild(notification);

  setTimeout(() => {
    document.body.removeChild(notification);
  }, 3000);
}
    

이렇게 하면 사용자는 자신의 연결 상태를 항상 알 수 있게 돼. 재능넷에서 이 기능을 사용하면 사용자들이 오프라인 상태에서도 안심하고 서비스를 이용할 수 있겠지? 😊

2. 백그라운드 동기화

사용자가 오프라인 상태에서 어떤 작업을 수행했을 때, 다시 온라인이 되면 자동으로 그 작업을 서버와 동기화하는 기능을 구현할 수 있어. 이를 위해 Background Sync API를 사용할 수 있지. 예를 들어:


// Service Worker에서
self.addEventListener('sync', function(event) {
  if (event.tag === 'sync-talents') {
    event.waitUntil(syncTalents());
  }
});

async function syncTalents() {
  const db = await openDB('MyDatabase', 1);
  const talents = await db.getAll('talents');
  
  for (let talent of talents) {
    if (talent.needsSync) {
      try {
        await fetch('/api/talents', {
          method: 'POST',
          body: JSON.stringify(talent)
        });
        talent.needsSync = false;
        await db.put('talents', talent);
      } catch (error) {
        console.error('동기화 실패:', error);
      }
    }
  }
}

// 메인 스크립트에서
navigator.serviceWorker.ready.then(function(swRegistration) {
  return swRegistration.sync.register('sync-talents');
});
    

이렇게 하면 재능넷 사용자가 오프라인 상태에서 새로운 재능을 추가하거나 기존 재능을 수정해도, 나중에 온라인이 되면 자동으로 서버와 동기화가 이루어지는 거야. 완전 편리하지 않아? 😎

3. 점진적 로딩

오프라인 상태에서는 모든 데이터를 한 번에 로드하기 어려울 수 있어. 이럴 때는 점진적 로딩 기법을 사용하면 좋아. 예를 들어:


async function loadTalents(page = 1, limit = 10) {
  const db = await openDB('MyDatabase', 1);
  const talents = await db.getAll('talents', null, limit, (page - 1) * limit);
  
  const container = document.getElementById('talents-container');
  talents.forEach(talent => {
    const element = createTalentElement(talent);
    container.appendChild(element);
  });

  if (talents.length === limit) {
    const loadMoreButton = document.createElement('button');
    loadMoreButton.textContent = '더 보기';
    loadMoreButton.onclick = () => loadTalents(page + 1, limit);
    container.appendChild(loadMoreButton);
  }
}

function createTalentElement(talent) {
  const element = document.createElement('div');
  element.textContent = `${talent.name} - ${talent.category}`;
  return element;
}

// 초기 로딩
loadTalents();
    

이렇게 하면 재능넷 사용자들이 오프라인 상태에서도 스무스하게 재능 목록을 탐색할 수 있어. 데이터를 조금씩 로드하니까 초기 로딩 시간도 빨라지고, 메모리 사용량도 줄일 수 있지. 일석이조야! 👍

4. 오프라인 전용 컨텐츠

오프라인 상태일 때만 볼 수 있는 특별한 컨텐츠를 제공하는 것도 좋은 아이디어야. 예를 들어, 간단한 미니게임이나 유용한 팁 같은 걸 보여줄 수 있지. 이렇게 하면 사용자들이 오프라인 상태를 불편하게 느끼지 않고 오히려 새로운 경험으로 받아들일 수 있어.


function showOfflineContent() {
  const offlineContent = document.getElementById('offline-content');
  offlineContent.style.display = 'block';
  
  const game = document.createElement('div');
  game.textContent = '🎮 오프라인 미니게임을 즐겨보세요!';
  game.onclick = startMiniGame;
  
  const tip = document.createElement('p');
  tip.textContent = '💡 오프라인에서도 이력서를 작성하고 수정할 수 있다는 사실, 알고 계셨나요?';
  
  offlineContent.appendChild(game);
  offlineContent.appendChild(tip);
}

function startMiniGame() {
  // 여기에 간단한 미니게임 로직을 구현
  alert('미니게임 시작!');
}

// 오프라인 상태 감지 시 호출
window.addEventListener('offline', showOfflineContent);
    

이런 식으로 하면 재능넷 사용자들이 오프라인 상태에서도 지루하지 않게 시간을 보낼 수 있겠지? 😄

자, 여기까지 오프라인 경험을 개선하는 여러 가지 방법들을 알아봤어. 이런 기능들을 잘 조합하면 사용자들에게 정말 멋진 경험을 선사할 수 있을 거야. 하지만 잊지 말아야 할 게 있어. 바로 보안이야. 오프라인 기능을 구현할 때는 항상 보안에 신경 써야 해. 민감한 정보는 절대로 클라이언트 측에 저장하지 말고, 데이터를 서버와 동기화할 때는 반드시 인증 절차를 거쳐야 해.

그리고 마지막으로, 사용자 경험을 최우선으로 생각해야 해. 아무리 멋진 기술을 사용해도 사용자가 불편함을 느낀다면 그건 좋은 구현이 아니야. 항상 사용자의 입장에서 생각하고, 테스트를 통해 개선점을 찾아나가는 게 중요해.

여기까지 JavaScript를 이용한 오프라인 기능 구현에 대해 알아봤어. 이제 너희들도 충분히 멋진 오프라인 웹 앱을 만들 수 있을 거야. 화이팅! 🚀🌟

관련 키워드

  • Service Worker
  • 캐시 전략
  • IndexedDB
  • 오프라인 기능
  • 웹 앱
  • JavaScript
  • 백그라운드 동기화
  • 점진적 로딩
  • 사용자 경험
  • 보안

지적 재산권 보호

지적 재산권 보호 고지

  1. 저작권 및 소유권: 본 컨텐츠는 재능넷의 독점 AI 기술로 생성되었으며, 대한민국 저작권법 및 국제 저작권 협약에 의해 보호됩니다.
  2. AI 생성 컨텐츠의 법적 지위: 본 AI 생성 컨텐츠는 재능넷의 지적 창작물로 인정되며, 관련 법규에 따라 저작권 보호를 받습니다.
  3. 사용 제한: 재능넷의 명시적 서면 동의 없이 본 컨텐츠를 복제, 수정, 배포, 또는 상업적으로 활용하는 행위는 엄격히 금지됩니다.
  4. 데이터 수집 금지: 본 컨텐츠에 대한 무단 스크래핑, 크롤링, 및 자동화된 데이터 수집은 법적 제재의 대상이 됩니다.
  5. AI 학습 제한: 재능넷의 AI 생성 컨텐츠를 타 AI 모델 학습에 무단 사용하는 행위는 금지되며, 이는 지적 재산권 침해로 간주됩니다.

재능넷은 최신 AI 기술과 법률에 기반하여 자사의 지적 재산권을 적극적으로 보호하며,
무단 사용 및 침해 행위에 대해 법적 대응을 할 권리를 보유합니다.

© 2025 재능넷 | All rights reserved.

댓글 작성
0/2000

댓글 0개

해당 지식과 관련있는 인기재능

  1.엑셀의 기본기능으로 하기 어렵거나 복잡한 내용 VBA로 자동화 2.셀메뉴형태 또는 리본메뉴형태의 프로그램 3.MY-SQ...

안녕하세요!현직 윈도우 개발자입니다. 진행한 프로젝트 회원관리프로그램 문서관리프로그램 E-book 뷰어& 에디터 등등 ...

서울 4년제 컴퓨터공학과 재학중이며, 대학 연구실에서 학부연구생으로 일하고 있습니다.사용가능한 언어는 C / Objective C / C# /Java / PH...

윈도우 프로그램밍 3년차 개발자 입니다.업무시간이 짧아 남는 시간에 재능이 필요한분께 도움이 되고자 합니다.구매 전 간단한 요구사항 및 금액 ...

📚 생성된 총 지식 11,394 개

  • (주)재능넷 | 대표 : 강정수 | 경기도 수원시 영통구 봉영로 1612, 7층 710-09 호 (영통동) | 사업자등록번호 : 131-86-65451
    통신판매업신고 : 2018-수원영통-0307 | 직업정보제공사업 신고번호 : 중부청 2013-4호 | jaenung@jaenung.net

    (주)재능넷의 사전 서면 동의 없이 재능넷사이트의 일체의 정보, 콘텐츠 및 UI등을 상업적 목적으로 전재, 전송, 스크래핑 등 무단 사용할 수 없습니다.
    (주)재능넷은 통신판매중개자로서 재능넷의 거래당사자가 아니며, 판매자가 등록한 상품정보 및 거래에 대해 재능넷은 일체 책임을 지지 않습니다.

    Copyright © 2024 재능넷 Inc. All rights reserved.
ICT Innovation 대상
미래창조과학부장관 표창
서울특별시
공유기업 지정
한국데이터베이스진흥원
콘텐츠 제공서비스 품질인증
대한민국 중소 중견기업
혁신대상 중소기업청장상
인터넷에코어워드
일자리창출 분야 대상
웹어워드코리아
인터넷 서비스분야 우수상
정보통신산업진흥원장
정부유공 표창장
미래창조과학부
ICT지원사업 선정
기술혁신
벤처기업 확인
기술개발
기업부설 연구소 인정
마이크로소프트
BizsPark 스타트업
대한민국 미래경영대상
재능마켓 부문 수상
대한민국 중소기업인 대회
중소기업중앙회장 표창
국회 중소벤처기업위원회
위원장 표창