React筆記-Hooks(九)(非常全面)

来源:https://www.cnblogs.com/SpicyPeper/archive/2023/05/22/17421590.html
-Advertisement-
Play Games

# React筆記-Hooks(九) ## Hooks ### 概念 >React Hooks 的意思是 組件儘量寫成純函數 如果需要外部功能和副作用 就用鉤子把外部代碼"鉤"進來 ### 函數組件和類組件區別 >- 函數組件沒有狀態(state) 類組件有 >- 函數組件沒有生命周期 類組件有(掛 ...


React筆記-Hooks(九)

Hooks

概念

React Hooks 的意思是 組件儘量寫成純函數 如果需要外部功能和副作用 就用鉤子把外部代碼"鉤"進來

函數組件和類組件區別

  • 函數組件沒有狀態(state) 類組件有
  • 函數組件沒有生命周期 類組件有(掛載-更新-銷毀)
  • 函數組件沒有this 類組件有
  • 函數組件更適合做UI展示 類組件更適合做複雜的業務邏輯組件

為什麼純函數組件逐漸取代類組件

React團隊希望 組件不要變成複雜的容器 最好只是數據流的管道 開發者根據需要 組合管道即可

Hooks用法

useState() 狀態鉤子

純函數組件沒有狀態,useState()用於設置和使用組件的狀態屬性

// Hooks是無狀態的 所以用這種方式替換類組件中的狀態(state)
const [state, setState] = useState(initialValue);
// state:初始的狀態屬性,指向狀態當前值,類似this.state
// setState:修改狀態屬性值的函數,用來更新狀態,小駝峰命名
// setState((currentState) => {]})可以傳入一個函數 函數接收一個參數 用於存放當前state
// initialValue:狀態的初始值,該值會賦給state

import { useState } from 'react';



function LearnHooks () {

    let a = 1;
    const [num, setNum] = useState(0);
    const [name, setName] = useState([{name : 'bob', age : 18}, {name : 'sam', age : 20}, {name : 'kitty', age : 22}]);

    const addNum = () => {
        setNum(num + 1)
        // setNum是非同步操作 所以console.log(num)輸出的是更新前的值
        console.log(num)
        a += 1;
        // 這裡a永遠輸出為2
        // 原因是hooks重新渲染是自調用 每次都會重新把a設為1 然後執行 a + 1
        console.log(a)
    }

    console.log('LearnHooks渲染了')
    return (
        <div>
            <h1>學習hooks的userState</h1>
            <div>當前num : {num}</div>
            <button onClick={() => addNum()}>num + 1</button>
            <div>{name.map((item) => {
                    return <div key={item.name}>name : {item.name}, age : {item.age}</div>
                })}
            </div>
            <input type="text" onChange={ ({target : {value}}) => setName([{name : value}])}/>
        </div>
    );
    
}

export default LearnHooks;

useEffect() 副作用鉤子

可以實現特定的功能 如非同步請求

useEffect(() => {
    // 回調函數,其中是要進行的非同步操作代碼
    return () => {}
    // useEffect中的return語句可以用於清除effect產生的副作用。當組件卸載時,React會執行return語句中的函數,以清除effect產生的副作用。例如,如果在useEffect中訂閱了一個事件,那麼在return語句中取消訂閱可以避免記憶體泄漏。
}, [array])
// [array]:useEffect執行的依賴,當該數組的值發生改變時,回調函數中的代碼就會被執行
// 如果[array]省略,則表示不依賴,在每次渲染時回調函數都會執行
// 如果[array]是空數組,即useEffect第二項為[],表示只執行一次

import React, { useState, useEffect } from 'react';

function hook() {

  const [num, setNum] = useState(1)
  /**
   * 第一個參數是回調函數
   * 第二個參數是依賴項
   * 每次num變化時都會變化
   * 
   * 註意初始化的時候,也會調用一次
   */
  useEffect(() => {
    console.log("每次num,改變我才會觸發")
  }, [num])


  return (
    <div>
      <button onClick={() => setNum(num + 1)}>+1</button>
      <div>你好,react hook{num}</div>
    </div>
  );
}

export default hook;

