찾다
웹 프론트엔드JS 튜토리얼Angularjs1 구성 요소 프로그래밍에 대한 심층 분석(예제 포함)

이 글은 주로 angularjs에 대한 심층적인 분석을 소개합니다. 어렵다고 생각하지 않는 한 여기에서 Anglejs를 배울 수 있습니다. 이제 이 글을 함께 읽어봅시다

angular 1에도 컴포넌트 지향 프로그래밍이 필요합니다

프런트엔드 컴포넌트화는 프런트엔드 개발 모델에서 돌이킬 수 없는 추세입니다. 세 가지 주요 프런트엔드 프레임워크인 Angular 2 code> <code>react vue 모두 구성 요소 프로그래밍을 주요 판매 포인트 중 하나로 간주합니다. angular 1 상대적으로 오랜 역사를 가진 프레임워크입니다. 사생아 angular 2는 마침내 컴포넌트 기반 프로그래밍의 마지막 열차에 올랐고, 회사의 오래된 프로젝트들은 마침내 컴포넌트 기반 프로그래밍의 맛을 경험할 기회를 얻었습니다. angular 2 react vue 都不约而同的把组件化编程作为自己的一大卖点,angular 1 作为一个历史相对悠久的框架,在私生子 angular 2 的推动下,终于也搭上了组件化编程的末班车,公司里那些老项目终于也有机会体验组件化编程的滋味。

angular 1 的组件化之路

angular 1 中类似组件化的编程思想其实很早就有,只不过那时候不叫组件,而叫指令(Directive),指定 restrict: 'E' 后这个指令就与如今组件的用法很相似了。angular 1.5 中,又将指令根据 angular 2 的类似概念加以限制,脱胎为如今的组件(Components)。

组件的特点

官方文档列举了组件和指令的不同点。除此之外,一个规范的组件还应符合以下几个特点。

  1. 组件的标签名称必须包含中划线

  2. 组件拥有良好的生命周期

  3. 组件有自包含性

  4. 组件有自封闭性

  5. 组件有可复用性

  6. 组件可以被定制化

下面依次说明。

组件的名称规范

与指令不同,组件必须是一个元素,HTML 对于这一点有特殊的规范。

HTML 规范把带有中划线的标签留给开发者使用,这样形成的元素又称作自定义元素(Custom Element)。我们虽然没有用到自定义元素的概念,但两者的行为是相似的。我们应该符合这一标准。

这一点规范对应到 angular 1 中即为:组件名称必须带有驼峰形式。

例如:

module.component('dialog', {
    // ...
});

这是不对的。HTML 规范已经定义了 dialog 这个标准元素,重复使用标签名可能导致我们自定义的组件行为和标准元素的行为混杂到一起,导致奇葩 bug;而且如果这样做也间接导致开发者不能使用原生的 dialog 标签。

另外,就算现在标准没有定义某个元素,不代表将来不会定义。我们的程序既然跑在浏览器里,就要按规矩办事。这是一种合法的写法:

module.component('customDialog', {
    // ...
});

组件的自包含性

一个设计良好的组件一定有它自己的行为和默认样式。

默认行为

默认行为在 angular 1 中用控制器(Controller)定义。

function CustomDialogController($service) {
    this.someField = 123;
    this.someMethod = function someMethod() {

    }
}
CustomDialogController.$inject = ['$service'];

module.component('customDialog', {
    controller: CustomDialogController,
    template: require('./customDialogTemplate.html'),
});

因为组件默认启用 controllerAs,所有变量和函数都是绑定到 this 上的,所以你也可以使用 ES2015class 语法来组织代码:

class CustomDialogController {
    constructor($service) {

    }

    someMethod() {

    }
}
CustomDialogController.$inject = ['$service'];

module.component('customDialog', {
    controller: CustomDialogController,
    template: require('./customDialogTemplate.html'),
});

这样做有一个问题就是其他函数不能使用 constructor 里注入的服务(Service),只能通过 this 中转一次。我个人的做法是这样:

class CustomDialogController {
    constructor($service) {
        this.services = { $service };
    }

    someMethod() {
        const { $service } = this.services;
    }
}
// 下略

