>웹 프론트엔드 >JS 튜토리얼 >C/C를 사용하여 Node.js 모듈 구현 (2)_node.js

C/C를 사용하여 Node.js 모듈 구현 (2)_node.js

WBOY
WBOY원래의
2016-05-16 16:35:311147검색

과거를 되돌아보고 새로운 것을 배우면 행복해질 수 있습니다

우선 V8 온라인 매뉴얼(http://izs.me/v8-docs/main.html)을 기억해 주세요.

지난번에 본 Building.gyp 파일을 아직도 기억하시나요?

코드 복사 코드는 다음과 같습니다.

{
"대상": [
{
"target_name": "애드온",
"소스": [ "addon.cc" ]
}
]
}

이렇듯 *.cc 파일이 몇 개 더 있으면 다음과 같습니다.
"소스": [ "addon.cc", "myexample.cc" ]

지난 번에는 두 단계를 분리했습니다. 실제로 구성과 컴파일을 함께 사용할 수 있습니다.
$ node-gyp 구성 빌드

리뷰는 다하셨나요? 없이? !

자, 계속하겠습니다.

목차

함수 매개변수

이제 드디어 매개변수에 대해 이야기해야 합니다.

a와 b를 더하고 결과를 반환하는 add(a, b) 함수가 있다고 가정해 보겠습니다. 먼저 함수 개요를 작성합니다.

코드 복사 코드는 다음과 같습니다.

#include
네임스페이스 v8 사용;

<값> 추가(const Arguments& args) 처리
{
HandleScope 범위;

//... 다시 시작합니다!
}

인수

함수의 매개변수입니다. 먼저 v8의 공식 매뉴얼 참고자료를 살펴보겠습니다.
•int 길이() const
•로컬 연산자[](int i) const

나머지는 신경 쓰지 않습니다. 이 두 가지가 중요합니다! 하나는 함수에 전달되는 매개변수의 수를 나타내고, 다른 괄호는 아래 첨자 색인을 통해 n번째 매개변수에 액세스하는 데 사용됩니다.

따라서 args.Length()는 2이고, args[0]은 a를 나타내고 args[1]은 b를 나타내므로 위의 요구 사항을 대략적으로 이해할 수 있습니다. 그리고 우리는 이 두 숫자의 유형이 숫자여야 함을 결정해야 합니다.

대괄호 안의 인덱스 연산자는 Node.js의 모든 유형의 기본 클래스인 Local를 반환합니다. 따라서 전달된 매개변수는 불확실한 유형이므로 어떤 매개변수인지 스스로 결정해야 합니다. 이는 이 값 유형의 일부 기능과 관련이 있습니다.

•IsArray()
•IsBoolean()
•IsDate()
•IsFunction()
•IsInt32()
•IsNativeError()
•IsNull()
•IsNumber()
•IsRegExp()
•IsString()
•...

하나씩 나열하지는 않겠습니다. 나머지는 문서를 읽어보세요. 。:.゚ヽ(*´∀`)ノ゚.:。

ThrowException

나중에 사용할 기능입니다. 자세한 내용은 v8 설명서에서 확인할 수 있습니다.

이름에서 알 수 있듯이 오류가 발생합니다. 이 문을 실행한 후에는 Node.js 로컬 파일에서 throw() 문을 실행하는 것과 같습니다. 예:
ThrowException(Exception::TypeError(String::New("인수 개수가 잘못됨")));

Node.js를 실행하는 것과 동일합니다.
throw new TypeError("인수 개수가 잘못되었습니다.");

정의되지 않음()

이 기능은 문서에도 있습니다.

구체적으로 말하면 일부 함수는 특정 값을 반환할 필요가 없거나 반환 값이 없기 때문에 null 값입니다. 이때 대신 Undefine()을 사용해야 합니다.

해보자, 사오니안!

위 사항을 이해하고 나면 곧 a b의 논리를 작성할 수 있을 것이라고 믿습니다. Node.js 공식 매뉴얼의 코드를 복사하여 여러분에게 제공하겠습니다.

코드 복사 코드는 다음과 같습니다.

#include
네임스페이스 v8 사용;

<값> 추가(const Arguments& args) 처리
{
HandleScope 범위;

// 2개 이상의 매개변수를 전달할 수 있다는 뜻이지만 실제로는 처음 두 개만 사용합니다.
If(args.Length() < 2)
{
// 오류 발생          ThrowException(Exception::TypeError(String::New("인수 개수가 잘못됨")));

                                                                // null 값을 반환합니다.           반환 범위.닫기(정의되지 않음());

}

// 처음 두 매개변수 중 하나가 숫자가 아닌 경우

If(!args[0]->IsNumber() || !args[1]->IsNumber())

{
                 // 오류를 발생시키고 null 값을 반환합니다
         ThrowException(Exception::TypeError(String::New("잘못된 인수")));
          반환 범위.닫기(정의되지 않음());
}

// 자세한 내용은 v8 문서를 참고하세요

//

http://izs.me/v8-docs/classv8_1_1Value.html#a6eac2b07dced58f1761bbfd53bf0e366
) // `NumberValue` 함수 로컬 num = Number::New(args[0]->NumberValue() args[1]->NumberValue());

반환 범위.Close(num);

}



기능이 완료되었습니다!

마지막으로 내보내기 기능을 마지막에 적어주시면 괜찮습니다.


void Init(Handle 내보내기)
{
내보내기->Set(String::NewSymbol("add"),
FunctionTemplate::New(Add)->GetFunction());
}

NODE_MODULE(애드온, 초기화)


컴파일한 후에는 다음과 같이 사용할 수 있습니다.


코드 복사 코드는 다음과 같습니다.var addon = require('./build/Release/ 애드온') ;
console.log(addon.add(1, 1) "b");


2b가 보일 거예요! ogue。٩(ˊᗜˋ)وcade*。

콜백 기능

지난 장에서는 Hello world에 대해서만 이야기했습니다. 이 장에서 할머니는 의식적인 발견을 하고 또 다른 콜백 함수를 작성했습니다.

평소와 같이 프레임워크를 먼저 작성합니다.


#include
네임스페이스 v8 사용;

<값> RunCallback 처리(const Arguments& args) {

HandleScope 범위;

// ... 딱딱딱딱

return range.Close(Undefine());

}


그런 다음 사용법은 다음과 같다고 결정했습니다.

func(함수(msg) {

console.log(msg);
});

즉, 콜백 함수에 매개변수를 전달하고 이를 문자열이라고 가정한 다음 console.log()로 출력할 수 있습니다.

먼저 스트링 시리즈가 있어야 합니다

더 이상 고민하지 말고 먼저 문자열을 입력한 다음 이야기해 보겠습니다. (√ζ ε:)

하지만 Node.js 코드는 약한 형식이기 때문에 이 문자열을 범용 형식으로 만들어야 합니다.
Local::New(String::New("hello world"));

뭐? Local가 무엇인지 물어보셨나요?

그럼 조금 이야기를 해볼께요. 여기와 V8 참고문서를 참고하세요.

문서에 표시된 것처럼 Local는 실제로 Handle에서 상속됩니다. 이전 장에서 이미 언급한 것을 기억합니다.

그럼 로컬(Local)에 대해 이야기해보겠습니다.


Handle에는 Local와 Pertant>의 두 가지 유형이 있습니다. 전자와 Handle 수명주기가 범위 내에 있습니다. 후자의 수명 주기는 범위를 벗어나므로 수명 주기를 종료하려면 Pertant::Dispose를 수동으로 호출해야 합니다. 즉, Local Handle은 C`스택에 객체를 할당하는 것과 동일하고 Persistant Handle은 C가 힙에 객체를 할당하는 것과 같습니다.

그런 다음 매개변수 테이블 시리즈가 필요합니다

터미널 명령줄에서 C/C를 호출한 후 명령줄 매개변수를 가져오는 방법은 무엇입니까?

코드 복사 코드는 다음과 같습니다.

#include

void main(int argc, char* argv[])
{
// ...
}

그런데 여기서 argc는 명령줄 매개변수의 개수이고, argv[]는 각 매개변수입니다. 그런 다음 Node.js의 콜백 함수를 호출하면 v8도 비슷한 방법을 채택합니다.

코드 복사 코드는 다음과 같습니다.
V8EXPORT Local v8::Function::Call(Handle< 개체>recv ,
정수 인수
argv[]
처리 );

~~QAQ가 Handle에 갇혔습니다! ! ! 내일 계속 쓸 예정입니다. ~~

자, 새로운 하루가 시작되었고 힘이 넘칩니다. (∩^o^)⊃━☆゚.*・。

여러 측면(SegmentFault, StackOverflow 및 KouKou 그룹)에서 확인한 후 마침내 위 함수의 세 가지 매개 변수의 의미를 해결했습니다.

다음 두 매개변수에 대해서는 많이 언급하지 않겠습니다. 하나는 매개변수의 개수이고 다른 하나는 매개변수의 배열입니다. 첫 번째 매개변수인 Handle에 대한 StackOverflow의 설명은 다음과 같습니다.


JS에서 적용하는 것과 동일합니다

코드 복사 코드는 다음과 같습니다.

var 컨텍스트 = ...;
cb.apply(context, [ ...args...]);

첫 번째 인수로 전달된 개체는 함수 범위 내에서 this가 됩니다. MDN에 대한 추가 문서는 여기에서 JS에 대한 자세한 내용을 읽을 수 있습니다: http://unschooled.org /2012/03/understanding-javascript-this/

——StackOverflow에서 발췌

간단히 말해서, 그 기능은 호출된 함수의 this 포인터를 지정하는 것입니다. 이 호출의 사용법은 JavaScript의 바인딩(), call() 및 Apply()와 유사합니다.

그래서 우리가 해야 할 일은 먼저 매개변수 테이블을 구축한 다음 호출 함수를 전달하여 실행하는 것입니다.

원래 개체 유형이므로 첫 번째 단계는 변환 기능을 표시하는 것입니다.
로컬 cb = 로컬::Cast(args[0]);

두 번째 단계는 매개변수 테이블(배열)을 만드는 것입니다.
Local argv[argc] = { Local::New(String::New("hello world")) };

라스트콜 기능 시리즈

cb를 호출하고 다음 매개변수를 전달합니다.
cb->Call(Context::GetCurrent()->Global(), 1, argv);

여기서 첫 번째 매개변수인 Context::GetCurrent()->Global()은 함수의 전역 컨텍스트를 가져오는 것을 의미합니다. 두 번째 매개변수는 매개변수 테이블의 숫자입니다(결국 Node.js에서는 배열에는 길이 속성이 있지만 시스템은 실제로 C 배열의 길이를 알지 못하므로 배열의 길이를 나타내기 위해 숫자를 직접 전달해야 합니다. 마지막 매개변수는 방금 만든 매개변수 테이블입니다. .

마지막 장 최종 문서 시리즈

함수를 작성한 다음 내보낸 함수에 넣고 마지막으로 선언하는 이 단계는 이미 다들 잘 알고 계실 거라 믿습니다.

코드를 직접 공개하거나 Node.js 문서로 직접 이동할 수 있습니다.

코드 복사 코드는 다음과 같습니다.

#include
네임스페이스 v8 사용;

<값> RunCallback 처리(const Arguments& args)
{
HandleScope 범위;
로컬 cb = 로컬::Cast(args[0]);
const unsigned argc = 1;
Local argv[argc] = { Local::New(String::New("hello world")) };
cb->Call(Context::GetCurrent()->Global(), argc, argv);

return range.Close(Undefine());
}

void Init(Handle 내보내기, Handle 모듈)
{
모듈->Set(String::NewSymbol("exports"),
FunctionTemplate::New(RunCallback)->GetFunction());
}

NODE_MODULE(애드온, 초기화)

수고하셨습니다. 마지막 남은 단계를 직접 수행해 보세요. JS에서 이 함수를 호출하는 것에 대해서는 이전에 언급한 적이 있습니다.

추가

음, 공부노트가 점점 무뎌지는 것 같아요. 풀어주세요~

오늘은 여기서 마치겠습니다. 학습 노트를 작성하는 과정에서 Call 함수의 매개변수에 대한 의미 등으로 또 고민에 빠졌습니다.

이 학습 노트 시리즈가 여전히 도움이 된다고 생각하신다면 저와 함께 즐겨보세요~Σ>―(〃°Ω°〃)♡→

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.