useLayoutEffect()

useLayoutEffect是React提供的一個Hook,與useEffect功能類似,但在組件更新DOM前執行,而不是之後。

與useEffect不同的是,useLayoutEffect會阻塞瀏覽器渲染,並立即同步執行副作用函數。這可以確保使用組件的代碼看到的是最新的DOM佈局,因為它們在組件掛載或更新時都會在渲染通道中優先處理。

// useLayoutEffect接收兩個參數:一個副作用函數和一個依賴項數組。副作用函數中可以進行DOM操作、計算佈局等任務,並且可以通過返回一個清除函數來清理副作用產生的任何資源。

useLayoutEffect(() => { 副作用函數執行邏輯 }, [ 依賴項 ])

// 1. 編寫副作用函數:useLayoutEffect需要傳遞一個函數作為參數,該函數稱為副作用函數。在這個函數里,你可以訪問到DOM、執行非同步操作或計算等其它操作,並考慮它們的收尾工作。當組件的props或state發生變化時,都將重新運行該函數。
import { useLayoutEffect } from 'react';

function useMyLayoutEffect() {
  // 執行DOM相關操作
  return () => {
    // 清理工作
  };
}


// 2. 將useLayoutEffect掛載到組件:要在組件中使用useLayoutEffect這個函數,只需要調用它即可,並把上一步編寫的副作用函數作為第一個參數。

function MyComponent() {

  useLayoutEffect(useMyLayoutEffect, []);

  return <div>Hello, world!</div>;

}


// 給useLayoutEffect傳遞依賴項數組:和useEffect一樣,useLayoutEffect的第二個參數是一個數組,其中包含在副作用函數中需要被“監視”的任何變數。當其中的變數發生更改時,useLayoutEffect將重新運行其副作用函數。

function MyComponent({ name }) {

  useLayoutEffect(() => {

    console.log(`MyComponent is mounted: ${name}`);

  }, [name]);

  return <div>Hello, {name}!</div>;
  
}

useContext() 共用狀態鉤子

可以共用狀態,作用是進行狀態的分發,避免了使用Props進行數據的傳遞

// 第一步:創建全局的Context
const AppContext = React.createContext([初始化參數])

// 第二步:通過全局的Context進行狀態值的共用
<AppContext.Provider value={{ 屬性名: 值 }}>
    <其他組件1 />
    <其他組件2 />
</AppContext>

// 第三步:使用context
const context = useContext(AppContext);
{context.name}

// 在 App.js 文件中
import React, { createContext, useState } from 'react';
import Child from './Child';

export const MyContext = createContext();

function App() {
  const [count, setCount] = useState(0);

  return (
    <MyContext.Provider value={{ count, setCount }}>
      <div>
        <h1>Count: {count}</h1>
        <button onClick={() => setCount(count + 1)}>Increment</button>
        <Child />
      </div>
    </MyContext.Provider>
  );
}

// 在 Child.js 文件中
import React, { useContext } from 'react';
import { MyContext } from './App';

function Child() {
  const { count, setCount } = useContext(MyContext);

  return (
    <div>
      <h2>Child Component</h2>
      <h3>Count: {count}</h3>
      <button onClick={() => setCount(count - 1)}>Decrement</button>
    </div>
  );
}

useMemo() 記憶鉤子(值)

useMemo是React中的一個hook,用於優化組件的性能。它的作用是緩存函數的返回值,只有當依賴項發生變化時才重新計算。這樣可以避免在每次渲染時都重新計算函數的返回值,從而提高組件的性能。

const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]);

// 第一個參數是一個回調函數,用於計算需要緩存的值
// 第二個參數是一個數組,用於指定依賴項。只有當依賴項發生變化時,才會重新計算memoizedValue的值。

import React, { useState, useMemo } from 'react';

function MyComponent() {
  const { a, setA } = useState(1);
  const result = useMemo(() => {
    // 只有當a發生變化時才會重新計算。這樣可以避免在每次其他渲染時都重新計算結果,提高組件的性能
    return a * 2;
  }, [a]);

  const add = () => {
    setA(a + 1)
  }

  return (
    <div>
      <div>{result}</div>
      <button onClick={() => add()}>a+1</button>
    </div>
  );
}