建议对于逻辑相对简单的组件的控制器使用 function 定义,复杂的组件使用 class 定义,后者代码的层次要更为清晰易读。

默认样式

组件的默认样式直接使用样式表指定。

custom-dialog {
    display: block;
    // ...
}

对于所有浏览器不认识的标签,默认都是内联元素(display: inline),对于组件来说通常不是想要的。所以自定义的组件通常至少要有 display: (inline-)block 来改变元素的默认显示方式。

组件的自封闭性

自封闭性包含两个方面:数据的自封闭性和样式的自封闭性。

数据的自封闭性

angular 1 中,组件自身的 scope 已经是隔离的(isolate),即组件的 scope 不继承自父级 scope(__proto__null)。除此之外,一个规范的组件不应该直接使用外部的数据,因为这样会破坏组件的可复用性。举几个例子:

  1. $rootScope

  2. $root、$parent(模板中)

  3. 路由参数

  4. localStorage、sessionStorage

这些数据都应该通过参数绑定 binding 传入。如果组件是路由插件生成,那么可以用 resolve。

其次,参数绑定不应使用双向绑定 =

angular 1의 컴포넌트화로 가는 길🎜🎜angular 1 컴포넌트화와 유사한 프로그래밍 아이디어는 실제로 오랫동안 존재해 왔지만 당시에는 컴포넌트라고 부르지 않고 를 지정하는 지시문이라고 불렀습니다. 제한: 'E' 이 지시문은 오늘날의 구성 요소 사용법과 매우 유사합니다. angular 1.5에서는 angular 2와 유사한 개념을 바탕으로 명령어를 제한하여 오늘날의 구성요소로 재탄생시켰습니다. 🎜🎜구성 요소의 특징🎜🎜공식 문서에는 구성 요소와 지침의 차이점이 나열되어 있습니다. 또한, 표준 부품은 다음과 같은 특성도 충족해야 합니다. 🎜
  1. 🎜구성 요소의 라벨 이름에는 밑줄이 포함되어야 합니다🎜
  2. 🎜구성 요소의 수명 주기가 양호합니다🎜
  3. 🎜 구성 요소는 자체 포함되어 있습니다🎜
  4. 🎜구성 요소는 자체 포함되어 있습니다🎜
  5. 🎜구성 요소는 재사용이 가능합니다🎜
  6. 🎜구성 요소는 사용자 정의 가능🎜
🎜다음을 순서대로 설명합니다. 🎜

구성 요소 이름 지정

🎜지시문과 달리 구성 요소는 요소여야 하며 HTML에는 이에 대한 특별한 사양이 있습니다. 🎜🎜HTML 사양에서는 개발자가 사용할 수 있도록 밑줄이 포함된 태그를 남겨 둡니다. 이러한 방식으로 형성된 요소를 맞춤 요소라고도 합니다. 커스텀 요소의 개념을 사용하지는 않았지만 둘의 동작은 유사합니다. 우리는 이 기준을 충족해야 합니다. 🎜🎜이 사양은 angular 1에 해당합니다. 구성 요소 이름은 카멜 표기법이어야 합니다. 🎜🎜예: 🎜
custom-dialog {
    display: block;
    // ...

    .title {
        // ...
    }

    .body {
        // ...
    }
}
🎜 이것은 옳지 않습니다. HTML 사양에서는 표준 요소 대화 상자를 정의했습니다. 태그 이름을 재사용하면 사용자 정의 구성 요소 동작이 표준 요소의 동작과 혼합되어 이상한 버그가 발생할 수 있으며, 그렇게 하면 간접적으로 개발자가 기본 를 사용하지 못하게 됩니다. 대화상자 태그. 🎜🎜또한, 현재 표준에 특정 요소가 정의되어 있지 않다고 해서 앞으로도 정의되지 않는다는 의미는 아닙니다. 우리 프로그램은 브라우저에서 실행되므로 규칙을 따라야 합니다. 이것은 합법적인 작성 방법입니다: 🎜
custom-dialog {
    display: block;

    .custom-dialog {
        &-title {
            // ..
        }

        &-body {

        }
    }
}

구성 요소의 자체 포함성

🎜잘 디자인된 구성 요소는 고유한 동작과 기본 스타일을 가져야 합니다. 🎜

