이번 포스팅에서는 대표적인 언어 C, C++, Java, C# 의 차이에 대해 설명 해 보겠습니다.
먼저, 개인적인 생각과 경험에 나온 기술이기에 제 설명이 틀릴 수도 있습니다.

프로그램 언어의 탄생은 C -> C++, Java, C# 순서입니다. 여기서 C#이 가장 나중에 개발 된 프로그램 언어라 하겠습니다만, 그렇다고 C#이 가장 우수한 언어, 좋은 언어라고 하기에는 오류가 있겠습니다.
먼저 기술적, 개념적으로 언어적 특성이나 언어의 처리속도는 차이가 없습니다.. 다만 문법적으로는 차이가 조금씩은 있습니다.

기술적으로는 차이가 없는 이 언어들을 구분을 지어보겠다면 C, C++(MFC)와 Java, C#으로 나뉠 수 있습니다. 혹자는 1세대, 2세대 언어라고도 표현합니다.
이 두 언어의 가장 큰 차이점은 C, C++은 소스레벨의 코딩이 가능하고, Java, C#는 클래스는 동적으로 움직이며 클래스 기반의 언어(?)라고 할 수 있곘네요.
또 다른 큰 차이는 프레임워크(Java에서는 JVM)이 있고 없고의 차이도 있겠네요

그렇다면 왜 C, C++, Java, C# 언어들은 각기 나뉘어져 있고 또 개발자들은 나뉘어서 개발을 할까요??

제 생각의 가장 큰 이유는 돈 때문입니다. 소프트웨어 공학 등의 학문도 그렇고 실무 프로젝트 구성도 보면 결국 전부 돈과 직결되어 있습니다.
개발을 할 시에 언어를 선택하는 것도 결국은 돈 때문에 선택을 하고 나뉘게 되겠습니다.
단지 C는 오픈언어, Java는 오라클사(과거 Sun), C++, C#은 MS사의 소유권 문제만은 아닌 듯 싶네요.

개발 단가로 살펴 보면은 C와 C++(MFC)의 프로그램 단가가 가장 비쌉니다. 그리고 C#이 비싸고 Java가 가장 싼 프로그램이 되겠네요.
물론 프로그램이 비싸다고 해서 우수한 프로그램은 아니고 Java가 가장 싼 프로그램이라고 해서 허접한 것은 아닙니다. 또 반대로 C와 C++ 개발자라고 해서 보수가 높은 것도 아니고 Java개발자라고 해서 싸구려 개발자라는 뜻도 아닙니다. 그럼 왜 이런 단가가 나누어져 있을까요?

먼저 프로그램 기간을 살펴보면 C와 C++(MFC)은 개발 개간이 깁니다. 반대로 C#과 Java는 C,C++에 비교해서 개발기간이 짧네요. 그렇다 보니 프로그램 가격의 차이가 나옵니다. 이런 생산성을 가진 C#과 Java인데도 불구 하고 아직 시장에는 C와 C++개발 건이 많이 있습니다. 그 이유는 Java와 C#은 프레임워크를 벗어난 네이티브적인 개발은 불가능 하나 C와 C++은 네이티브 개발도 가능하기에 운영체제의 커널이나 하드웨어, 드라이버등의 개발에 용이하기 때문입니다.
그럼 왜 이런차이가 생기는 지에 대해 자세히 설명하겠습니다.


1990년대 초,중반에 과거 Sun사에서는 Java라는 언어를 발표헀습니다. Java의 프로젝트의 목표는 어떠한 플랫폼에서도 개발을 할 수 있는 언어가 목표였습니다.이건 지극히 제 생각인데 C언어도 어떠한 플랫폼에서 개발할 수 있지 않았나 싶은데?? 모르겠네요.
다시 돌아와서 Java가 발표되고 어떤 언어였나 싶으면 C, C++개발자가 보기엔 획기적인 언어였습니다. 현재에서는 하드웨어가 컴퓨터도 그렇지만 통신회선도 그렇고 많은 표준이 생가면서 프로그램 속도, 통신속도가 비약적으로 향상되었지만 1990년대에만 해도 전화모뎀을 쓰는 열약하고 느린 환경에서 JVM머신을 이용하여 이 모든 것이 빠르게 개선 된 것처럼 보이는 것이 아이디어에 놀라웠습니다.
그리고 또 당시 전 세계의 소프트웨어 시장을 잠식하고 있었던 MS에 압박할 수 있었던 무기는 오픈소스 정책이 있었습니다. 기존의 C언어 C++를 생각해 보면 개발자들이 소스를 공유, 모듈을 공유한다는 것에 대해 개념도 없었지만, 서로 정해 놓은 개발서의 형식, 개발 표준이 정해지지 않아서 소스공유의 에로 사항도 많았습니다. 아니 소스 공유에 대한 인식 조차 없었다고 하는게 맞겠네요.
또 프로그램을 보더라도 라이브러리는 DLL로 묶여있어서 해당 모듈 개발자가 아니면 거의 손댈 수 없는 형식이여서 프로젝트가 시작되면 하나부터 열까지 다 개발해야 하는 형식이었습니다. 즉 프로젝트를 할 때마다 똑같은 반복작업이 엄청 많다는 뜻이지요.

그러나 Java는 오픈소스정책, 즉 자주 쓰는 모듈은 공유, 자바독으로 프로그램 사용하는 방법에 대한 기술이 공개가 되고, 또 자바는 개발하는 규칙을 일목요연하게 정리를 해 놓아서 누가 봐도 알 수 있는 소스식으로 나오면서 인기가 얻기 시작합니다.

