TIL 2022-12-30

JYR00·2022년 12월 30일
0

TIL

목록 보기
52/60

java505_react_test2 프로젝트 생성


이런거 markup language로 만드거임



파일명이랑 프로젝트명 다르면 package.json에 name 변경해줘야한다

JSX -> 함수형 클래스형 두 가지로 나뉘는데 우리는 함수형으로 진행할 예정


index.html root - > potato로 바꾸면 아무것도 뜨지 않는다




이렇게 오류가 난다.

potato.jsx생성



화면 구성 차이점 앞이 SPRING 뒤가 리엑트
자바스크립트를 사용하면서 이용한다.

function Movie(){
    return(
        <h2>나는 감자가 조아</h2>
    );
}






function App() {
  //  이곳은 일반적인 js 소스코드 부분



    //return 화면을 생성하는 가상 DOM 부분 (virtual dom)
    return (
        //부모 태그는 반드시 한 개만 존재해야 함 (자식태그는 상관없음)
        //jsx 문법에서 class라는 키워드는 자바 스크립트의 클래스 키워드와 겹치기 때문에
        //자바스크립트의 클래스와 css 선택자의 클래스를 구별을 해주기 위해서
        //css선택자인 class를 className으로 사용한다
        <div className="App">
        {/*    SPA: Single Page Application 의 줄임말, (spring은 multipage application)
                index.html와 같은 하나의 화면 구성 파일로 모든 화면을 구성하는 웹 페이지*/}
        {/*    SPA는 프론트엔드 프레임워크를 사용하여 하나의 화면에서 필요한 부분을 실시간으로 변경하여
                웹을 표시한다. 해당 화면을 구성하는 요소를 컴포넌트라고 한다.*/}
        {/*    컴포넌트 : 리엑트에서 화면을 구성하는 요소*/}

        <h2>리액트 사용하기</h2>
        <Movie/>
        <Movie/>
        </div>

function 이름 바뀌어도 잘 작동된다



리엑트에서 dom개념 설명해 놓은 블로그
https://velog.io/@mollog/React%EC%97%90%EC%84%9C%EC%9D%98-%EA%B0%80%EC%83%81%EB%8F%94-%EA%B0%9C%EB%85%90

<Food fav = {"kimchi"} something={true} papapapa={['hello',1,2,3,4,true]} ></Food>


이렇게 콘솔창에 뜬다


import './App.css';
// import Potatao from "./Potatao";
// 반드시 경로명을 적어야 한다


// 이렇게 한 파일에서 function을 만들어 넣어도 된다
// props : 부모 컴포넌트에서 자식 컴포넌트로 데이터를 전달하는 매개변수
// props는 읽기 전용 
function Food(props){
    const some = props.something;
    console.log(props);
    return(
        <h2>I love {props.fav}</h2>
    );
}


function App() {
  //  이곳은 일반적인 js 소스코드 부분



    //return 화면을 생성하는 가상 DOM 부분 (virtual dom)
    return (
        //부모 태그는 반드시 한 개만 존재해야 함 (자식태그는 상관없음)
        //jsx 문법에서 class라는 키워드는 자바 스크립트의 클래스 키워드와 겹치기 때문에
        //자바스크립트의 클래스와 css 선택자의 클래스를 구별을 해주기 위해서
        //css선택자인 class를 className으로 사용한다

        
        
        
        
        <div className="App">
        {/*    SPA: Single Page Application 의 줄임말, (spring은 multipage application)
                index.html와 같은 하나의 화면 구성 파일로 모든 화면을 구성하는 웹 페이지*/}
        {/*    SPA는 프론트엔드 프레임워크를 사용하여 하나의 화면에서 필요한 부분을 실시간으로 변경하여
                웹을 표시한다. 해당 화면을 구성하는 요소를 컴포넌트라고 한다.*/}
        {/*    컴포넌트 : 리엑트에서 화면을 구성하는 요소*/}

            {/* {Food라는 컴포넌트에 변수명이 kimchi인 데이터를 전달한다}*/}

            {/*{}를 사용해야만 데이터가 전달이 된다 그냥 다 감싸 */}
            <Food fav = {"kimchi"} something={true} papapapa={['hello',1,2,3,4,true]} ></Food>
            <Food fav={"돈까스"}/>
            <Food fav={"햄버거"}/>
            <Food fav={"스시"}/>
            <h2>리액트 사용하기</h2>
            {/*    함수안에 들어있는 정보를 화면에 뿌려준다*/}

        </div>

    //    오로지 div 안에서만 구성해야 한다
    );
}
// return 화면 렌더링 - 무조건 한 개의 html태그만 사용한다
// 주로 <div>로 부모태그만들고 안에 화면을 그린다. return 뒤에 있는 ()는 구분하는 용도. 어디서부터 어디까지 return 되는지 헷갈리니까
// 안 붙여도 괜찮지만 붙이는 것이 더 바람직하다. (한 줄 이상 될 경우)


//export에 default가 붙어있다면 다른 곳에서 import할 때 {}를 붙여야하는데 안 붙이고 바로 호출하는 것이 가능하다
export default App;


이런 형태로 props사용해도 된다

컴포넌트와 props는 꼭 기억해야한다!

import './App.css';
// import Potatao from "./Potatao";
// 반드시 경로명을 적어야 한다


// 이렇게 한 파일에서 function을 만들어 넣어도 된다
// props : 부모 컴포넌트에서 자식 컴포넌트로 데이터를 전달하는 매개변수
// props는 읽기 전용 
//  사용법 :
// 1) 컴포넌트 명(props){
//  {props.부모가 전달한 키 이름}
// }

// 2) 컴포넌트 명({부모가 전달한 키 이름, ...}) {
//  {키 이름1}, {키 이름2}, ...
//  console.log({키이름1});
// }

// 3) 컴포넌트명(props){
//      const {키이름, ...} = props;
//      console.log({키이름});
// }



// 1번
// function Food(props){
//     const some = props.something;
//     console.log(props);
//     return(
//         <h2>I love {props.fav}</h2>
//     );
// }


// 2번
function Food({name, pic}){
    // const some = name.something;
    // console.log(name);
    return(
        <div>
            <h2>I love {name}</h2>
            <img src={pic}/>
        </div>

    );
}


// 3번
// function Food(props){
//     const some = props.something;
//     console.log(props);
//     return(
//         <h2>I love {props.fav}</h2>
//     );
// }



// 새로운 예제
const foodList = [
    {
        name: "돈까스",
        image : "https://t1.daumcdn.net/thumb/R720x0/?fname=http://t1.daumcdn.net/brunch/" +
            "service/user/3YN/image/XIsief9ks0Re5EL1N87ztiBfaok.jpg"
    },
    {
        name :"순대국밥",
        image: "" +
            "1NbmozXDGRW4ArRNbilONorEVkV6uOoxFYismtaNgo9HWsRXq9RCYmsE141rXAM1msVmgAyK9Na140" +
            "TjYVhS+VeO1aCuONhWaxWze9EB//2Q=="
    },
    {
        name : "햄버거",
        image: "" +
            "s6DTKjBdBHqar6mcwvYvBp/ucx/4mZvoTH0pxguG2rX+ztW0/yIq/kKt15TqEVhFcqknlnte0UUwoUUUUAFFFFABRRRQAUUUUAf/Z"
    }
];



function App() {
  //  이곳은 일반적인 js 소스코드 부분 
  //  원래 이렇게 넣어야 했다
  //   foodList.map(function(dish:{image,name}){
  //       <Food name={dish.name}/>
  //   });




    //return 화면을 생성하는 가상 DOM 부분 (virtual dom)
    return (
        //부모 태그는 반드시 한 개만 존재해야 함 (자식태그는 상관없음)
        //jsx 문법에서 class라는 키워드는 자바 스크립트의 클래스 키워드와 겹치기 때문에
        //자바스크립트의 클래스와 css 선택자의 클래스를 구별을 해주기 위해서
        //css선택자인 class를 className으로 사용한다


        
        
        
        <div className="App">
            {/*    SPA: Single Page Application 의 줄임말, (spring은 multipage application)
                index.html와 같은 하나의 화면 구성 파일로 모든 화면을 구성하는 웹 페이지*/}
            {/*    SPA는 프론트엔드 프레임워크를 사용하여 하나의 화면에서 필요한 부분을 실시간으로 변경하여
                웹을 표시한다. 해당 화면을 구성하는 요소를 컴포넌트라고 한다.*/}
            {/*    컴포넌트 : 리엑트에서 화면을 구성하는 요소*/}

            {/*{} : jsx 문법에서 {}데이터를 표현하는 데 사용함, 간단한 연산식도 사용가능*/}

            {/* {Food라는 컴포넌트에 변수명이 kimchi인 데이터를 전달한다}*/}

            {/*{}를 사용해야만 데이터가 전달이 된다 그냥 다 감싸 */}

            {foodList.map(dish => <Food name = {dish.name} pic={dish.image}/>)}

            {/*<Food fav = {"kimchi"} something={true} papapapa={['hello',1,2,3,4,true]} ></Food>*/}
            {/*<Food fav={"돈까스"}/>*/}
            {/*<Food fav={"햄버거"}/>*/}
            {/*<Food fav={"스시"}/>*/}
            <h2>리액트 사용하기</h2>
            {/*    함수안에 들어있는 정보를 화면에 뿌려준다*/}

        </div>

    //    오로지 div 안에서만 구성해야 한다
    );
}
// return 화면 렌더링 - 무조건 한 개의 html태그만 사용한다
// 주로 <div>로 부모태그만들고 안에 화면을 그린다. return 뒤에 있는 ()는 구분하는 용도. 어디서부터 어디까지 return 되는지 헷갈리니까
// 안 붙여도 괜찮지만 붙이는 것이 더 바람직하다. (한 줄 이상 될 경우)


//export에 default가 붙어있다면 다른 곳에서 import할 때 {}를 붙여야하는데 안 붙이고 바로 호출하는 것이 가능하다
export default App;


unique key값이 없기 때문에 오류



prop-type 설치해야 한다. 터미널에다 쳐야함
npm install prop-types


삿갓 표시는 이 숫자 이상이면 잘 작동된다는 의미다

Food.propTypes = {
    name : PropTypes.string.isRequired, //값 안들어오면 오류체크하라는 의미
    pic : PropTypes.string.isRequired,
    rating : PropTypes.string.isRequired
};


데이터형이 맞지 않다고 뜸



Food.propTypes = {
    name : PropTypes.string.isRequired, //값 안들어오면 오류체크하라는 의미
    img : PropTypes.string.isRequired,
    rating : PropTypes.number.isRequired
};

변수 이름이 달라지면 틀렸다고 알려줌
원래 pic인데 img로 바꾸어보았다



State

ppt 5장

App2.jsx생성
클래스로 import하는 방법



constructor(props) {
        super(props);
        console.log('생성자 실행');
    }
    componentDidMount() {
        console.log('컴포넌트 생성 완료')
    }
    componentDidUpdate(prevProps, prevState, snapshot) {
        console.log("컴포넌트 업데이트 완료")
    }
    componentWillUnmount() {
        console.log('컴포넌트 제거 완료')
    }

+버튼 누르면

마이너스 버튼 누르면


App2.jsx 최종

import React from "react";

// 클래스 타입의 컴포넌트 (요즘 많이 사용하지는 않는 편)
// 클래스 컴포넌트는 React.Component를 상속받아 사용함
// 함수형이 더 간편하다
class App2 extends React.Component {
    
    // state : 현재 컴포넌트에서 사용하는 상태 값
    // props는 부모 컴포넌트에서 전달되는 값이지만 state는 현재 컴포넌트에서만 사용되는 값으로 수정 가능함
    // props는 수정 불가능.
    // state의 값이 변경되면 리엑트는 화면을 다시 렌더링을 함
    // state의 값을 직접적으로 변경하는 것은 불가능, setState()함수를 사용하여 값을 수정
    
    
    state = {
        count : 0,
    };

    plus = () => {
        this.setState({count:this.state.count + 1}) //이렇게 해야 값이 변동이 된다.
        console.log('plus');
    }

    minus = () => {
        console.log('minus')
        this.setState({count: this.state.count-1})
    }

    // render() : 클래스 컴포넌트에서 화면을 렌더링하기 위한 메서드
    // render은 함수 컴포넌트와 사용 방법이 동일하다
    render() {
        return(
            <div>
                <h1>클래스 컴포넌트 App2</h1>
                <h3>카운트 수 : {this.state.count}</h3>
            {/*    버튼 추가*/}
                <button onClick={this.plus}>plus</button>
                <button onClick = {this.minus}>minus</button>
            </div>
        );
    }

    constructor(props) {
        super(props);
        console.log('생성자 실행');
    }
    
    //componentDidMount,componentDidUpdate,componentWillUnmount는 리액트의 생명주기에 관련되어 있는 이벤트 함수
    // componentDidMount() : 해당 컴포넌트가 DOM에 추가된 후 동작하는 이벤트
    // componentDidUpdate() : 해당 컴포넌트의 데이터가 변경된 후 동작하는 이벤트
    // componentWillUnmount() : 해당 컴포넌트가 더이상 사용되지 않아 삭제된 후 동작하는 이벤트
    //클래스 컴포넌트에서만 사용할 수 있음
    // 함수 컴포넌트에서는 해당 이벤트 함수들을 사용할 수 없음
    // 함수 컴포넌트에서 해당 이벤트 함수들을 사용하기 위해서 HOOKS라는 기능을 추가하여 해당 이벤트를 구현함
    componentDidMount() {
        console.log('컴포넌트 생성 완료')
    }
    componentDidUpdate(prevProps, prevState, snapshot) {
        console.log("컴포넌트 업데이트 완료")
    }
    // componentWillUnmount() {
    //     console.log('컴포넌트 제거 완료') //삭제할 때 사용됨
    // }

}

export default App2;

Notification.jsx

import React from "react";

const styles={
    wrapper : {
        margin:8,
        padding:8,
        display: 'flex',
        flexDirection : 'row', //밑으로 쌓이게끔
        boarder :'1px solid gray',
        borderRadius : 16,
    },
    messageText : {
        color :'black',
        fontSize:16,

    },
};

class Notification extends React.Component {
    constructor(props) {
        super(props);

        this.state={};
    }
    render() {
        return(
            <div style={styles.wrapper}>
                <span style = {styles.messageText}>{this.props.message}</span>
            </div>
        );
    }
}
export default Notification;

NotificationList.jsx

//NotificationList.jsx
import React from "react";
import Notification from "./Notification";
import notification from "./Notification";

//js 배열은 다른 언어의 list와 같다.

// 댓글 데이터 생성
const reservedNotifications = [
    {id:1,message : "안녕하세요, 오늘 일정 알려드립니다"},
    {id:2,message: "오후 수업 시간입니다."},
    {id:3,message: "이제 곧 쉬는 시간입니다"},
];

// 자바스크립트 타이머 객체 정보를 저장하는 변수
// setTimeout : 1회용 타이머 , 지정된 시간 이후에 한 번 동작, 실행시 타이머 정보를 반환
//              타이머 삭제 시 clearTimeout()을 사용
// setInterval : 지정된 시간마다 동작하는 타이머,실행시 타이머 정보를 반환 
//               멈출수가 없음. 타이머 삭제 시 clearInterval()을 사용한다
var timer;


class NotificationList extends React.Component {
    constructor(props) {
        super(props);

        // 현재 컴포넌트의 상태인 notifications 선언
        this.state = {
            // 빈 배열 타입인 state가 생성
            notifications:[],
            // notifications: reservedNotifications,
        };
    }

    // 화면에 처음 그려질 내용
    render() {
        return (
                // state의 notifications 배열을 가지고 화면을 그려줌
            // notifications 배열의 기본값이 비었기 때문에 화면에 아무것도 그리지 않는다
        // notifications:[] 이 값을 가져와 돌린다.
        // 지금 빈 배열이라 돌릴 수 없다. 그래서 밑에 부분이 화면에 그려지지 않는다.
        <div>
            {
                this.state.notifications.map((item) => {
                    return <Notification key={item.id} message={item.message}/>
                })
            }

            {/*원래는 아래 처럼 해야 했다. 하지만 귀찮으니까 위의 것 처럼 행한다.*/}
            {/*<Notification key={reservedNotifications[0].id} message = {reservedNotifications[0].message}/>,*/}
            {/*<Notification key={reservedNotifications[1].id} message = {reservedNotifications[0].message}/>,*/}
            {/*<Notification key={reservedNotifications[2].id} message = {reservedNotifications[0].message}/>,*/}
        </div>
    )

    }

    // render() 함수 실행 후 동작(다 끝내고 동작하는 함수)
    componentDidMount() {
        // object 타입의 확장 표현식을 통해서 변수 notifications에 state가 가지고 있는 notifications의 데이터를
        // 대입한다.
        //const/let [변수명1, 변수명2, ...] = [원본배열] -> 배열 값을 차례대로 하나씩 대입시킨다.
        // const val1 = 10; 이거랑 유사하다.
        // const val2 = 20;

        const {notifications} = this.state;
        //state의 notification의 데이터가 비어 있음
        // const {notificaions} = this.state.notifications
        // 랑 같은 거.(확장표현식)


        // 타이머를 사용하여 지정된 시간마다 동작하도록 설정함
        // timer = setInterval(() => {
        //     if (notifications.length < reservedNotifications.length) { //notifications.length는 무조건 0으로 뜨고 reservedNotifications.length는 3
        //         const index = notifications.length;
        //         // 배열 notifications에 데이터 추가
        //         notifications.push(reservedNotifications[index]); //데이터 하나를 추가해준다
        //
        //         //state의 상태 수정
        //         this.setState({
        //             // this.state에 있는 notifications에 현재 componentDidMount 안에 있는 지역변수
        //             // notifications의 데이터를 저장한다
        //             notifications: notifications,
        //         });
        //     } else {
        //         clearInterval(timer);
        //     }
        // }, 2000);


        setTimeout(() => {
            this.setState({
                notifications : reservedNotifications,
            });
            console.log("데이터 추가 완료");
        }, 3000);

        console.log("마운트 완료")
    }

    componentDidUpdate() {
        console.log("상태 업데이트")
    }
}

export default NotificationList;

index.js

import React from 'react';
import ReactDOM from 'react-dom/client';
import './index.css';
// default로 되어있어서 {}없이 바로 import 가능하다
import App from './App';
import reportWebVitals from './reportWebVitals';
import App2 from "./App2";
import NotificationList from "./mounts/NotificationList";

const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(
    <React.StrictMode>
    {/*    js의 문법. js의 return부분을 사용하겠다는 의미.
    jsx = js + html + xml
    여기에 넣을 때 전부 시작태그 끝 태그 다 있어야 한다.
    <APP></APP> 혹은 <APP / > 둘 중 하나를 골라 해야한다.*/}

    {/*    무조건 대문자A를 사용해줘야 한다.*/}
    {/*    jsx문법에서 화면을 렌더링하기 위한 태그의 이름은 반드시 첫글자가 "대문자"이어야 함*/}
    {/*    대문자를 사용하는 이유는 일반적으로 html 태그를 모두 소문자로 사용하기 때문에 일반 html태그인지
            jsx문법으로 생성된 태그인지를 구분하기 위함이다*/}

    <App />
        <br/><hr /><br />
    <App2/>
        <NotificationList/> <----- 여기 추가됨!!
        <br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br /><br />
    </React.StrictMode>
    // <h2>이렇게 부모태그가 2개가 되면 오류가 발생한다.</h2>
);
reportWebVitals();
























0개의 댓글