기본 동작

🎜기본 동작은 컨트롤러를 사용하여 angular 1에 정의됩니다. 🎜
<custom-dialog><!-- 与标签名一样,自定义属性名也应该使用中划线 -->
    <!--content -->
</custom-dialog>
🎜구성 요소는 기본적으로 controllerAs를 활성화하기 때문에 모든 변수와 함수는 this에 바인딩되므로 ES2015 를 사용할 수도 있습니다. class 구문을 사용하여 코드 구성: 🎜
module.component('customDialog', {
    template: require('./customDialogTemplate.html'),
    transclude: true,
    bindings: {
        title: "@",
        modal: '🎜이 문제는 다른 함수가 <code>constructor</code>에 삽입된 서비스를 사용할 수 없으며 <code>this</code> 전송을 한 번만 사용할 수 있다는 것입니다. . 내 개인적인 접근 방식은 다음과 같습니다. 🎜<pre class="brush:php;toolbar:false">custom-dialog {
    display: block;
    // ...

    .title {
        font-size: 16px;
        // ...
    }

    &.big {
        .title {
            font-size: 24px;
        }
    }
}
🎜 상대적으로 간단한 로직을 사용하는 구성 요소의 컨트롤러는 함수를 사용하여 정의하고, 복잡한 구성 요소는 클래스를 사용하여 정의하는 것이 좋습니다. 코드 수준 더 명확하고 읽기 쉬워집니다. 🎜

기본 스타일

🎜구성 요소의 기본 스타일은 스타일 시트를 사용하여 직접 지정됩니다. 🎜
<custom-dialog></custom-dialog>
🎜브라우저가 인식하지 못하는 모든 태그의 경우 기본값은 인라인 요소(display: inline)이며 일반적으로 구성 요소에는 바람직하지 않습니다. 따라서 사용자 정의 구성 요소에는 일반적으로 요소의 기본 표시 모드를 변경하기 위한 최소한 display: (inline-)block이 있습니다. 🎜

구성요소의 자체 폐쇄성

🎜자체 폐쇄에는 데이터의 자체 폐쇄성과 스타일의 자체 폐쇄성이라는 두 가지 측면이 포함됩니다. 🎜

데이터 자체 폐쇄성

🎜angular 1, 구성 요소 자체 범위는 이미 격리되어 있습니다. 즉, 구성 요소의 범위는 상위 범위(__proto__)에서 상속되지 않습니다. null입니다). 또한 정식 구성 요소는 외부 데이터를 직접 사용해서는 안 됩니다. 이렇게 하면 구성 요소의 재사용성이 손상되기 때문입니다. 몇 가지 예: 🎜
  1. 🎜$rootScope🎜
  2. 🎜$root, $parent(템플릿 내)🎜
  3. 🎜 라우팅 매개변수🎜
  4. 🎜localStorage, sessionStorage🎜
🎜이 데이터는 매개변수 바인딩 바인딩을 통해 전달되어야 합니다. 구성 요소가 라우팅 플러그인에 의해 생성된 경우 확인을 사용할 수 있습니다. 🎜🎜두 번째로, 매개변수 바인딩은 양방향 바인딩 =을 사용해서는 안 되며, 표준 구성 요소는 구성 요소 외부에 전달된 데이터를 (직접) 수정해서는 안 됩니다. 공식적으로 권장되는 두 가지 매개변수 바인딩 방법이 있습니다🎜
  1. 单向绑定,绑定可变数据。通常用于给组件传递数据

  2. @ 字符串绑定,绑定字符串。通常用于指定组件行为

对于单向绑定对象的情况,由于是引用传递,也不应该修改对象内部的属性。

遇到要向外部传值的情况,推荐使用 ngModel 或 事件绑定(下面会提到)

样式的自封闭性

组件间的样式不应该互相干扰,这一点可以简单的通过 scss 的样式嵌套(Nesting)实现:

custom-dialog {
    display: block;
    // ...

    .title {
        // ...
    }

    .body {
        // ...
    }
}

这样可以简单的把组件的内置样式表限制在组件内部,从而避免样式外溢。但是这种方法对在组件内部的其他组件不起效果。如果这个组件的模板中还引用了别的组件,或者这个组件被定义为可嵌入的(transclude),那么可以考虑加类名前缀:

custom-dialog {
    display: block;

    .custom-dialog {
        &-title {
            // ..
        }

        &-body {

        }
    }
}

组件的可复用性

组件为复用而生,拥有良好自封闭性的组件必然是可复用的,因为这个组件不受任何外部因素干扰。组件的复用形式包括

  1. 一个页面中使用多次

  2. 在多个页面中使用

  3. ng-repeat

  4. 自己套自己(递归树结构)

  5. 整个源代码拷贝到其他项目中

等等。一个高度可复用的组件则可以被称为控件,是可以单独投稿 npm 项目库的。

当然,有些组件(比如单独的页面)可能复用需求没那么高,可以视组件的复用程度不同,从组件的自封闭性和整体代码量做一些取舍。

组件的定制化

一个高度可复用的组件一定可以被定制。

行为的定制化

通过参数绑定实现组件行为的定制化。例如:

<custom-dialog><!-- 与标签名一样,自定义属性名也应该使用中划线 -->
    <!--content -->
</custom-dialog>
module.component('customDialog', {
    template: require('./customDialogTemplate.html'),
    transclude: true,
    bindings: {
        title: "@",
        modal: '<p>出于使用方便的考虑,定制用的参数都是可选的,组件内部实现应该给每个定制参数设定默认值。(想看更多就到PHP中文网<a href="http://www.php.cn/course/47.html" target="_blank">angularjs学习手册</a>中学习)</p><h4 id="样式的定制化">样式的定制化</h4><p>组件风格定制可以使用 class 判断。</p><pre class="brush:php;toolbar:false">custom-dialog {
    display: block;
    // ...

    .title {
        font-size: 16px;
        // ...
    }

    &.big {
        .title {
            font-size: 24px;
        }
    }
}

使用时

<custom-dialog></custom-dialog>

深度定制样式比较好的方式是 CSS 属性(CSS Variable,注意不是 SCSS 属性)。

custom-dialog {
    display: block;
    // ...
    .title {
        font-size: 16px;
        color: var(--dialog-title-color, #333);
        // ...
    }
    &.big {
        .title {
            font-size: 24px;
        }
    }
}

这时只需要文档中说明标题颜色使用 --dialog-title-color 这个 CSS 变量就好,外部使用不依赖于组件内部 DOM 实现。使用时

.mydialog {
    --dialog-title-color: red;
}

组件的生命周期

从创建至销毁,组件有自己的生命周期(lifecycle),而不像指令那样把 scope 作为生命周期。常用的回调函数如下:

  • $onInit():组件被初始化时调用。与 constructor 不同,angular 1 确保 $onInit 被调用时组件的所有参数绑定都被正确赋值。

  • $onChanges(changeObj):组件参数绑定值被改变时调用。用于监听绑定值的变化,初次绑定时也会调用这个函数。

  • $onDestroy():组件被销毁时调用。用于清理内部资源如 $interval 等。

这些函数也是绑定在 this 上的。如果 controller 使用 ES2015class 定义方式,可以这么写:

class CustomDialogController {
    constructor() {}

    onInit() {}

    onChanges({ prop1, prop2 }) {}

    onDestroy() {}
}

组件间的通信

组件间通信是一个让很多人头疼的问题,通常有这样 3 种情况

子 -> 父

这种情况有标准的实现方式:事件绑定。例如

class CustomDialogController {
    close($value) {
        this.hide = true;
        this.onClose({ $value });
    }
}

module.component('customDialog', {
    controller: CustomDialogController,
    template: require('./customDialogTemplate.html'),
    bindings: {
        onClose: '&',
    },
});

使用时:

<custom-dialog></custom-dialog>

这种方式也可以用于子组件向父组件传值。

父 -> 子

用于触发子组件的某个动作。除了改变某个在子组件内部监听变化的绑定参数值外,行之有效的方式就只有事件广播。

子组件先监听某个事件

$scope.$on('custom-dialog--close', () => this.close());

父组件发送广播

$scope.$broadcast('custom-dialog--close');

切记:事件是全局性的。当有组件复用的情况时请使用标识指定接收对象(BUS 模型);另外最好给事件名添加组件前缀。

同级组件

请通过父级组件中转

子 -> 某全局性组件

这个显示 Notification 时最常用。遇到这种情况时,可以封装服务(Service)。例如:

module.component('globalNotification', {
    controller: class GlobalNotificationController {
        constructor(notificationService) {
            notificationService.component = this;
        }

        show(props) {
            // ...
        }
    }
});

module.factory('notify', function NotifyService() {
    return {
        warn(msg) {
            this.show({ type: 'warn', text: msg });
        }
        error(msg) {
            this.show({ type: 'error', text: msg });
        }
    }
});

方案并不完美。如果有更好的建议欢迎提出。

结语

有人可能问既然三大前端框架都是组件化的,何必还要在 angular 1 上实现。殊不知 angular 1 的组件诞生的初衷就是为了减少向 angular 2 迁移的难度。机会总是留给有准备的人,哪天老板大发慈悲表示给你把代码重写的时间,你却看着项目里满屏的 $scope.abc = xxx 不知所措,这岂不是悲剧。。。

이 글은 여기에서 끝납니다. (자세한 내용을 보려면 PHP 중국어 웹사이트 angularjs 학습 매뉴얼 을 방문하세요. 궁금한 점이 있으면 아래에 메시지를 남겨주세요. # 🎜🎜## 🎜🎜#

위 내용은 Angularjs1 구성 요소 프로그래밍에 대한 심층 분석(예제 포함)의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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

javaandjavaScriptAredistIntLanguages ​​: javaisusedforenterpriseandmobileApps, whilejavaScriptisforInciveWebPages.1) javaiscompiled, 정적으로 정적, Andrunsonjvm.2) javaScriptISNaterPreted, doineslicallytyted, andrunsinbrowsorsornode.js.3) javausepith

JavaScript 데이터 유형 : 브라우저와 Nodejs 사이에 차이가 있습니까?JavaScript 데이터 유형 : 브라우저와 Nodejs 사이에 차이가 있습니까?May 14, 2025 am 12:15 AM

JavaScript 코어 데이터 유형은 브라우저 및 Node.js에서 일관되지만 추가 유형과 다르게 처리됩니다. 1) 글로벌 객체는 브라우저의 창이고 node.js의 글로벌입니다. 2) 이진 데이터를 처리하는 데 사용되는 Node.js의 고유 버퍼 객체. 3) 성능 및 시간 처리에는 차이가 있으며 환경에 따라 코드를 조정해야합니다.