useCallback() 記憶鉤子(函數)

useCallback是React中的一個Hook函數,用於優化函數組件的性能。它的作用是返回一個記憶化的回調函數,當依賴項發生變化時才會重新生成新的回調函數。這樣可以避免在每次渲染時都創建新的回調函數,從而提高組件的性能。

const memoizedCallback = useCallback(
  () => {
    doSomething(a, b);
  },
  [a, b],
);

// 第一個參數是回調函數
// 第二個參數是依賴項數組。當依賴項數組中的任意一個值發生變化時,useCallback會重新生成新的回調函數。如果依賴項數組為空,則每次渲染都會返回同一個回調函數。


// 需要註意的是,useCallback返回的是一個記憶化的回調函數,而不是一個普通的函數。因此,如果需要在組件外部使用該回調函數,需要將其作為props傳遞給子組件。

// 舉例:使用useCallback優化組件性能

import React, { useState, useCallback } from 'react';

function MyComponent() {
  const [count, setCount] = useState(0);

  // 普通的回調函數
  const handleClick = () => {
    setCount(count + 1);
  };

  // 使用useCallback優化的回調函數
  const handleClickMemoized = useCallback(() => {
    setCount(count + 1);
  }, [count]);

  return (
    <div>
      <p>Count: {count}</p>
      <button onClick={handleClick}>普通的回調函數</button>
      <button onClick={handleClickMemoized}>使用useCallback優化的回調函數</button>
    </div>
  );
}

export default MyComponent;

useReducer() 行為鉤子

useReducer是React中一個狀態管理的Hooks,用於處理複雜的組件狀態邏輯。它和useState類似,都是用於管理組件狀態的,但是useReducer可以更好地處理複雜的狀態邏輯,尤其是在多個狀態相互影響的情況下會更加方便和清晰。


// 1. 定義初始狀態(initial state)和reducer函數。reducer函數的作用是根據當前的狀態和操作類型(action)來返回新的狀態值。

const initialState = {
  count: 0,
};

function reducer(state, action) {
  switch (action.type) {
    case 'ADD':
      return { count: state.count + 1 };
    case 'SUB':
      return { count: state.count - 1 };
    default:
      throw new Error();
  }
}


// 2. 在組件中使用useReducer Hook,傳入reducer函數和initial state參數,獲取當前的state值和dispatch函數。

import React, { useReducer } from 'react';

function Counter() {
  const [state, dispatch] = useReducer(reducer, initialState);

  return (
    <div>
      <h2>Counter: {state.count}</h2>
      <button onClick={() => dispatch({ type: 'ADD' })}>+</button>
      <button onClick={() => dispatch({ type: 'SUB' })}>-</button>
    </div>
  );
}

useRef() 保存引用值

useRef是React中的一個hook,用於創建一個可變的引用,類似於在類組件中使用的this.refs。與useState不同,useRef返回一個可變的值,而不會觸發重新渲染組件。

useRef可以用於保存任何可變值,例如DOM元素的引用、定時器的標識符、上一個渲染周期的狀態等。它還提供了一個.current屬性來訪問保存的值。

使用場景

  1. 獲取DOM元素的引用

  2. 保存上一個渲染周期的狀態

  3. 在useEffect中訪問最新的props和state值

  4. 保存定時器的標識符,以便在組件卸載時清除


// useRef創建的ref對象與組件生命周期不相關,因此它不會在調用setState或props更新時自動更新,ref改變取決於使用它的具體方式

const ref = useRef()

// 因為hooks重新渲染其實是組件的自調用,所以我們不能在組件中直接定義一個值,let a = 1 或 const arr = []都是錯誤的,此時我們使用useRef來保存和訪問持久性數據

useImperativeHandle()

useImperativeHandle是React Hook中的一個函數,用於在使用ref時,向父組件暴露子組件的方法或屬性。它可以覆蓋預設情況下通過ref自動公開該組件實例的方式,從而更加精準的控制哪些內容公開給父組件。

