文件瀏覽器及數位相框 -2.3.2-freetype_arm-2

来源:http://www.cnblogs.com/CZM-/archive/2016/03/27/5325524.html
-Advertisement-
Play Games

顯示多行文字 兩行文字左邊對齊 簡單使用兩個迴圈顯示兩行字體 根據上一行字體的寬度來進行下一行左邊的計算 #include #include #include #include #include #include #include #include #include #include #includ... ...


 

顯示多行文字

兩行文字左邊對齊

簡單使用兩個迴圈顯示兩行字體

根據上一行字體的寬度來進行下一行左邊的計算

 

#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <linux/fb.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <math.h>
#include <wchar.h>
#include <ft2build.h>
#include <stdlib.h>
#include FT_FREETYPE_H
#include FT_GLYPH_H   


#include <linux/font.h>

    int fd_fb;
    struct fb_var_screeninfo var;    /* Current var */
    struct fb_fix_screeninfo fix;    /* Current fix */
    int screen_size;                    //空間大小
    unsigned char *fb_mem;                //framebaffer空間地址
    unsigned int line_width;
    unsigned int pixel_width;

void lcd_put_pixel(int x, int y, unsigned int color)
{
    unsigned char *pen_8 = fb_mem + y * line_width + x * pixel_width;     //當前像素對應記憶體位置
    unsigned short *pen_16;
    unsigned int *pen_32;

    unsigned int red, blue, green;
    
    pen_16 = (unsigned short *)pen_8;
    pen_32 = (unsigned int *)pen_8;
    
    switch(var.bits_per_pixel)
    {
        case 8:
        {
            *pen_8 = color;            //對應調色板顏色
            
            break;
        }
        case 16:
        {
            /* 5*6*5 */
            red   = (color >> 16) & 0xff;
            green = (color >> 8) & 0xff;
            blue  = (color >> 0) & 0xff;

            color = ((red >> 3 ) << 11) | ((green >> 2) << 5) | ( blue >> 3);
            
            /* 顏色數據為高位 */
            *pen_16 = color;
            
            break;
        }
        case 32:
        {
            *pen_32 = color;
            break;
        }
        
    }

}

void draw_bitmap( FT_Bitmap*  bitmap,
             FT_Int      x,
             FT_Int      y)
{
  FT_Int  i, j, p, q;
  FT_Int  x_max = x + bitmap->width;
  FT_Int  y_max = y + bitmap->rows;

    printf("x = %d, y = %d\n", x, y);
    
  for ( i = x, p = 0; i < x_max; i++, p++ )
  {
    for ( j = y, q = 0; j < y_max; j++, q++ )
    {
      if ( i < 0      || j < 0       ||
           i >= var.xres || j >= var.yres )
        continue; 

    //  image[j][i] |= bitmap->buffer[q * bitmap->width + p];

        lcd_put_pixel(i, j, bitmap->buffer[q * bitmap->width + p]);
    }
  }
}