프로그램을 시작하는 사람으로서는 접근하기 쉬우니 자바에 관심이 가고 기존 개발자들은 반복되는 작업을 생략하고 좀 더 창의적이고 생산적인 개발환경에 마음이 들어 많은 개발자들이 C, C++에서 Java로 갈아타기 시작합니다.

요즘에야 C, C++도 개발 규약이 많이 생겨서 많이 정리가 됐지, 가끔 돌아다니는 90년대 초반 개발 소스를 보면 알아먹을 수가 없습니다. 뭐 이리 복잡하게 만들었는지..

그에 압박을 느끼기 시작한 MS사가 90년대 말 Basic의 생산성, C++ 성능, 델파이의 개념이 결합된 언어라고 C#를 내 놓습니다. (자바언급은 없어요.. 내가 보기엔 자바 판박인데 말이죠..). C#은 그렇게 탄생이 되었습니다.

그렇게 언어들은 탄생이 되고, 90년대의 또 하나의 큰 이슈는 WWW라고 하는 월드 와이드 웹이라는 개념의 웹이 등장하게 됩니다. 이 웹의 등장으로 채택된 언어는 Asp, PHP 가 있으나 Java의 등장으로 웹과 Java가 만나게 됩니다.
이는 Java의 폭발적인 인기를 얻게 되는 또 하나의 이유가 되기도 하는데 웹의 발전 상승과 Java의 발전 상승과 맞물리게 됩니다. 그렇게 등장한 Java와 2000년도 초반의 C#은 인터넷 기반, 웹기반의 프로젝트에서 서로 경쟁하듯이 발전을 하게 됩니다.
C# FrameWork 1.1에서 3.5까지는 Web기반의 프레임워크가 많은 발전을 했습니다.

그런 상황에 놓이다 보니 C와 C++은 90년대 후반부터 입지가 조금씩 밀리기 시작했습니다. Java로 갈아 탄 개발자들이야 프로젝트가 많이 지니 좋지만 갈아타지 못한 사람들은 프로젝트가 줄어드니 달갑지는 않지요. 개인적인 생각이지만 이 때부터 MS개발자와 Java는 웬수가 되는 듯….ㅎㅎ 여담입니다.

그렇게 입지가 좋아진 C, C++은 웹과 인터넷 기반의 프로그램이 아닌 Java와 C#이 접근하기 힘든 커널과 하드웨어, 모바일쪽으로 특화가 되기 시작합니다. 90년대 후반부터 핸드폰의 보급이 시작되는데 이 때 들어가는 언어가 심바오C++로써 C언어와 C++개발자들은 그 쪽으로 특화가 되기 시작합니다. 지금은 모바일도 Java 안드로이드가 점령을 해서 이제는 아예 드라이버, 임베디드, 커널 개발등으로 특화가 되어 이제는 시스템적인 언어로 인식이 박히게 되지요..

다시 돌아와서 2000년대 초반에 Java와 C#은 인터넷의 패권으로 경쟁을 하게 됩니다. 
여기서는 이제 Java와 C#의 차이가 나기 시작합니다.
Java는 개발하기 위해서는 비용이 들지 않습니다. Java언어 자체도 무료고, 대표 개발툴인 이클립스도 무료 입니다. 기동되는 서버는 Linux인데 RedHat을 많이 사용합니다. RedHat자체는 유료이나 그와 같은 무료버전인 CentOs가 있으므로 실질적으로 무료로 개발이 가능합니다. 또 데이터 베이스는 오라클인데 오라클은 가격이 있지만 또 무료로 전환가능한 Mysql, postgre 가 있으므로 이 부분도 비용절감이 가능합니다.
즉 개발에서 릴리즈까지 돈을 한푼도 안쓰고 개발이 가능한다는 뜻입니다. 개발자 인건비, 서버 회선비 정도가 들겠네요.

그에 비해 C#은 C#자체는 무료 입니다만, 대표 개발툴인 Visual Studio는 고가의 개발툴입니다. 또 기동되는 서버는 Window Server인데 이것도 상당히 비싸지요. 데이터 베이스는 MSSQL입니다. 뭐 이건 Mysql로 전환이 가능합니다만 C#은 보통 Mssql를 사용하지요. 자바와 같이 개발자 인건비, 서버 회선비가 든다는 것입니다.

여기까지 오면 프로그램을 만드는 입장에서는 어떤 언어를 선택하게 되는 것일까요?
품질의 아주 큰 차이가 있지 않는 이상은 가격이 싼 Java를 선택하게 되어있습니다. 그래서 그런지 C#은 초반에는 상당한 분전을 하다가 웹 계통은 상당히 밀린 듯한 모습을 보입니다.

그럼 C#을 누가 쓰냐? 하는 질문이 생기게 되겠습니다. C#를 배포하는 MS사는 Java를 압도 할 수 있는 무기가 무었이냐.. 점유율입니다.
웹 쪽에서는 밀리는 듯 하지만 CS 인터넷 기반 프로그램에서는 완전한 강세를 보이고 있지요.개인적으로도 Java Swing은 사용도 어렵고 메모리 릭도 많고 호환성도 떨어지더라구요. 그렇지만 같은 MS계열의 윈도우와 C#은 이야기가 다름니다..

단가는 비쌀지언정 확실한 품질차이가 있기 때문에 WindowForm의 윈도우개발은 C#이 강세를 가져갈 수 밖에 없습니다.

그러나 최근에는 안드로이드와 IOS 애들의 강세로 MS도 많이 주춤거리고 있습니다. Window Mobile도 많은 전략을 내세웠으나 뚜렷한 성장세를 보이고 있지 않는 듯 하구요..

앞으로는 언어대결은 어떻게 흘러갈까요?? 개발하는 입장에서 이를 지켜보는 것도 또 하나의 즐거움이겠습니다.