useImperativeHandle接受兩個參數:ref對象和一個callback函數。callback函數應該返回包含想要掛載到ref上的任何公共方法或屬性的對象。當父組件從ref調用該方法或訪問該屬性時,callback函數定義的邏輯將被執行。

// 1. 在子組件中使用forwardRef高階組件轉發ref,以在父組件中獲得對子組件的引用。

import React, { forwardRef } from 'react';

const Child = forwardRef((props, ref) => {
  // 組件...
});

// 2. 使用useImperativeHandle Hook,將可供父組件訪問的方法或屬性包裝在一個callback函數中,並將該函數作為useImperativeHandle的第二個參數傳遞

import React, { forwardRef, useImperativeHandle } from 'react';

const Child = forwardRef((props, ref) => {
  const someMethod = () => {
    console.log('Hello from the child component');
  };

  useImperativeHandle(ref, () => ({
    someMethod,
  }));

  return <div>...</div>;
});

export default Child;

// 3. 在父組件中使用ref來調用從子組件暴露的方法

import React, { useRef } from 'react';
import Child from './Child';

function Parent() {
  const childRef = useRef(null);

  const handleClick = () => {
    // 調用子組件中公開的 someMethod 方法
    childRef.current.someMethod();
  };

  return (
    <div>
      <button onClick={handleClick}>調用子組件方法</button>
      <Child ref={childRef} />
    </div>
  );
}

其他

Immutable Data(不可變數據)

解決的問題

在 js 中,對象都是引用類型,在按引用傳遞數據的場景中,會存在多個變數指向同一個記憶體地址的情況,如果有多個代碼塊同時更改這個引用,就會產生競態

實現的原理

Persistent Data Structure(持久化數據結構):用一種數據結構來保存數據。當數據被修改時,會返回一個對象,但是新的對象會儘可能的利用之前的數據結構而不會對記憶體造成浪費,也就是使用舊數據創建新數據時,要保證舊數據同時可用且不變,同時為了避免 deepCopy把所有節點都複製一遍帶來的性能損耗,Immutable 使用了 Structural Sharing(結構共用)

redux / flux

// redux / flux 要求採用返回新對象的形式,來觸發數據更新、re-render,一般推薦的做法就是採用對象解構的方式。如果 state 對象巨大(註意:對象巨大),在結構、拷貝 state 的過程中,耗時會較長。

return {
  ...state,
  settings: {
    ...state.settings,
    profile:{
      ...state.settings.profile,
      darkmode: true,
    }
  }
}

immer

immer中文網址


// 開源庫實現思路:原始對象先做了一層 Proxy 代理,得到 draftState 傳遞給 function。function(帶副作用) 直接更改 draftState,最後 produce 返回新的對象

// 安裝
npm install immer

//使用
import React, { useState } from "react";
import produce from "immer";


export default function App() {
  const [list, setList] = useState([1, 2, 3]);

  const addMutable = () => {
    list.push("新數據");
    setList(list);
  };

  const addImmutable = () => {
    /**
     * 第一個參數是要代理的數據
     * 第二個參數是一個函數
     */
    const newVal = produce(list, draft => {
      /**
       * draft 相當於 list
       * 在這個方法裡面,可以直接修改draft,註意draft也只能在這個方法裡面修改
       * 不需要返回值,immer內部已經幫我處理好了
       */
      draft.push('新數據')
    })
    console.log(newVal)
    setList(newVal);
  };

  return (
    <div className="App">
      <button onClick={addMutable}>已可變的方式添加</button>
      <button onClick={addImmutable}>已不可變的方式添加</button>
      {list.map((item, index) => (<li key={index}>{item}</li>))}
    </div>
  );
}

函數組件傳值

hooks直接通過props傳值

//父組件中在子組件標簽上定義屬性
import 子組件
<Son 屬性={值}></Son>

// 子組件接收父組件中傳遞屬性
props.屬性
父傳子

// 父組件
import { useState } from 'react'
import Son from './son'

function Father() {


  const [data, setData] = useState(0)

  return (
    <div>
      <Son
      {/*通過定義屬性傳值*/}
      name = 'bob' d = {data}
      ></Son>
      <button onClick={() => setData(data + 1)}>+1</button>
    </div>
  )

}