int main(int argc, char **argv)
{
    wchar_t * str1 = L"陳志朋gif";
    wchar_t * str2 = L"hello the world";
    FT_BBox bbox;  
    FT_Glyph  glyph;
    
    
      FT_Library    library;
      FT_Face       face;
    FT_Vector     pen;                    /* untransformed origin  */
    FT_GlyphSlot  slot;
    FT_Matrix     matrix;                 /* transformation matrix */

    int error;
    double angle;
    int i;
    int line_box_ymin = 10000;
    int line_box_ymax = 0;
    
    if(argc != 2)
    {
        printf("Usage : %s <font_file> <angle>\n",argv[0]);
        return -1;
    }

    
    fd_fb = open("/dev/fb0",O_RDWR);
    if(fd_fb < 0)
    {
        printf("can't open /dev/fb0 \n");
        return -1;
    }
    if(ioctl(fd_fb, FBIOGET_VSCREENINFO, &var))
    {
        printf("can't get var \n");
        return -1;    
    }
    if(ioctl(fd_fb, FBIOGET_FSCREENINFO, &fix))
    {
        printf("can't get fix \n");
        return -1;    
    }
    screen_size = var.xres * var.yres * var.bits_per_pixel / 8;    //單位位元組
    line_width = var.xres *  var.bits_per_pixel / 8;
    pixel_width =  var.bits_per_pixel / 8;
    
    fb_mem = (unsigned char *)mmap(NULL, screen_size, \
        PROT_READ | PROT_WRITE, MAP_SHARED, fd_fb, 0);
    if(fb_mem == (unsigned char *) -1)
    {
        printf("can't mmap \n");
        return -1;
    }
    memset(fb_mem, 0, screen_size);

    
    /* 顯示矢量文字 */
    
  error = FT_Init_FreeType( &library );              /* initialize library */
  /* error handling omitted */

  error = FT_New_Face( library, argv[1], 0, &face ); /* create face object */
  /* error handling omitted */

 slot = face->glyph;
    
  FT_Set_Pixel_Sizes(face, 24, 0);

 
    
  /*確定坐標
  *lcd_x = 0 
  *lcd_y = 24
  *笛卡爾坐標
  *x = lcd_x = 0 
  *y = var.yres -  24
  */
  
  pen.x = ( 0 ) * 64;
  pen.y = ( var.yres -  24 ) * 64;

  for(i = 0; i < wcslen(str1); i++)
  {
        
      /* set transformation */
      FT_Set_Transform( face,0, &pen );

      /* load glyph image into the slot (erase previous one) */

      error = FT_Load_Char( face, str1[i], FT_LOAD_RENDER );
      if ( error )
      {
          printf("FT_Load_Char error\n");
        return -1;
      }
      
        error = FT_Get_Glyph( face->glyph, &glyph );
        if (error)
        {
            printf("FT_Get_Glyph error!\n");
            return -1;
        }
        
      FT_Glyph_Get_CBox(glyph, FT_GLYPH_BBOX_TRUNCATE, &bbox );

      if(line_box_ymin > bbox.yMin)
          line_box_ymin = bbox.yMin;
      if(line_box_ymax < bbox.yMax)
          line_box_ymax = bbox.yMax;
      
       /* now, draw to our target surface (convert position) */
       draw_bitmap( &slot->bitmap,
                     slot->bitmap_left,
                     var.yres - slot->bitmap_top );
      pen.x += slot->advance.x;
            
      }
      
  /*確定坐標
  *lcd_x = 0 
  *lcd_y = 24
  *笛卡爾坐標
  *x = lcd_x = 0 
  *y = var.yres -  24
  */
  
  pen.x = ( 0 ) * 64;
  pen.y = ( var.yres -  (line_box_ymax - line_box_ymin + 24)  ) * 64;

  for(i = 0; i < wcslen(str2); i++)
  {
        
      /* set transformation */
      FT_Set_Transform( face,0, &pen );

      /* load glyph image into the slot (erase previous one) */

      error = FT_Load_Char( face, str2[i], FT_LOAD_RENDER );
      if ( error )
      {
          printf("FT_Load_Char error\n");
        return -1;
      }
      
        error = FT_Get_Glyph( face->glyph, &glyph );
        if (error)
        {
            printf("FT_Get_Glyph error!\n");
            return -1;
        }
        
      FT_Glyph_Get_CBox(glyph, FT_GLYPH_BBOX_TRUNCATE, &bbox );

      if(line_box_ymin > bbox.yMin)
          line_box_ymin = bbox.yMin;
      if(line_box_ymax < bbox.yMax)
          line_box_ymax = bbox.yMax;
      
       /* now, draw to our target surface (convert position) */
       draw_bitmap( &slot->bitmap,
                     slot->bitmap_left,
                     var.yres - slot->bitmap_top );
      pen.x += slot->advance.x;
            
      }
}

 

使用freetpye中的函數實現一行文字居中

 