Posted by 모과이IT
,

기존에 만들어놓은 Frame이 Activity Base로 되어져 있어서 FragmentActivity 의 속성에서 실행되어지는 라이브러리를 적용해야할 상황이 있었다. 그런 상황에서 AppCompatActivity 란를 보니 위의 두 Activity와 FragmentActivity의 모두 상속한것들이었다. 특히 요즘에 이쁜 UI 라이브러리는 Base class for activities that use the support library action bar features. 의 라이브러리를 사용되어지는것이 많기 때문에 기존 Frame을 AppCompatActivity로 전환은 필연적이었다.







그리고 Frame의 BaseActivity를 기존에 Activity에서 AppCompatactivity로 바꾸었다. 이론적으로 상속개념에서는 위의 Activity와 FragmentActivity의 속성을 모두 사용할수가 있다.


전환방법은


1. BaseActivity의 상속자를 Activity에서 AppCompatActivity로 해주었다.


이것만 해주니 실행할때 Runtime 오류가 났다.


2. android:theme="@style/Theme.AppCompat.NoActionBar" 처리

- AndroidManifest.xml 파일에 <application 영역에서 액션바 속성에 대한 처리가 포함되어지 테마를 설정해주어야지된다. 참고로 기존에 Frame에서 ActionBar는 사용하지 않으므로 NoActionBar처리를 해주었다.


위의 1,2 를 수행해주면  Activity -> AppCompatActivity로 전환해서 사용이 가능하다


Posted by 모과이IT
,

http://studyforus.tistory.com/63


위의 사이트에서 설명이 되어져있지만, 위의 내용중 핵심을 정리하자면


루트 폴더 



1. 가상디렉토리 추가

2. 가상디렉토리 추가시에 윈도우7의 계정 이름으로 생성

3. ../윈도우계정/LocaUser/ 실제 폴더로 가서 IIS_USRS 그룹소속의 계정과 "FTP 권한 부여 규칙"에 허용한 규칙의 계정 이름과 동일한 폴더를 만들어 준다


위의 같이 해주면 IIS_USRS 계정에 등록한 계정으로 홈디렉토리로 FTP접속이 되어진다