export default Father


// 子組件
import { useState } from 'react'


function Son (props) {

  return (
    <div>
      <div>{props.name}</div>
      <div>{props.d}</div>
    </div>
  )
}



export default Son

子傳父

// 父組件
import { useState } from 'react'
import Son from './son'

function Father() {


  const [data, setData] = useState(0)

  const getSon = (msg) => {
    console.log(msg)
  }

  return (
    <div>
      <Son
      {/*定義一個方法用於接收子組件傳值*/}
      name = 'bob' d = {data} giveFather={(msg) => getSon(msg)}
      ></Son>
      <button onClick={() => setData(data + 1)}>+1</button>
    </div>
  )

}



export default Father


// 子組件
import { useState } from 'react'


function Son (props) {

  // 接收父組件方法 通過此方法傳值給父組件
  const set = () => {
    props.giveFather('兒子給父親的')
  }


  return (
    <div>
      <div>{props.name}</div>
      <div>{props.d}</div>
      <button onClick={set}>給父親</button>
    </div>
  )
}



export default Son

React.memo

一個高階組件,用於優化React組件的性能。它可以幫助我們避免不必要的渲染,從而提高應用程式的性能。當組件的props沒有改變時,React.memo會使用之前的渲染結果,而不會重新渲染組件。這對於那些渲染開銷較大的組件特別有用。

import React from 'react';

const MyComponent = React.memo(props => {
  // 組件代碼
}, (prevProps, currentProps) => {
  
  // prevProps 上次props
  // currentProps 當前props

  return Boolean;
  // false 渲染
  // true 不渲染
});

// 在上面的代碼中,我們將一個函數組件傳遞給React.memo(),並將其返回的新組件賦值給MyComponent。現在,MyComponent將只在其props發生更改時重新渲染。

// 需要註意的是,React.memo()僅檢查props的淺層比較。如果props包含複雜的對象或函數,可能需要手動實現更深層次的比較。
import React, { useState } from "react";

// 子組件
const SonMemo = React.memo(
  // 第一個參數 接收一個hook
  (props) => {
    return (
      <div>{props.data}</div>
    )
    // 第二個參數 接收一個函數
  }, (prevProps, currentProps) => {
    // 偶數不渲染
    // 奇數渲染
    return currentProps.data % 2 === 0
  }
)


function FatherMemo () {

  const [num, setNum] = useState(0)

  return (
    <div>
      <h1>{num}</h1>
      {/*點擊加一*/}
      <button onClick={() => setNum(num + 1)}>按鈕</button>
      <SonMemo data={num}></SonMemo>
    </div>
  )

}


export default FatherMemo;

React hooks中的過期閉包問題

什麼是閉包

過期閉包概念

過期閉包(stale closure)是指一個閉包在創建之後,所引用的外部作用域內的變數已經被修改,但閉包內仍然保存了舊值。這就導致閉包中的代碼與外部作用域內的實際狀態不一致,從而造成錯誤的結果。

useEffect中過期閉包體現和解決

// react hook中useEffect的過期閉包
import { useEffect, useState } from "react"


function ExpiredClosure () {

  const [num, setNum] = useState(0)

  useEffect(
    () => {
      setInterval(() => {
        // 這裡的num在初始化useEffect執行時取到0之後 因為閉包num值不會自動更新
        console.log(num)
      }, 2000)
    }, []
  )

return (
  <div>
    <h1>{num}</h1>
    <button onClick={() => setNum(num + 1)}>+1</button>
  </div>
)

}


export default ExpiredClosure


// 解決react hook中useEffect過期閉包問題
import { useEffect, useState } from "react"


function ExpiredClosure () {

  const [num, setNum] = useState(0)

  useEffect(
    () => {
      const timer = setInterval(() => {
        console.log(num)
      }, 2000)
      return () => {
        // 當組件卸載時 清除計時器
        clearInterval(timer)
      }
      // 添加num為依賴項
    }, [num]
  )

return (
  <div>
    <h1>{num}</h1>
    <button onClick={() => setNum(num + 1)}>+1</button>
  </div>
)

}