4.6 高級文本渲染:變換 + 居中 + 字距調整
現在我們將修改我們的代碼,以便可以容易地變換已渲染的字元串,例如旋轉它。我們將以實行少許小
改進開始:
4.6.1 打包然後平移字形
我們先把與一個字形圖像相關的信息打包到一個結構體,而不是並行的數組。因此我們定義下麵的結構
體類型:
typedef struct TGlyph_
{
FT_UInt index; /* 字形索引 */
FT_Vector pos; /* 基線上面的字形原點 */
FT_Glyph image; /* 字形圖像 */
} TGlyph, *PGlyph;
我們在裝載每一個字形圖像過程中,在把它裝載它在基線所在位置後便直接平移它。我們將看到,這有若幹
好處。我們的字形序列裝載其因而變成:
FT_GlyphSlot slot = face->glyph; /* 一個小捷徑 */
FT_UInt glyph_index;
FT_Bool use_kerning;
FT_UInt previous;
int pen_x, pen_y, n;
TGlyph glyphs[MAX_GLYPHS]; /* 字形表 */
PGlyph glyph; /* 表中的當前字形*/
FT_UInt num_glyphs;
... 初始化庫 ...
... 創建 face 對象 ...
... 設置字元尺寸 ...
pen_x = 0; /* 以 (0,0) 開始 */
pen_y = 0;
num_glyphs = 0;
use_kerning = FT_HAS_KERNING( face );
previous = 0;
glyph = glyphs;
for ( n = 0; n < num_chars; n++ )
{
glyph->index = FT_Get_Char_Index( face, text[n] );
if ( use_kerning && previous && glyph->index )
{
FT_Vector delta;
FT_Get_Kerning( face, previous, glyph->index,
FT_KERNING_MODE_DEFAULT, &delta );
pen_x += delta.x >> 6;
}
/* 保存當前筆位置 */
glyph->pos.x = pen_x;
glyph->pos.y = pen_y;
error = FT_Load_Glyph(face,glyph_index,FT_LOAD_DEFAULT);
if ( error ) continue;
error = FT_Get_Glyph( face->glyph, &glyph->image );
if ( error ) continue;
/* 現在平移字形圖像 */
FT_Glyph_Transform( glyph->image, 0, &glyph->pos );
pen_x += slot->advance.x >> 6;
previous = glyph->index;
/* 增加字形的數量 */
glyph++;
}
/* 計算已裝載的字形的數量 */
num_glyphs = glyph - glyphs;
註意,這個時候平移字形有若幹好處。第一是當我們計算字元串的邊界框時不需要平移字形 bbox。代碼將會
變成這樣:
void compute_string_bbox( FT_BBox *abbox )
{
FT_BBox bbox;
bbox.xMin = bbox.yMin = 32000;
bbox.xMax = bbox.yMax = -32000;
for ( n = 0; n < num_glyphs; n++ )
{
FT_BBox glyph_bbox;
FT_Glyph_Get_CBox( glyphs[n], &glyph_bbox );
if (glyph_bbox.xMin < bbox.xMin)
bbox.xMin = glyph_bbox.xMin;
if (glyph_bbox.yMin < bbox.yMin)
bbox.yMin = glyph_bbox.yMin;
if (glyph_bbox.xMax > bbox.xMax)
bbox.xMax = glyph_bbox.xMax;
if (glyph_bbox.yMax > bbox.yMax)
bbox.yMax = glyph_bbox.yMax;
}
if ( bbox.xMin > bbox.xMax )
{
bbox.xMin = 0;
bbox.yMin = 0;
bbox.xMax = 0;
bbox.yMax = 0;
}
*abbox = bbox;
}
更詳細描述: compute_string_bbox 函數現在可以計算一個已轉換的字形字元串的邊界框。例如,我們可以做
如下的事情:
FT_BBox bbox;
FT_Matrix matrix;
FT_Vector delta;
... 裝載字形序列 ...
... 設置 "matrix" 和 "delta" ...
/* 變換字形 */
for ( n = 0; n < num_glyphs; n++ )
FT_Glyph_Transform( glyphs[n].image, &matrix, &delta );
/* 計算已變換字形的邊界框 */
compute_string_bbox( &bbox );

 

 

 

#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <linux/fb.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <math.h>
#include <wchar.h>
#include <ft2build.h>
#include <stdlib.h>
#include FT_FREETYPE_H
#include FT_GLYPH_H   
#include <linux/font.h>

#define MAX_GLYPHS 100

typedef struct TGlyph_
{
FT_UInt index; /* ???? */
FT_Vector pos; /* ????????? */
FT_Glyph image; /* ???? */
} TGlyph, *PGlyph;


    int fd_fb;
    struct fb_var_screeninfo var;    /* Current var */
    struct fb_fix_screeninfo fix;    /* Current fix */
    int screen_size;                    //空間大小
    unsigned char *fb_mem;                //framebaffer空間地址
    unsigned int line_width;
    unsigned int pixel_width;

void lcd_put_pixel(int x, int y, unsigned int color)
{
    unsigned char *pen_8 = fb_mem + y * line_width + x * pixel_width;     //當前像素對應記憶體位置
    unsigned short *pen_16;
    unsigned int *pen_32;

    unsigned int red, blue, green;
    
    pen_16 = (unsigned short *)pen_8;
    pen_32 = (unsigned int *)pen_8;
    
    switch(var.bits_per_pixel)
    {
        case 8:
        {
            *pen_8 = color;            //對應調色板顏色
            
            break;
        }
        case 16:
        {
            /* 5*6*5 */
            red   = (color >> 16) & 0xff;
            green = (color >> 8) & 0xff;
            blue  = (color >> 0) & 0xff;

            color = ((red >> 3 ) << 11) | ((green >> 2) << 5) | ( blue >> 3);
            
            /* 顏色數據為高位 */
            *pen_16 = color;
            
            break;
        }
        case 32:
        {
            *pen_32 = color;
            break;
        }
        
    }

}