Posted by 모과이IT
,
  1. http://www.visualsvn.com/downloads/로 가서 최신 SVN command line tool 다운 .
  2. Android Studio 에 File -> Settings -> Version Control -> Subversion 로 들어가서 Use command line client 에 1.에서 다운로드 받은 폴더의 svn.exe의 경로를 세팅.
  3. 메뉴에서 VCS -> Import into Version Control (다른 폴더의 원본 폴더를 import)
  4. Android Studio를 재시작 하고 VCS -> Check from Version Control -> Subversion 에서 SVN URL을 입력하고 계정 입력후 Check Out버튼 클릭 -> Local 폴더 지정한다 (새로운 Local 폴더를 설정


[참고사이트]
http://chiyo85.tistory.com/93


Posted by 모과이IT
,
<resources>

<!-- Base application theme. -->
<style name="AppTheme" parent="android:Theme.Holo.Light.DarkActionBar">
<!-- Customize your theme here. -->

</style>

</resources>



[오류]

Error retrieving parent for item: No resource found that matches the given name 'Theme.AppCompat.Light.DarkActionBar'.


위의 오류를 위의 스크린샷 처럼 styles.xml 파일을 수정해 준다.

Posted by 모과이IT
,

안드로이드 Activity 생명주기


1. Activity : 안드로이드 Application 의 기본 단위로, 사용자 인터페이스를 화면에 표시하고 사용자의 입력을받아서 처리하는 역할을 함


2. 생명 주기 (Life cycle) : Activity 는 시작, 실행, 활성, 비활성화, 정지, 종료되는 일련의 상태를 순환


3. system 은 task의 실행중인 Activity들을 stack으로 관리함

  activity 가 새로 생성되면 stack의 제일 위에 놓여 활성화 됨

  이 상태에서 다른 activity 가 또 생성되면, 기존의 activity는 잠시 가려지고 새 activity가 stack의 제일 위에 배치됨

  stack 제일 위의 activity 가 종료되면 바로 아래쪽에 잇는 activity 가 자연스럽게 활성화 될 것임

  stack 의 activity 는 넣고 빼기만 할 뿐 순서가 바뀌지 않음


4. stack activity 의 세가지 상태

   - 실행 (active, running)

     : 사용자가 직접 사용하는 상태. 스택의 제일 위에 있으며 화면상에서도 제일 위에 있음.

       입력 포커스를 가지며, 사용자의 입력을 직접 처리함

   - 일시 정지 (pause)

     : 포커스는 잃었지만 사용자에게는 보이는 상태

      위쪽에 다른 activity 가 있지만, 화면 전체를 다 가리지 않았거나 반투명한 경우가 이에 해당됨.

      살아있는 상태와 같지만 시스템에 의해 강제 종료될 수도 있음

   - 정지 (stoped)

     : 다른 activity 에 의해 완전해 가려진 상태이며, 사용자 눈에 보이지 않음

      그러나 모든 정보를 다 유지하고 있으므로 언제든 다시 활성화 될 수 있음

      system 은 메모리가 부족하면 정지 상태의 activity를 언제든지 강제 종료할 수 있음

 

5. Activity 의 전체 생명주기






- onCreate()

     : activity 를 초기화 함

      중지했다 재시작 하는 경우라면 activity 의 이전 상태 정보인 bundle 이 전달됨. 이정보대로 재초기화 함

 

   - onRestart()

     : 재시작될 때 호출 됨.

       가시 수명으로 이어지기전 acvitity 처리를 위해 호출. 항상 onStart() 호출 직전에 호출됨

 

   - onStart()

     : 액티비티가 사용자에게 보이기 직전에 호출.

 

   - onResume()

     : 사용자와 상효작용을 하기 직전에 호출됨. 이 단계에서 스택의 제일 위로 올라옴

       activity 스택의 맨 위에 있어서 activity 가 사용자에게 보여지고, 사용자의 입력을 처리할 수 있을 때 호출

 

   _ onPause()

     : 다른 activity 가 실행될 때 호출됨.

       이 단계에서 미저장한 데이터가 있으면 저장하고, 애니메이션은 중지해야 함

       이 메서드가 리턴되어야 새 activity 가 활성화 되므로 시간을 너무 많이 끌어서는 안됨

 

       즉, 활성 수명 끝에서 호출 되는데, 포커스를 잃고 이전의 activity 가 resume 되기 전에 호출됨.

       onPause() 에서는 데이터의 저장, 애니메이션의 중지, cpu 를 소비하는 작업 중단과 같은 일들을 수행해야함

       사용자가 activity 를 떠날 때 다루어야 할 매서드임

       application 실행 도중 전화가 걸려오면 그때까지의 내용을 모두 저장해야 하는데, onPause() 에서 이러한 저장과정이 진행됨

 

   - onStop()

     : 더이상activity 가 스택의 최상위에 있지않으므로 사용자에게 보이지 않게 될 때 호출됨

 

   - onDestory()

     : activity 가 파괴될 때 호출됨

       system 에 의해 강제로 종료되는 것인지 아니면 finish 메서드 호출에 의해 스스로 종료하는 것인지는 infinish 메서드로 조사 가능

       onDestory() 메서드가 호출되면 system 에서 activity 가 존재하지 않음


출처 : http://apponline.tistory.com/entry/%EC%95%88%EB%93%9C%EB%A1%9C%EC%9D%B4%EB%93%9CAndroid-%EC%95%88%EB%93%9C%EB%A1%9C%EC%9D%B4%EB%93%9C-Activity-%EC%83%9D%EB%AA%85%EC%A3%BC%EA%B8%B0




안드로이드 액티비티 생명주기 (Android Activity LifeCycle)

- 액티비티는 효율적인 메모리 관리를 위해 생성, 소멸되는 조건인 생명주기를(Lifecycle) 가지고 있다

- 생명주기는 7가지 단계로 구분되며 다음 순서를 기본으로 진행된다
  onCreate(), onStart(), onResume(), onPause(), onStop(), onRestart() 또는 onDestroy()

- 각 상태에 따라 필요한 작업을 메소드를 오버라이드 하여 처리해주면 된다

- 여러개의 액티비티를 이용중 시스템의 리소스가 부족하면 다음상태의 액티비티를 강제로 종료시킨다
  onDestroy() - 1순위, onStop() - 2순위, onPause() - 3순위

- onPause() 상태의 Callback은(실행완료) 보장하지만, onStop()과 onDestroy()는 보장할 수 없다
  (미완료 상태에서 System이 강제종료 시킬 수 있으므로 되도록 onPause()에서 종료를 대비 해야한다)

- 강제종료된 Activity는 리소스가 확보되면 다시 복구 시킨다

- 강제종료시에는 데이터 복원를 대비하여 onPause() 실행 전 onSaveInstanceState(Bundle b) 를 실행시킨다

- 강제종료시에는 onStart() 호출 후 실행되는 onRestoreInstanceState(Bundle b) 또는 
  onCreate(Bundle b) 를 사용하여 복원작업을 수행할 수 있다

- 화면의 Orientation(가로/세로) 변경시 onDestroy()를 호출 후 onCreate()가 실행된다

 

- onCreate()
액티비티가 처음 생성될때와(화면이 회전되어 orientation이 변경될때도 기본으로 실행) 실행되며
레이아웃배치와 변수선언 등 각종 초기화 작업을 구현하는 단계이다
완료 후 onStart()를 호출한다

- onStart()
onCreate() 이후 또는 onRestart()로 다시 시작된 후 처음으로 호출하는 단계이며
화면생성이 완료되고 사용자 입력을 받기 바로 직전의 단계이다
완료 후 onResume()를 호출한다

- onResume()
onResume() 이 실행되면 상호작용이 가능하며 모든준비가 완료되고 액티비티가 완벽하게 활성 상태가 된다.
onPause()에서 다시 시작되었을때 호출하는 단계이기도 하다

- onPause()
상태에서 다른 Activity에 포커스를 넘겨준 후 비활성화(background) 되며 상호작용이 중단된 단계이다
이 상태 이후부터는 언제든지 시스템의 강제종료가 가능하며 이 상태가 완료 되어야 다음 Activity가
활성화(foreground) 되고 이후 이 액티비티는 onStop() 상태가 된다
단, 다음 Activity 의 테마가 translucent 인 투명한 배경의 Activity 와 같이 풀스크린이 아닌
Activity일 경우에는 onStop() 상태가 되지 않고 onPause()를 유지한다
하나 더 덧붙이면, 투명배경의 Activity와 비슷한 Diaglog 의 경우에는 onPause()가 적용되지 않는다

- onStop()
Activity가 사용자에게 보이지 않는 비활성화 상태로 돌아서게 되면 호출하는 단계이다
=> 다른 activity 를 활성화 시켰을때는 background로 onStop() 상태가 유지되며,
=> 현 액티비티에서 finish()를 호출하거나 back key를 이용해 home activity로 돌아갈때는
onStop() 후 onDestroy()를 호출하고
=> 다음 Activity에서 back 하여 현 액티비티로 돌아올때는 onRestart(), onStart(), onResume 순으로 호출한다

- onDestroy()
finish(), finishActivity() 같은 정상적인 종료시에 호출되며, onCreate()에서 호출한 자원의 반납등을 하는 단계이다

- onRestart()
onStop()인 비활성화 상태에서 다시 활성화되는 단계에서 발생한다


Posted by 모과이IT
,

안드로이드를 개발하면서 json를 파씽해서 조작하고 지금은 직접 DB를 설계해서 php로 json를 직접 만들고 있으면서도 지금 내가 하는것이 정확히 무슨 용어가 있는지 몰랐다. 우현히 페이스북에서 RESTful 관련 강좌를 훝어 보니 이때까지 내가 했던것이 RESTFul 이었다. 이제 이런 용어로 사용해야겠다. 오랜동안 응용프로그램개발 하면서 현재 RESTful API를 php로 만들면서 웹도 건들고있는 상황이지만 웹쪽에서 이런 개념이 이미 오래전부터 사용해 왔는듯 하다.



  RESTful API란? 

기본적으로 데이터를 생성, 읽기, 수정, 삭제 하는 작업이다. 

이작업들은 간단하게 CRUD(Create, Read, Update, Delete)라고 한다. 

DB SQL 작업도 CRUD의 범주에 있다. 


RESTful API 란 HTTP를 통하여 웹서버 및 DB로 구성된 서버에게 데이타 operation 을 요청하고,

그결과를 받는 것이다. 



 RestFul 이란 무엇인가?

REST(Representational State Transfer)

요즘 서버는 1인데 클라이언트가 굉장히 다양해 졌다. 

하나의 클라이언트를 위한 서버를 구성하는 것은 비효율적인 일이 되버렸다. 

하나의 서버로 여러대의 클라이언트를 대응하도록 할때 필요한 것이 RESTFul API 이다. 


※ REST = Resource(20%) + URI(70%) + HTTP method(10%)



출처 : http://blog.naver.com/2hyoin/220401546821


Posted by 모과이IT
,


ListdataReply.java


public class ListDataReply {


    String num;

    String id;

    String text;

    String time;

    String psurl;


    public ListDataReply(String num, String id, String text, String time, String psurl) {

        this.num = num;

        this.id = id;

        this.text = text;

        this.time = time;

        this.psurl = psurl;

    }


    public String getNum() {

        return num;

    }


    public void setNum(String num) {

        this.num = num;

    }


    public String getId() {

        return id;

    }


    public void setId(String id) {

        this.id = id;

    }


    public String getText() {

        return text;

    }


    public void setText(String text) {

        this.text = text;

    }


    public String getTime() {

        return time;

    }


    public void setTime(String time) {

        this.time = time;

    }


    public String getPsurl() {

        return psurl;

    }


    public void setPsurl(String psurl) {

        this.psurl = psurl;

    }

}



ReplyListAdapter.java


package com.src.ReviewBBS;


import java.util.ArrayList;

import java.util.Collections;


import android.view.View;

import android.view.ViewGroup;


import com.widget.InfiniteScrollListAdapter;

public class ReplyListAdapter extends InfiniteScrollListAdapter {


    // A placeholder for all the data points

    private ArrayList<ListDataReply> entries = new ArrayList<ListDataReply>();

    private NewPageListener newPageListener;


    // A demo listener to pass actions from view to adapter

    public static abstract class NewPageListener {

        public abstract void onScrollNext();

        public abstract View getInfiniteScrollListView(int position, View convertView, ViewGroup parent);

    }


    public ReplyListAdapter(NewPageListener newPageListener) {

        this.newPageListener = newPageListener;

    }


    public void addEntriesToTop(ArrayList<ListDataReply> entries) {

        // Add entries in reversed order to achieve a sequence used in most of messaging/chat apps

        if (entries != null) {

            Collections.reverse(entries);

        }

        // Add entries to the top of the list

        this.entries.addAll(0, entries);

        notifyDataSetChanged();

    }


    public void addEntriesToBottom(ArrayList<ListDataReply> entries) {

        // Add entries to the bottom of the list

        this.entries.addAll(entries);

        notifyDataSetChanged();

    }


    public void clearEntries() {

        // Clear all the data points

        this.entries.clear();

        notifyDataSetChanged();

    }


    @Override

    public int getCount() {

        return entries.size();

    }


    @Override

    public Object getItem(int position) {

        return entries.get(position);

    }


    @Override

    public long getItemId(int position) {

        return position;

    }


    @Override

    public void onScrollNext() {

        if (newPageListener != null) {

            newPageListener.onScrollNext();

        }

    }


    @Override

    public View getInfiniteScrollListView(int position, View convertView, ViewGroup parent) {

        if (newPageListener != null) {

            return newPageListener.getInfiniteScrollListView(position, convertView, parent);

        }

        return convertView;

    }


}



ReviewViewReplyActivity.java


public class ReviewViewReplyActivity extends BaseActivity {


    ////////////////////////// <infinelistScrollsview> start ///////////////////////////////////////

    ArrayList<ListDataReply> moguwaiEntries;

    private ReplyListAdapter replyListAdapter;

    static DrawableManager DM = new DrawableManager();

    private static final int SEVER_SIDE_BATCH_SIZE = 10;

    private InfiniteScrollListView demoListView;

    private BogusRemoteService bogusRemoteService;

    private Handler handler;

    private AsyncTask<Void, Void, Void> fetchAsyncTask;

    ////////////////////////// <infinelistScrollsview> end ///////////////////////////////////////


.......................(중략)

}


   @Override

    protected void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);

        moguwaiListScrollsview();