export default ExpiredClosure

useState中過期閉包體現和解決

// react hook中useState的過期閉包
import { useState } from "react";


function ExpiredClosure2 () {

  const [num, setNum] = useState(0)


  // 點擊正常+1
  const add = () => {
    setNum(num + 1)
  }
  
  // 假設點擊時num為3 兩秒+2 = 5 在兩秒之間不管點擊多少次+1操作num變為678910...最後num都為5
  const add2 = () => {
    setTimeout(() => {
      setNum(num + 2)
    }, 2000)
  }

  // 當我們點擊+2時候會取得當前值 之後點擊其他改變num值 +2中的num都不會隨之改變 兩秒後取得的num+2給setNum後 渲染頁面

  return (
    <div>
      <h1>{num}</h1>
      <button onClick={add}>+1</button><br />
      <button onClick={add2}>+2</button>
    </div>
  )

}


export default ExpiredClosure2;


// 解決react hook中useState過期閉包問題
import { useState } from "react";


function ExpiredClosure2 () {

  const [num, setNum] = useState(0)



  const add = () => {
    setNum(num + 1)
  }

  const add2 = () => {
    setTimeout(() => {
      // setNum中可以傳入一個函數 這個函數接收一個參數 用於獲取當前num值
      setNum((currentNum) => currentNum + 2)
    }, 2000)
  }

  return (
    <div>
      <h1>{num}</h1>
      <button onClick={add}>+1</button><br />
      <button onClick={add2}>+2</button>
    </div>
  )

}