JavaScript 댓글 : / / * * /사용 안내서JavaScript 댓글 : / / * * /사용 안내서May 13, 2025 pm 03:49 PM

javaScriptUSTWOTYPESOFSOFCOMMENTS : 단일 라인 (//) 및 multi-line (//)

Python vs. JavaScript : 개발자를위한 비교 분석Python vs. JavaScript : 개발자를위한 비교 분석May 09, 2025 am 12:22 AM

Python과 JavaScript의 주요 차이점은 유형 시스템 및 응용 프로그램 시나리오입니다. 1. Python은 과학 컴퓨팅 및 데이터 분석에 적합한 동적 유형을 사용합니다. 2. JavaScript는 약한 유형을 채택하며 프론트 엔드 및 풀 스택 개발에 널리 사용됩니다. 두 사람은 비동기 프로그래밍 및 성능 최적화에서 고유 한 장점을 가지고 있으며 선택할 때 프로젝트 요구 사항에 따라 결정해야합니다.

Python vs. JavaScript : 작업에 적합한 도구 선택Python vs. JavaScript : 작업에 적합한 도구 선택May 08, 2025 am 12:10 AM

Python 또는 JavaScript를 선택할지 여부는 프로젝트 유형에 따라 다릅니다. 1) 데이터 과학 및 자동화 작업을 위해 Python을 선택하십시오. 2) 프론트 엔드 및 풀 스택 개발을 위해 JavaScript를 선택하십시오. Python은 데이터 처리 및 자동화 분야에서 강력한 라이브러리에 선호되는 반면 JavaScript는 웹 상호 작용 및 전체 스택 개발의 장점에 없어서는 안될 필수입니다.