................(중략)

    }


   public void moguwaiListScrollsview(){



        ////////////////////////////////////////////

        handler = new Handler();


        demoListView = (InfiniteScrollListView) this.findViewById(R.id.listview_review);


        LayoutInflater layoutInflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);

        demoListView.setLoadingView(layoutInflater.inflate(R.layout.loading_view_demo, null));

        replyListAdapter = new ReplyListAdapter(new ReplyListAdapter.NewPageListener() {

            @Override

            public void onScrollNext() {

                Util.log("moguwai","===========> onScrollNext()22 <==========");

                fetchAsyncTask = new AsyncTask<Void, Void, Void>() {

                    @Override

                    protected void onPreExecute() {

                        // Loading lock to allow only one instance of loading

                        replyListAdapter.lock();

                    }

                    @Override

                    protected Void doInBackground(Void ... params) {

                        Util.log("moguwai","===========> onScrollNext()11 <==========");

                        //String temp = params[0];

                        // Mimic loading data from a remote service


                        // 이부분 거창하게 클래스에 이런 함수 쓸필요없이

                        //

                        Util.log("moguwai","===========> onScrollNext()55 <==========");

                        //result = getDBtoArray(iPage);

                        Util.log("moguwai","===========> onScrollNext()66 <==========");

                        //iPage+=10;

                        //return result;

                        return null;

                    }

                    @Override // jimin

                    protected void onPostExecute(Void result2) { // 추가하는부분

                        Util.log("moguwai", "===========> onPostExecute 시작<==========");

                        Util.log("moguwai","===========>iState<===["+iState);

                        // ArrayList<ListDataReview> result;


                        switch(iState){

                            case 0:


                                getListview(0);



                                break;


                        }





                    };

                    @Override

                    protected void onCancelled() {

                        // Tell the adapter it is end of the list when task is cancelled

                        replyListAdapter.notifyEndOfList();

                    }

                }.execute();

            }

            @Override

            public View getInfiniteScrollListView(int position, View convertView, ViewGroup parent) {

                // Customize the row for list view

                if(convertView == null) {

                    LayoutInflater layoutInflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);

                    convertView = layoutInflater.inflate(R.layout.row_reply, null);

                }

                ListDataReply name = (ListDataReply) replyListAdapter.getItem(position);

                if (name != null) {

                    TextView rowText = (TextView) convertView.findViewById(R.id.row_text2);

                    ImageView rowPsurl = (ImageView) convertView.findViewById(R.id.row_psurl2);

                    TextView rowTime = (TextView) convertView.findViewById(R.id.row_time2);

                    TextView rowId = (TextView) convertView.findViewById(R.id.row_id2);


                    //row_id row_catalog

                    //row_time



                    // 실제 Card에 Display하는 부분

                    rowId.setText(name.getId());

                    rowText.setText(name.getText());

                    DM.fetchDrawableOnThread(name.getPsurl(), rowPsurl);  //비동기 이미지 로더

                    rowTime.setText(name.getTime());





                }

                return convertView;

            }

        });

        demoListView.setAdapter(replyListAdapter);

        // Display a toast when a list item is clicked

        demoListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {


            public void onItemClick(AdapterView<?> parent, View view, final int position, long id) {


                boolean post = handler.post(new Runnable() {

                    @Override

                    public void run() {

                        ListDataReply name = (ListDataReply) replyListAdapter.getItem(position);

                        // Toast.makeText(YouTubeActivity.this, name.getTitle() + " " + getString(R.string.ordered), Toast.LENGTH_SHORT).show();



                        String strGetID = name.getId();

                        String strGetTitle = name.getText();

                        String strUrl = name.getPsurl();

                        String strNum = name.getNum();

                        // 새로 추가

                        String strPsurl = name.getPsurl();

                        String strTime = name.getTime();


                        String strText = name.getText();

                        // hXBBH3aN0LA

                        Log.d("moguwai", "strGetID = " + strGetID);

                        Log.d("moguwai", "strGetTitle = " + strGetTitle);

                        Log.d("moguwai", "strNum= " + strNum);

                        Log.d("moguwai", "strPsurl= " + strPsurl);

                        Log.d("moguwai", "strTime= " + strTime);


                        Log.d("moguwai", "strText= " + strText);



                    }

                });

            }

        });

    }

    public void getListview(int iNum){

        String strUrl="";

        ArrayList<ListDataReply> result;

        switch(iNum)

        {

            case 0: // 전체

                strUrl="http://121.182.120.85/getreviewlist1.php?page=";

                break;


        }



        result = getDBtoArray(iPage,strUrl);

        iPage += 10;

        if (result == null || result.isEmpty()) {

            replyListAdapter.notifyEndOfList();


        } else {


            // List

            replyListAdapter.addEntriesToBottom(result);


            // Add or remove the loading view depend on if there might be more to load

            if (result.size() < SEVER_SIDE_BATCH_SIZE) {

                replyListAdapter.notifyEndOfList();

            } else {

                replyListAdapter.notifyHasMore();

            }

            // Get the focus to the specified position when loading completes

        } // else


        Util.log("moguwai", "===========> onPostExecute 끝<==========");

    }

    public ArrayList<ListDataReply> getDBtoArray(int page,String strUrl){

        ArrayList<ListDataReply> tempEntries = new ArrayList<ListDataReply>();



        String strRet="";

        strRet = requestDataForServer(1, strUrl+page, this);


        String num="";

        String psurl="";

        String id="";

        String time="";

        String text="";

        JSONArray ja = null;

        try {

            Util.log("moguwai","===========> Debug1");

            ja = new JSONArray(strRet);

            Util.log("moguwai","===========> Debug2");

        } catch (JSONException e) {

            e.printStackTrace();

            Util.log("moguwai", "===========> Debug3");

        }

        Log.d("moguwai", "ja.length()="+ja.length());

        for(int i=0;i<ja.length();i++)

        {

            JSONObject obj = null;

            try {

                obj = ja.getJSONObject(i);

                num = obj.getString("num");

                psurl = obj.getString("psurl");

                Log.d("moguwai", "psurl: "+i+"번째 :"+psurl);

                id = obj.getString("id");

                time = obj.getString("time");

                text = obj.getString("text");



            } catch (JSONException e) {

                e.printStackTrace();

            }


            ListDataReply tempList1 =  new ListDataReply(num,id,text,time,psurl);

            tempEntries.add(tempList1);

        }


        return tempEntries;

    }


 public void getListFromDB(int State){

        iState =State;

        replyListAdapter.onScrollNext();

    }

    public void resetListview(){

        ///// 전에꺼 지우는 코드

        iPage=0;

        replyListAdapter.clearEntries();

        replyListAdapter.notifyEndOfList();

        ////////////////////////////

    }


    @Override

    protected void onResume() {

        super.onResume();

        Log.d("moguwai", "onResume()");

        resetListview();

        getListFromDB(0);

        //reviewListAdapter.onScrollNext();

    }