void draw_bitmap( FT_Bitmap*  bitmap,
             FT_Int      x,
             FT_Int      y)
{
  FT_Int  i, j, p, q;
  FT_Int  x_max = x + bitmap->width;
  FT_Int  y_max = y + bitmap->rows;

    printf("x = %d, y = %d\n", x, y);
    
  for ( i = x, p = 0; i < x_max; i++, p++ )
  {
    for ( j = y, q = 0; j < y_max; j++, q++ )
    {
      if ( i < 0      || j < 0       ||
           i >= var.xres || j >= var.yres )
        continue; 

    //  image[j][i] |= bitmap->buffer[q * bitmap->width + p];

        lcd_put_pixel(i, j, bitmap->buffer[q * bitmap->width + p]);
    }
  }
}

int  Get_Glphs_Frm_Wstr(FT_Face face, wchar_t * wstr, TGlyph glyphs[])
{
    PGlyph glyph = glyphs;
    FT_GlyphSlot  slot = face->glyph;
    int n;
    int pen_x =0;
    int pen_y = 0;
    int error;
    
    for(n = 0; n < wcslen(wstr); n++ )
    { 
        glyph->index = FT_Get_Char_Index( face, wstr[n] );
        /* 保存當前筆位置 */
        glyph->pos.x = pen_x;
        glyph->pos.y = pen_y;
        /* load 是把glyph 載入到face->glyph */
        error = FT_Load_Glyph(face, glyph->index, FT_LOAD_DEFAULT);
        if ( error )
            continue;
        
        error = FT_Get_Glyph( face->glyph, &glyph->image );
        if ( error )
            continue;    
        
        /* 現在平移字形圖像 */
        /* 使得glyph->image裡面有位置信息 */
        FT_Glyph_Transform( glyph->image, 0, &glyph->pos );

        pen_x += slot->advance.x;        /* 1 / 64 piont */

        /* 增加字形的數量 */
        glyph++;

    }
    /* 計算已裝載的字形的數量 */

    return (glyph - glyphs);

}

void compute_string_bbox( TGlyph glyphs[], FT_UInt num_glyphs, FT_BBox *abbox )
{
    FT_BBox bbox;
    int n;
    
    bbox.xMin = bbox.yMin = 32000;
    bbox.xMax = bbox.yMax = -32000;
    
    for ( n = 0; n < num_glyphs; n++ )
    {
        FT_BBox glyph_bbox;
        FT_Glyph_Get_CBox( glyphs[n].image, FT_GLYPH_BBOX_TRUNCATE, &glyph_bbox );
        if (glyph_bbox.xMin < bbox.xMin)
            bbox.xMin = glyph_bbox.xMin;
        if (glyph_bbox.yMin < bbox.yMin)
            bbox.yMin = glyph_bbox.yMin;
        if (glyph_bbox.xMax > bbox.xMax)
            bbox.xMax = glyph_bbox.xMax;
        if (glyph_bbox.yMax > bbox.yMax)
            bbox.yMax = glyph_bbox.yMax;
    }
    *abbox = bbox;
}

void Draw_Glyphs(TGlyph glyphs[], FT_UInt num_glyphs,    FT_Vector pen)
{
    int n;
    int error;
    
    for ( n = 0; n < num_glyphs; n++ )
    {
        FT_Glyph_Transform( glyphs[n].image, 0, &pen );
        error = FT_Glyph_To_Bitmap(&glyphs[n].image, FT_RENDER_MODE_NORMAL, 0, 1 ); /* 沒有附加的平移*//* 銷毀 "image" 指向的副本 */
        if ( !error )
        {
            FT_BitmapGlyph bit = (FT_BitmapGlyph)glyphs[n].image;
            draw_bitmap( &bit->bitmap, bit->left, var.yres - bit->top);
            FT_Done_Glyph( glyphs[n].image );
        }

    }
}