파이썬 및 자바 스크립트 : 각각의 강점을 이해합니다파이썬 및 자바 스크립트 : 각각의 강점을 이해합니다May 06, 2025 am 12:15 AM

파이썬과 자바 스크립트는 각각 고유 한 장점이 있으며 선택은 프로젝트 요구와 개인 선호도에 따라 다릅니다. 1. Python은 간결한 구문으로 데이터 과학 및 백엔드 개발에 적합하지만 실행 속도가 느립니다. 2. JavaScript는 프론트 엔드 개발의 모든 곳에 있으며 강력한 비동기 프로그래밍 기능을 가지고 있습니다. node.js는 풀 스택 개발에 적합하지만 구문은 복잡하고 오류가 발생할 수 있습니다.

JavaScript의 핵심 : C 또는 C에 구축 되었습니까?JavaScript의 핵심 : C 또는 C에 구축 되었습니까?May 05, 2025 am 12:07 AM

javaScriptisNotBuiltoncorc; it'SangretedLanguageThatrunsonOngineStenWrittenInc .1) javaScriptWasDesignEdasAlightweight, 해석 hanguageforwebbrowsers.2) Endinesevolvedfromsimpleplemporectreterstoccilpilers, 전기적으로 개선된다.

JavaScript 응용 프로그램 : 프론트 엔드에서 백엔드까지JavaScript 응용 프로그램 : 프론트 엔드에서 백엔드까지May 04, 2025 am 12:12 AM