row.reply.xml


<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    xmlns:tools="http://schemas.android.com/tools"

    android:layout_width="match_parent"

    android:paddingTop="5dp"

    android:paddingBottom="5dp"

    android:paddingLeft="10dp"

    android:paddingRight="5dp"

    android:layout_height="100dip"

    android:background="#ffffff"

    android:orientation="horizontal" >


    <LinearLayout

        android:layout_width="0dip"

        android:layout_height="100dip"

        android:layout_weight="1.5"

        android:orientation="horizontal" >


        <ImageView

            android:id="@+id/row_psurl2"

            android:layout_width="match_parent"

            android:layout_height="match_parent" >

        </ImageView>

    </LinearLayout>


    <LinearLayout

        android:layout_width="0dip"

        android:layout_height="100dip"

        android:layout_weight="4"

        android:weightSum="1"

        android:orientation="vertical" >


        <LinearLayout

            android:layout_width="match_parent"

            android:layout_weight="0.35"

            android:layout_height="0dip"

            android:weightSum="1"

            android:orientation="horizontal" >


            <LinearLayout

                android:layout_width="0dip"

                android:layout_weight="0.25"

                android:layout_height="match_parent"

                android:orientation="horizontal" >

                <TextView

                    android:id="@+id/row_id2"

                    android:textColor="#000000"

                    android:layout_width="match_parent"

                    android:layout_height="match_parent"

                    android:text="아이디"

                    android:textSize="11dp"

                    android:singleLine="true"

                    >

                </TextView>




            </LinearLayout>



            <LinearLayout

                android:layout_width="0dip"

                android:layout_weight="0.75"

                android:layout_height="match_parent"

                android:orientation="horizontal" >

                <TextView

                    android:id="@+id/row_time2"

                    android:textColor="#000000"

                    android:layout_width="match_parent"

                    android:layout_height="match_parent"

                    android:text="시간"

                    android:textSize="11dp"

                    android:singleLine="true"

                    >

                </TextView>




            </LinearLayout>


        </LinearLayout>

        <LinearLayout

            android:layout_width="match_parent"

            android:layout_weight="0.65"

            android:layout_height="0dip"

            android:orientation="horizontal" >

            <TextView

                android:id="@+id/row_text2"

                android:textColor="#000000"

                android:layout_width="match_parent"

                android:layout_height="match_parent"

                android:text="본문"

                >

            </TextView>



        </LinearLayout>



    </LinearLayout>