int main(int argc, char **argv)
{
    wchar_t * str1 = L"陳志朋gif";
    wchar_t * str2 = L"hello the world";
    FT_BBox bbox;  
    FT_Glyph  glyph;
    
    
      FT_Library    library;
      FT_Face       face;
    FT_Vector     pen;                    /* untransformed origin  */
    FT_GlyphSlot  slot;
    FT_Matrix     matrix;                 /* transformation matrix */

    TGlyph glyphs[MAX_GLYPHS]; /* ??? */
    FT_UInt num_glyphs;
    
    int error;
    double angle;
    int i;
    int line_box_ymin = 10000;
    int line_box_ymax = 0;

    int line_box_width;
    int line_box_hight;
        
    if(argc != 2)
    {
        printf("Usage : %s <font_file> <angle>\n",argv[0]);
        return -1;
    }

    
    fd_fb = open("/dev/fb0",O_RDWR);
    if(fd_fb < 0)
    {
        printf("can't open /dev/fb0 \n");
        return -1;
    }
    if(ioctl(fd_fb, FBIOGET_VSCREENINFO, &var))
    {
        printf("can't get var \n");
        return -1;    
    }
    if(ioctl(fd_fb, FBIOGET_FSCREENINFO, &fix))
    {
        printf("can't get fix \n");
        return -1;    
    }
    
    screen_size = var.xres * var.yres * var.bits_per_pixel / 8;    //單位位元組
    line_width = var.xres *  var.bits_per_pixel / 8;
    pixel_width =  var.bits_per_pixel / 8;
    
    fb_mem = (unsigned char *)mmap(NULL, screen_size, \
        PROT_READ | PROT_WRITE, MAP_SHARED, fd_fb, 0);
    if(fb_mem == (unsigned char *) -1)
    {
        printf("can't mmap \n");
        return -1;
    }
    memset(fb_mem, 0, screen_size);

    
    /* 顯示矢量文字 */
    
    error = FT_Init_FreeType( &library );              /* initialize library */
    /* error handling omitted */

    error = FT_New_Face( library, argv[1], 0, &face ); /* create face object */
    /* error handling omitted */

    slot = face->glyph;
    
    FT_Set_Pixel_Sizes(face, 24, 0);
  
    num_glyphs =  Get_Glphs_Frm_Wstr(face, str1, glyphs);
    compute_string_bbox(glyphs, num_glyphs, &bbox);

    line_box_width = bbox.xMax - bbox.xMin;
    line_box_hight = bbox.yMax - bbox.yMin;

    pen.x = (var.xres - line_box_width) / 2 * 64;
    pen.y = (var.yres - line_box_hight) / 2 * 64;

    Draw_Glyphs(glyphs, num_glyphs, pen);

    num_glyphs =  Get_Glphs_Frm_Wstr(face, str2, glyphs);
    compute_string_bbox(glyphs, num_glyphs, &bbox);

    line_box_width = bbox.xMax - bbox.xMin;
    line_box_hight = bbox.yMax - bbox.yMin;

    pen.x = (var.xres - line_box_width) / 2 * 64;
    pen.y = pen.y - 24 * 64;
    
    Draw_Glyphs(glyphs, num_glyphs, pen);

}

 

中心雙行輸出

 