JavaScript는 프론트 엔드 및 백엔드 개발에 사용할 수 있습니다. 프론트 엔드는 DOM 작업을 통해 사용자 경험을 향상시키고 백엔드는 Node.js를 통해 서버 작업을 처리합니다. 1. 프론트 엔드 예 : 웹 페이지 텍스트의 내용을 변경하십시오. 2. 백엔드 예제 : node.js 서버를 만듭니다.

See all articles

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

Nordhold : Fusion System, 설명
1 몇 달 전By尊渡假赌尊渡假赌尊渡假赌
<exp exp> 모호한 : 원정 33- 완벽한 크로마 촉매를 얻는 방법
2 몇 주 전By尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

ZendStudio 13.5.1 맥

ZendStudio 13.5.1 맥

강력한 PHP 통합 개발 환경

SecList

SecList

SecLists는 최고의 보안 테스터의 동반자입니다. 보안 평가 시 자주 사용되는 다양한 유형의 목록을 한 곳에 모아 놓은 것입니다. SecLists는 보안 테스터에게 필요할 수 있는 모든 목록을 편리하게 제공하여 보안 테스트를 더욱 효율적이고 생산적으로 만드는 데 도움이 됩니다. 목록 유형에는 사용자 이름, 비밀번호, URL, 퍼징 페이로드, 민감한 데이터 패턴, 웹 셸 등이 포함됩니다. 테스터는 이 저장소를 새로운 테스트 시스템으로 간단히 가져올 수 있으며 필요한 모든 유형의 목록에 액세스할 수 있습니다.

Atom Editor Mac 버전 다운로드

Atom Editor Mac 버전 다운로드

가장 인기 있는 오픈 소스 편집기

MinGW - Windows용 미니멀리스트 GNU

MinGW - Windows용 미니멀리스트 GNU

이 프로젝트는 osdn.net/projects/mingw로 마이그레이션되는 중입니다. 계속해서 그곳에서 우리를 팔로우할 수 있습니다. MinGW: GCC(GNU Compiler Collection)의 기본 Windows 포트로, 기본 Windows 애플리케이션을 구축하기 위한 무료 배포 가능 가져오기 라이브러리 및 헤더 파일로 C99 기능을 지원하는 MSVC 런타임에 대한 확장이 포함되어 있습니다. 모든 MinGW 소프트웨어는 64비트 Windows 플랫폼에서 실행될 수 있습니다.

맨티스BT

맨티스BT

Mantis는 제품 결함 추적을 돕기 위해 설계된 배포하기 쉬운 웹 기반 결함 추적 도구입니다. PHP, MySQL 및 웹 서버가 필요합니다. 데모 및 호스팅 서비스를 확인해 보세요.