</LinearLayout>



[Github 필요한 파일 추출]

modifybymoguwai.zip



Posted by 모과이IT
,

한참을 삽질하고, 보내다가 어떤 블로그를 보니, 하고싶은 것이 된다. 

블로그 정리로 시작하자. 



01. DLL Project 간단한 생성 방법은 이전에 작성한 posting 을 참조.

http://vbflash.net/900


02. Posting 하고자하는 내용 설명

 2.1 C와 C#에서의 커뮤니케이션 방법(통신방법)

...검색하다보니깐 -_- PDF로 나온다..


 Calling_C_Library_DLLs_from_C_sharp.pdf


-2013.09.30-

by 퓨림노


현재 Unity3D를 이용하여 증강현실 시스템을 개발중이다. Unity3D는 C#, Javascript의 언어를 사용한다. 

현재 기존에 개발하였던 모듈의 언어가 C,C++,Java 로 구성되어 있다. 그래서 난 C++ 로 구성되어 있는 클래스를 그대로 사용하기 위하여 C의 DLL 모듈로 만들어서 땡겨서 쓰려고 한다. 


이전에 Android <-> C의 통신을 위하여 native C를 사용해보았는데, C<->C# 또한 조금 다르게 코딩이 되어진다. 더군다나 DLL 로 만들어서 사용하기 때문에 디버깅은 미궁에 빠지게 된다. 


첫 경험이란게 정말 소중할 정도로. -_- 포스팅해두자. 난중에 되면 까먹는다. 


그래서 현재 아래의 방법을 이용하면 거진 하고 싶은 것은 할 수 있다. 라고 생각함. 

다른방법도 있지만 지금은 현재 완료되어 있는 부분부터 해결 하고 넘어가도록 하자. 


- 2013.09.27 - 

  금요일 밤인데, 집중해서 주절주절. by 퓨림노

   

     

03. C와 C#에서 다양한 방법을 통해 값을 넘겨주고 받자. 

    현재 Unity3D 의 C#코드에서 C코드를 사용하여 processing하고자 한다. 상황을 잘 이해 하였는가?


 3.0 C#에서 C의 native code 삽입하자.