4.6.2 渲染一個已變換的字形序列
無論如何,如果我們想重用字形來以不同的角度或變換方式繪製字元串,直接變換序列中的字形都不是
一個好主意。更好的方法是在字形被渲染前執行放射變換,如下麵的代碼所示:
FT_Vector start;
FT_Matrix transform;
/* 獲取原始字形序列的 bbox */
compute_string_bbox( &string_bbox );
/* 計算整數象素表示的字元串尺度 */
string_width = (string_bbox.xMax - string_bbox.xMin) / 64;
string_height = (string_bbox.yMax - string_bbox.yMin) / 64;
/* 設置 26.6 笛卡兒空間表示的筆起始位置 */
start.x = ( ( my_target_width - string_width ) / 2 ) * 64;
start.y = ( ( my_target_height - string_height ) / 2 ) * 64;
/* 設置變換(旋轉) */
matrix.xx = (FT_Fixed)( cos( angle ) * 0x10000L );
matrix.xy = (FT_Fixed)(-sin( angle ) * 0x10000L );
matrix.yx = (FT_Fixed)( sin( angle ) * 0x10000L );
matrix.yy = (FT_Fixed)( cos( angle ) * 0x10000L );
for ( n = 0; n < num_glyphs; n++ )
{
FT_Glyph image;
FT_Vector pen;
FT_BBox bbox;
/* 創建原始字形的副本 */
error = FT_Glyph_Copy( glyphs[n].image, &image );
if ( error ) continue;
/* 變換副本(這將平移它到正確的位置) */
FT_Glyph_Transform( image, &matrix, &start );
/* 檢查邊界框;如果已變換的字形圖像不在*/
/* 我們的目標錶面中,我們可以避免渲染它 */
FT_Glyph_Get_CBox( image, ft_glyph_bbox_pixels, &bbox );
if ( bbox.xMax <= 0 || bbox.xMin >= my_target_width ||
bbox.yMax <= 0 || bbox.yMin >= my_target_height )
continue;
/* 把字形圖像轉換為點陣圖(銷毀字形的副本!) */
error = FT_Glyph_To_Bitmap(
&image,
FT_RENDER_MODE_NORMAL,
0, /* 沒有附加的平移*/
1 ); /* 銷毀 "image" 指向的副本 */
if ( !error )
{
FT_BitmapGlyph bit = (FT_BitmapGlyph)image;
my_draw_bitmap( bitmap->bitmap,
bitmap->left,
my_target_height - bitmap->top );
FT_Done_Glyph( image );
}
}
這份代碼相對於原始版本有少許改變:
z 我們沒改變原始的字形圖像,而是變換該字形圖像的拷貝。
z 我們執行“剪取”操作以處理渲染和繪製的字形不在我們的目標錶面(surface)的情況。
z 當調用 FT_Glyhp_To_Bitmap 時,我們總是銷毀字形圖像的拷貝,這是為了銷毀已變換的圖像。註意,即
使當這個函數返回錯誤碼,該圖像依然會被銷毀(這就是為什麼 FT_Done_Glyph 只在複合語句中被調用
的原因)。
z 平移字形序列到起始筆位置集成到 FT_Glyph_Transform 函數,而不是 FT_Glyph_To_Bitmap 函數。
可以多次調用這個函數以渲染字元串到不同角度的,或者甚至改變計算 start 的方法以移動它到另外的地
方。無論如何,要註意通常的實現會使用一個字形緩衝以減少記憶體消耗。據個例子,讓我們假定我們的字元

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

-Advertisement-
Play Games
更多相關文章
  • 1.創建表並插入數據 在Sql Server2008中創建測試資料庫Test,接著創建資料庫表並插入數據,sql代碼如下: 執行完上述sql代碼以後我們會發現在Test資料庫中多出了一張emp_pay表,資料庫表的內容如下圖所示: 2.無索引查找 從上圖我們可以看出資料庫中存儲的數據排列順序與我們插 ...
  • 在資料庫中單獨創建一張表,保存當前存儲狀態,“存儲過程” 設置訪問條件root初始值為“0” 如果root值不為0的時候就不可訪問併進行相關操作。 在事務執行前將root值設置為1,事務結束後將root值設置為0. ...
  • Transact-SQL中的存儲過程,非常類似於Java語言中的方法,它可以重覆調用。當存儲過程執行一次後,可以將語句緩存中,這樣下次執行的時候直接使用緩存中的語句。這樣就可以提高存儲過程的性能。 Ø 存儲過程的概念 存儲過程Procedure是一組為了完成特定功能的SQL語句集合,經編譯後存儲在數 ...
  • LINQ的書寫格式如下: from 臨時變數 in 集合對象或資料庫對象 where 條件表達式 [order by條件] select 臨時變數中被查詢的值 [group by 條件] Lambda表達式的書寫格式如下: (參數列表) => 表達式或者語句塊 其中: 參數個數:可以有多個參數,一個 ...
  • ...
  • 開發的時候,寫了個很簡單的Sql ,大概就是 總數除以數量 得出的平均值。看起來很平常是不是!簡單來說就是 Total / Count 嘛!最多轉個2位小數用Convert就完事了對不對。 但是呢,有些數據的Count值本身是就是0的。然後就會報遇到以0作為除數的錯誤的問題了啊~ 然後演示幾種可能出 ...
  • 問題 我們在生產環境中使用SQLite時中發現建表報“table xxx already exists”錯誤,但DB文件中並沒有該表。後面才發現這個是SQLite在實現過程中的一個bug,而這個bug與數據字典的一致性相關,下麵這篇文章主要討論SQLite的緩存機制,以及緩存一致性實現的策略,希望對 ...
  • 昨天突然被問到traceroute的原理,一時竟也說不出來,有些命令平時雖然經常在用,但實際原理確並不瞭解,趁這次機會就來梳理一下。 traceroute:是網路診斷中,用來分析IP包經過那些路由的命令。 學前知識: IP包中有個欄位TTL,這個是最大跳轉次數的欄位,每經過一個路由器,值會-1,當值 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...