export default ExpiredClosure2;


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • # 概述 Typecho是一款輕量級的開源PHP博客系統,它簡單易用,界面整潔,性能高效,主題、插件眾多。我使用的是騰訊雲輕量伺服器,Typecho的應用模版,一鍵安裝環境。構建自己的博客網站,記錄生活、分享經驗。 ## 購買功能變數名稱、備案、申請SSL 這樣在之後創建完typecho伺服器,就會在ngi ...
  • 本文是線上問題處理案例系列之一,旨在通過真實案例向讀者介紹發現問題、定位問題、解決問題的方法。本文講述了從垃圾回收耗時過長的表象,逐步定位到資料庫連接池保活問題的全過程,並對其中用到的一些知識點進行了總結。 ...
  • 摘要:本文詳細梳理分析了DWS服務面臨軟硬體故障場景和對應的修複原理,希望藉此能夠讓你對DWS的集群故障修複有個全面深入的瞭解。 本文分享自華為雲社區《GaussDB(DWS)故障修複系統性介紹》,作者: 聞鮮生。 DWS是一個分散式架構的MPP集群,物理部署上涉及數百數千台主機和對應的磁碟,以及這 ...
  • **本系列為:MySQL資料庫詳解,為千鋒資深教學老師獨家創作** **致力於為大家講解清晰MySQL資料庫相關知識點,含有豐富的代碼案例及講解。如果感覺對大家有幫助的話,可以【關註】持續追更\~** **文末有本文重點總結,技術類問題,也歡迎大家和我們溝通交流!** ![在這裡插入圖片描述](ht ...
  • 《1萬多條司法資格考試題庫ACCESS版》搜集了大量司法資格考試試題,包括試卷一、試卷二、試卷三、試卷四等科目。同類的資料庫有《9萬多條執業醫師資格考試題庫ACCESS資料庫》、《6萬多條會計從業資格考試題庫ACCESS版》、《近7萬多條證券從業資格考試題庫ACCESS版》、《1萬多條一級建造師資格 ...
  • 摘要:MySQL一張表最多能存多少數據? 本文分享自華為雲社區《為什麼MySQL單表不能超過2000萬行?》,作者: GaussDB 資料庫 。 最近看到一篇《我說MySQL每張表最好不要超過2000萬數據,面試官讓我回去等通知》的文章,非常有趣。 文中提到,他朋友在面試的過程中說,自己的工作就是把 ...
  • 數學上有一個“計算漢明重量”的問題,即求取一個二進位位中非 0 的數量。使用 Redis 提供的 Bitmap 統計時恰恰是這樣一個問題,學習後能發現解決辦法卻是如此巧妙。 ...
  • 2022年的程式員節, #大齡程式員去哪兒了#成為了社交媒體上最火的話題之一,程式員的職場成長問題在社會上引起了廣泛關註。 有2位在技術領域摸爬滾打很多年的開發者,35歲後的他們,有70後,有80後,依然在編程開發,依然有離職創業的勇氣,努力實現自己的人生價值。走進他們的故事,你會發現,這個世上沒有 ...
一周排行
    -Advertisement-
    Play Games
  • 通過WPF的按鈕、文本輸入框實現了一個簡單的SpinBox數字輸入用戶組件並可以通過數據綁定數值和步長。本文中介紹了通過Xaml代碼實現自定義組件的佈局,依賴屬性的定義和使用等知識點。 ...
  • 以前,我看到一個朋友在對一個系統做初始化的時候,通過一組魔幻般的按鍵,調出來一個隱藏的系統設置界面,這個界面在常規的菜單或者工具欄是看不到的,因為它是一個後臺設置的關鍵界面,不公開,同時避免常規用戶的誤操作,它是作為一個超級管理員的入口功能,這個是很不錯的思路。其實Winform做這樣的處理也是很容... ...
  • 一:背景 1. 講故事 前些天有位朋友找到我,說他的程式每次關閉時就會自動崩潰,一直找不到原因讓我幫忙看一下怎麼回事,這位朋友應該是第二次找我了,分析了下 dump 還是挺經典的,拿出來給大家分享一下吧。 二:WinDbg 分析 1. 為什麼會崩潰 找崩潰原因比較簡單,用 !analyze -v 命 ...
  • 在一些報表模塊中,需要我們根據用戶操作的名稱,來動態根據人員姓名,更新報表的簽名圖片,也就是電子手寫簽名效果,本篇隨筆介紹一下使用FastReport報表動態更新人員簽名圖片。 ...
  • 最新內容優先發佈於個人博客:小虎技術分享站,隨後逐步搬運到博客園。 創作不易,如果覺得有用請在Github上為博主點亮一顆小星星吧! 博主開始學習編程於11年前,年少時還只會使用cin 和cout ,給單片機點點燈。那時候,類似async/await 和future/promise 模型的認知還不是 ...
  • 之前在阿裡雲ECS 99元/年的活動實例上搭建了一個測試用的MINIO服務,以前都是直接當基礎設施來使用的,這次準備自己學一下S3相容API相關的對象存儲開發,因此有了這個小工具。目前僅包含上傳功能,後續計劃開發一個類似圖床的對象存儲應用。 ...
  • 目錄簡介快速入門安裝 NuGet 包實體類User資料庫類DbFactory增刪改查InsertSelectUpdateDelete總結 簡介 NPoco 是 PetaPoco 的一個分支,具有一些額外的功能,截至現在 github 星數 839。NPoco 中文資料沒多少,我是被博客園群友推薦的, ...
  • 前言 前面使用 Admin.Core 的代碼生成器生成了通用代碼生成器的基礎模塊 分組,模板,項目,項目模型,項目欄位的基礎功能,本篇繼續完善,實現最核心的模板生成功能,並提供生成預覽及代碼文件壓縮下載 準備 首先清楚幾個模塊的關係,如何使用,簡單畫一個流程圖 前面完成了基礎的模板組,模板管理,項目 ...
  • 假設需要實現一個圖標和文本結合的按鈕 ,普通做法是 直接重寫該按鈕的模板; 如果想作為通用的呢? 兩種做法: 附加屬性 自定義控制項 推薦使用附加屬性的形式 第一種:附加屬性 創建Button的附加屬性 ButtonExtensions 1 public static class ButtonExte ...
  • 在C#中,委托是一種引用類型的數據類型,允許我們封裝方法的引用。通過使用委托,我們可以將方法作為參數傳遞給其他方法,或者將多個方法組合在一起,從而實現更靈活的編程模式。委托類似於函數指針,但提供了類型安全和垃圾回收等現代語言特性。 基本概念 定義委托 定義委托需要指定它所代表的方法的原型,包括返回類 ...