// Library
	[DllImport("UnityPlugin")]
    public static extern void Scalars(double invar, ref double outvar);

    [DllImport("UnityPlugin")]
    public static extern void OneDArray(double [] anArray, int n);

    [DllImport("UnityPlugin")]
    public static extern void TwoDArray(double [,] a2dArray, int m, int n);

    [DllImport("UnityPlugin")]
    public static extern void TryComplex(Complex inputVar, ref Complex outputVar,
               [In, Out] Complex [] array, int n);

    [DllImport("UnityPlugin")]
    public static extern void MarshalStructC(ref marshalStruct pointerinStruct);
    
	[DllImport("UnityPlugin")]
    public static extern void FreeMarshalStructptr(ref marshalStruct pointerinStruct);

 - C#의 젤 상위에 DLL 을 import 한다고 작성하자.


 3.1 변수의 포인터를 이용하여 값을 넘겨주고 받자.

// 아래의 코드는 항상 들어가기 때문에 필수라 생각하자. 
#define EXPORT_API __declspec(dllexport)

//01. C side code
EXPORT_API void Scalars( double in, double *out ) {
		*out = in;
	}

// 02. C# side code
double invar = 5.0;
double outvar = 0.0;
Scalars(invar, ref outvar);		
Debug.Log( "invar = " + invar +  "outvar = " + outvar );		

포인터를 이용하여 invar의 값을 outvar 로 삽입이 되었다. 

C side에서 다르게 응용이 되어진다. 


 3.2 1차원 배열의 값을 주고 받자.

아래쪽 코드는 글자가 깨지는 관계로.....그냥 박스 상자에서 보시길. 

//01. C side code

EXPORT_API void OneDArray( double anArray[], int n ) {
for( int i=0; i<n; i++ ) {
anArray[i] = i;
}
};


// 02. C# side code

double [,] a2dArray = new double [3,2];
TwoDArray(a2dArray, a2dArray.GetLength(0), a2dArray.GetLength(1));
for( int i=0; i<3; i++) {
for( int j=0; j<2; j++ ){
Debug.Log( "array[" + i + "][" + j + "] = " + a2dArray[i,j] );
}

}


 3.3 2차원 배열을 주고 받자.
// 01. C side code
EXPORT_API void TwoDArray( double a2Array[], int m, int n ) {

		a2Array[0*n+0] = 1; // 0,0
		a2Array[1*n+0] = 2; // 1,0
		a2Array[2*n+1] = 3;	// 2,1
	}

// 02. C# side code
double [,] a2dArray = new double [3,2];				
        TwoDArray(a2dArray, a2dArray.GetLength(0), a2dArray.GetLength(1));
		for( int i=0; i<3; i++) {
			for( int j=0; j<2; j++ ){
				Debug.Log( "array[" + i + "][" + j + "] = " + a2dArray[i,j] );
			}
		}

 3.4 C,C#의 구조체를 주고 받자. 

// 01. C side code
struct myComplex {
	double re;
	double im;
};

EXPORT_API void TryComplex( myComplex inputVar, myComplex *outputVar, myComplex array[], int n ) {
		outputVar->re = ++inputVar.re;
		outputVar->im = ++inputVar.im;

		array[0].re = 99.0;
		array[0].im = 98.0;
		array[1].re = 97.0;
		array[1].im = 96.0;
	}

// 02. C# side code
Complex inputVar = new Complex();
		Complex outputVar = new Complex();
		Complex [] array = new Complex[3];
		inputVar.re = 1.0;
		inputVar.im = 2.0;
		TryComplex(inputVar, ref outputVar, array, array.GetLength(0));



 3.5 구조체의 동적할당을 이용하자. (미해결)

 - 다른 블로그 참조해서 해결하시기 바랍니다. 


<추가> Char/String 을 전달하는 방법

- 첨부파일을 참조하기 바람. 그리고 아래의 방법은 char을 사용하는 것.

// 01. C side code
void EXPORT_API ReadPort( char *filename ) {
		// for Read Port Number & Connect
	}

void EXPORT_API SendDataToSerialPort( char *ch ){
     if('q' == *ch) {
      }
	}

// 02. C# side code
	[DllImport("LibModule.dll")]	
	private static extern void ReadPort( char[] filename );

	[DllImport("LibModule.dll")]	
	private static extern void SendDataToSerialPort( char[] dir );


	void Start () {
		string str = "Port.txt";
		char[] szArr = str.ToCharArray();
		
		ReadPort (szArr);
		Debug.Log ("ReadPort");
	}

	void Update () {
		
		char[] ch = new char[1];
		
		if(Input.GetKeyUp(KeyCode.E)) { 				// Up
			Debug.Log ( Count++ + "KeyDown : E ");			
			ch[0] = 'e';
			SendDataToSerialPort(ch);
		}
	}



Reference


[1] htna 님의 블로그 ( http://htna.tistory.com/342 )
     - C와 C#통신하는 부분을 링크 해두었습니다. 위의 코드와 동일하나. 음....그대로 가따붙이니 안되네요.  조금 수정하였습니다. 


Posted by 모과이IT
,

Cannot start session without errors, please check errors given in your PHP and/or webserver log file and configure your PHP installation properly.

오류 해결방법

 

로그인해서 빨간색 으로 색이 칠해지고

Cannot start session without errors, please check errors given in your PHP and/or webserver log file and configure your PHP installation properly.

문구가 나오면

 

뒤로 한번 누르고

 

새로고침 한번 누르고

 

다시 로그인하면 됩니다.

Posted by 모과이IT
,