Butterknife全方位解析

概述

Butterknife是供职于Square公司的JakeWharton大神开发的开源库,使用这个库,在AS中搭配Android ButterKnife Zelezny插件,可以大大提高开发的效率,从此摆脱繁琐的findViewById(int id),也不用自己手动@bind(int id) , 直接用插件生成即可。本篇博客将对Butterknife进行深入解析。

项目地址: JakeWharton/butterknife

ButterKnife有以下优点:

1、强大的View绑定和Click事件处理功能,简化代码,提升开发效率

2、方便的处理Adapter里的ViewHolder绑定问题

3、运行时不会影响APP效率,使用配置方便

4、代码清晰,可读性强

如何导入ButterKnife

在项目的build.grade文件中进行如下配置:

 
  1. buildscript { 
  2.     repositories { 
  3.         jcenter() 
  4.         mavenCentral() 
  5.         maven { 
  6.             url "https://plugins.gradle.org/m2/" 
  7.         } 
  8.     } 
  9.     dependencies { 
  10.         classpath 'com.android.tools.build:gradle:2.2.0' 
  11.         //这里配置 apt 供butterknife使用 
  12.         classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8' 
  13.  
  14.     } 
  15.  

例如:

 
  1. buildscript { 
  2.     repositories { 
  3.         jcenter() 
  4.         mavenCentral() 
  5.         maven { 
  6.             url "https://plugins.gradle.org/m2/" 
  7.         } 
  8.  
  9.     } 
  10.  
  11.     dependencies { 
  12.         classpath 'com.android.tools.build:gradle:2.2.2' 
  13.         classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8' 
  14.     } 
  15.  
  16. allprojects { 
  17.     repositories { 
  18.         jcenter() 
  19.     } 
  20.  
  21. task clean(type: Delete) { 
  22.     delete rootProject.buildDir 
  23.  

在app的build.grade文件中进行如下配置:

 
  1. apply plugin: 'com.android.application' 
  2. apply plugin: 'com.neenbedankt.android-apt' 
  3.  
  4. android{...} 
  5.  
  6. dependencies { 
  7.     //视图绑定 butterknife 
  8.     compile 'com.jakewharton:butterknife:8.4.0' 
  9.     apt 'com.jakewharton:butterknife-compiler:8.4.0' 
  10.  

例如:

 
  1. apply plugin: 'com.android.application' 
  2. apply plugin: 'android-apt' 
  3.  
  4. android { 
  5.     compileSdkVersion 24 
  6.     buildToolsVersion "24.0.3" 
  7.  
  8.     defaultConfig { 
  9.  
  10.         minSdkVersion 14 
  11.         targetSdkVersion 24 
  12.         versionCode 1 
  13.         versionName "1.0" 
  14.     } 
  15.     buildTypes { 
  16.         release { 
  17.             minifyEnabled false 
  18.             proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 
  19.         } 
  20.     } 
  21.  
  22. dependencies { 
  23.     compile fileTree(dir: 'libs', include: ['*.jar']) 
  24.  
  25.     compile 'com.jakewharton:butterknife:8.4.0' 
  26.     apt 'com.jakewharton:butterknife-compiler:8.4.0' 
  27.  

如何使用ButterKnife

1) 由于每次都要在Activity中的onCreate绑定Activity,所以个人建议写一个BaseActivity完成绑定,子类继承即可

注:ButterKnife.bind(this);绑定Activity 必须在setContentView之后:

实现如下(FragmentActivity 实现一样):

 
  1. public abstract class BaseActivity extends Activity {   
  2.     public abstract int getContentViewId();   
  3.  
  4.     @Override   
  5.     protected void onCreate(Bundle savedInstanceState) {   
  6.         super.onCreate(savedInstanceState);   
  7.         setContentView(getContentViewId());   
  8.         ButterKnife.bind(this);   
  9.         initAllMembersView(savedInstanceState);   
  10.     }   
  11.  
  12.     protected abstract void initAllMembersView(Bundle savedInstanceState);   
  13.  
  14.     @Override   
  15.     protected void onDestroy() {   
  16.         super.onDestroy();   
  17.         ButterKnife.unbind(this);//解除绑定,官方文档只对fragment做了解绑   
  18.     }   
  19. }    

2) 绑定fragment

 
  1. public abstract class BaseFragment extends Fragment {   
  2.     public abstract int getContentViewId();   
  3.     protected Context context;   
  4.     protected View mRootView;   
  5.  
  6.     @Nullable   
  7.     @Override   
  8.     public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {   
  9.         mRootView =inflater.inflate(getContentViewId(),container,false);   
  10.         ButterKnife.bind(this,mRootView);//绑定framgent   
  11.         this.context = getActivity();   
  12.         initAllMembersView(savedInstanceState);   
  13.         return mRootView;   
  14.     }   
  15.  
  16.     protected abstract void initAllMembersView(Bundle savedInstanceState);   
  17.  
  18.     @Override   
  19.     public void onDestroyView() {   
  20.         super.onDestroyView();   
  21.         ButterKnife.unbind(this);//解绑   
  22.     }   
  23. }    

3) 控件id 注解: @BindView()

 
  1. package com.myl.test; 
  2.  
  3. import android.support.v7.app.AppCompatActivity; 
  4. import android.os.Bundle; 
  5. import android.widget.Button; 
  6.  
  7. import butterknife.BindView; 
  8. import butterknife.ButterKnife; 
  9.  
  10. public class ButterknifeActivity extends AppCompatActivity { 
  11.  
  12.     @BindView( R.id.button1 ) 
  13.     public Button button1 ; 
  14.  
  15.     // 注意:button 的修饰类型不能是:private 或者 static 。 否则会报错:错误: @BindView fields must not be private or static. (com.myl.test.ButterknifeActivity.button1) 
  16.  
  17.     @Override 
  18.     protected void onCreate(Bundle savedInstanceState) { 
  19.         super.onCreate(savedInstanceState); 
  20.         setContentView(R.layout.activity_butterknife); 
  21.         //绑定activity 
  22.         ButterKnife.bind( this ) ; 
  23.  
  24.         button1.setText( "I am a button "); 
  25.     } 
  26.  

4) 多个控件id 注解: @BindViews()

 
  1. package com.myl.test; 
  2.  
  3. import android.support.v7.app.AppCompatActivity; 
  4. import android.os.Bundle; 
  5. import android.widget.Button; 
  6. import java.util.List; 
  7. import butterknife.BindViews; 
  8. import butterknife.ButterKnife; 
  9.  
  10. public class Main2Activity extends AppCompatActivity { 
  11.  
  12.     @BindViews({ R.id.button1  , R.id.button2 ,  R.id.button3 }) 
  13.     public List<Button> buttonList ; 
  14.  
  15.     @Override 
  16.     protected void onCreate(Bundle savedInstanceState) { 
  17.         super.onCreate(savedInstanceState); 
  18.         setContentView(R.layout.activity_main2); 
  19.  
  20.         ButterKnife.bind(this); 
  21.  
  22.         buttonList.get( 0 ).setText( "hello 1 "); 
  23.         buttonList.get( 1 ).setText( "hello 2 "); 
  24.         buttonList.get( 2 ).setText( "hello 3 "); 
  25.     } 
  26.  

5) @BindString() :绑定string 字符串

 
  1. package com.myl.test; 
  2.  
  3. import android.os.Bundle; 
  4. import android.support.v7.app.AppCompatActivity; 
  5. import android.widget.Button; 
  6.  
  7. import butterknife.BindString; 
  8. import butterknife.BindView; 
  9. import butterknife.ButterKnife; 
  10.  
  11. public class ButterknifeActivity extends AppCompatActivity { 
  12.  
  13.     @BindView( R.id.button1 ) //绑定button 控件 
  14.     public Button button1 ; 
  15.  
  16.     @BindString( R.string.app_name )  //绑定string 字符串 
  17.     String meg; 
  18.  
  19.     @Override 
  20.     protected void onCreate(Bundle savedInstanceState) { 
  21.         super.onCreate(savedInstanceState); 
  22.         setContentView(R.layout.activity_butterknife); 
  23.  
  24.         //绑定activity 
  25.         ButterKnife.bind( this ) ; 
  26.  
  27.         button1.setText( meg ); 
  28.     } 
  29.  

6) @BindArray() : 绑定string里面array数组

 
  1. <resources> 
  2.     <string name="app_name">校园助手</string> 
  3.  
  4.     <string-array name="city"
  5.         <item>东莞市</item> 
  6.         <item>广州市</item> 
  7.         <item>珠海市</item> 
  8.         <item>肇庆市</item> 
  9.         <item>深圳市</item> 
  10.     </string-array> 
  11.  
  12. </resources> 
  13. ----------------------------------------------------------------- 
  14. package com.myl.test; 
  15.  
  16. import android.os.Bundle; 
  17. import android.support.v7.app.AppCompatActivity; 
  18. import android.widget.Button; 
  19.  
  20. import butterknife.BindArray; 
  21. import butterknife.BindView; 
  22. import butterknife.ButterKnife; 
  23.  
  24. public class ButterknifeActivity extends AppCompatActivity { 
  25.  
  26.     @BindView( R.id.button1 ) //绑定button 控件 
  27.     public Button button1 ; 
  28.  
  29.     @BindArray(R.array.city )  //绑定string里面array数组 
  30.     String [] citys ; 
  31.  
  32.     @Override 
  33.     protected void onCreate(Bundle savedInstanceState) { 
  34.         super.onCreate(savedInstanceState); 
  35.         setContentView(R.layout.activity_butterknife); 
  36.  
  37.         //绑定activity 
  38.         ButterKnife.bind( this ) ; 
  39.  
  40.         button1.setText( citys[0] ); 
  41.     } 
  42.  

7) @BindBitmap( ) : 绑定Bitmap 资源

 
  1. package com.myl.test; 
  2.  
  3. import android.graphics.Bitmap; 
  4. import android.os.Bundle; 
  5. import android.support.v7.app.AppCompatActivity; 
  6. import android.widget.ImageView; 
  7.  
  8. import butterknife.BindBitmap; 
  9. import butterknife.BindView; 
  10. import butterknife.ButterKnife; 
  11.  
  12. public class ButterknifeActivity extends AppCompatActivity { 
  13.  
  14.     @BindView( R.id.imageView ) //绑定ImageView 控件 
  15.     public ImageView imageView ; 
  16.  
  17.     @BindBitmap( R.mipmap.wifi )  //绑定Bitmap 资源 
  18.     public Bitmap wifi_bitmap ; 
  19.  
  20.     @Override 
  21.     protected void onCreate(Bundle savedInstanceState) { 
  22.         super.onCreate(savedInstanceState); 
  23.         setContentView(R.layout.activity_butterknife); 
  24.  
  25.         //绑定activity 
  26.         ButterKnife.bind( this ) ; 
  27.  
  28.         imageView.setImageBitmap( wifi_bitmap ); 
  29.     } 
  30.  

8) @BindColor( ) : 绑定一个颜色值

 
  1. package com.myl.test; 
  2.  
  3. import android.os.Bundle; 
  4. import android.support.v7.app.AppCompatActivity; 
  5. import android.widget.Button; 
  6.  
  7. import butterknife.BindColor; 
  8. import butterknife.BindView; 
  9. import butterknife.ButterKnife; 
  10.  
  11. public class ButterknifeActivity extends AppCompatActivity { 
  12.  
  13.     @BindView( R.id.button1 )  //绑定一个控件 
  14.     public Button button1 ; 
  15.  
  16.     @BindColor( R.color.colorAccent ) int black ;  //绑定一个颜色值 
  17.  
  18.     @Override 
  19.     protected void onCreate(Bundle savedInstanceState) { 
  20.         super.onCreate(savedInstanceState); 
  21.         setContentView(R.layout.activity_butterknife); 
  22.  
  23.         //绑定activity 
  24.         ButterKnife.bind( this ) ; 
  25.  
  26.         button1.setTextColor(  black ); 
  27.  
  28.     } 
  29.  

9) Adapter ViewHolder 绑定

 
  1. public class TestAdapter extends BaseAdapter {   
  2.     private List<String> list;   
  3.     private Context context;   
  4.  
  5.     public TestAdapter(Context context, List<String> list) {   
  6.         this.list = list;   
  7.         this.context = context;   
  8.     }   
  9.  
  10.     @Override   
  11.     public int getCount() {   
  12.         return list==null ? 0 : list.size();   
  13.     }   
  14.  
  15.     @Override   
  16.     public Object getItem(int position) {   
  17.         return list.get(position);   
  18.     }   
  19.  
  20.     @Override   
  21.     public long getItemId(int position) {   
  22.         return position;   
  23.     }   
  24.  
  25.     @Override   
  26.     public View getView(int position, View convertView, ViewGroup parent) {   
  27.         ViewHolder holder;   
  28.         if (convertView == null) {   
  29.             convertView = LayoutInflater.from(context).inflate(R.layout.layout_list_item, null);   
  30.             holder = new ViewHolder(convertView);   
  31.             convertView.setTag(holder);   
  32.         } else {   
  33.             holder = (ViewHolder) convertView.getTag();   
  34.         }   
  35.         holder.textview.setText("item=====" + position);   
  36.         return convertView;   
  37.     }   
  38.  
  39.     static class ViewHolder {   
  40.         @Bind(R.id.hello_world)   
  41.         TextView textview;   
  42.  
  43.         public ViewHolder(View view) {   
  44.             ButterKnife.bind(this, view);   
  45.         }   
  46.     }   
  47. }    

10) 点击事件的绑定:不用声明view,不用setOnClickLisener()就可以绑定点击事件

a. 直接绑定一个方法

 
  1. @OnClick(R.id.submit)   
  2. public void submit(View view) {   
  3.   // TODO submit data to server...   
  4. }    

b. 所有监听方法的参数是可选的

 
  1. @OnClick(R.id.submit)   
  2. public void submit() {   
  3.   // TODO submit data to server...   
  4. }   

c. 定义一个特定类型,它将自动被转换

 
  1. @OnClick(R.id.submit)   
  2. public void sayHi(Button button) {   
  3.   button.setText("Hello!");   
  4. }    

d. 多个view统一处理同一个点击事件,很方便,避免抽方法重复调用的麻烦

 
  1. @OnClick(R.id.submit)   
  2. public void sayHi(Button button) {   
  3.   button.setText("Hello!");   
  4. }   

e. 自定义view可以绑定自己的监听,不指定id

 
  1. public class FancyButton extends Button {   
  2.   @OnClick   
  3.   public void onClick() {   
  4.     // TODO do something!   
  5.   }   
  6. }    

f. 给EditText加addTextChangedListener(即添加多回调方法的监听的使用方法),利用指定回调,实现想回调的方法即可,哪个注解不会用点进去看下源码上的注释

 
  1. @OnTextChanged(value = R.id.mobileEditText, callback = OnTextChanged.Callback.BEFORE_TEXT_CHANGED)   
  2. void beforeTextChanged(CharSequence s, int start, int countint after) {   
  3.  
  4. }   
  5. @OnTextChanged(value = R.id.mobileEditText, callback = OnTextChanged.Callback.TEXT_CHANGED)   
  6. void onTextChanged(CharSequence s, int start, int before, int count) {   
  7.  
  8. }   
  9. @OnTextChanged(value = R.id.mobileEditText, callback = OnTextChanged.Callback.AFTER_TEXT_CHANGED)   
  10. void afterTextChanged(Editable s) {   
  11.  
  12.  

代码混淆

 
  1. -keep class butterknife.** { *; }   
  2. -dontwarn butterknife.internal.**   
  3. -keep class **$$ViewBinder { *; }   
  4.  
  5. -keepclasseswithmembernames class * {   
  6.     @butterknife.* <fields>;   
  7. }   
  8.  
  9. -keepclasseswithmembernames class * {   
  10.     @butterknife.* <methods>;   
  11. }    

Zelezny插件的使用

在AndroidStudio->File->Settings->Plugins->搜索Zelezny下载添加就行 ,可以快速生成对应组件的实例对象,不用手动写。使用时,在要导入注解的Activity 或 Fragment 或 ViewHolder的layout资源代码上,右键——>Generate——Generate ButterKnife Injections,然后就出现如图的选择框。

ButterKnife实现原理

对ButterKnife有过了解人 , 注入字段的方式是使用注解@BindView(R.id.tv_account_name),但首先我们需要在Activity声明注入ButterKnife.bind(Activity activity) 。我们知道,注解分为好几类, 有在源码生效的注解,有在类文件生成时生效的注解,有在运行时生效的注解。分别为RetentionPolicy.SOURCE,RetentionPolicy.CLASS,RetentionPolicy.RUNTIME ,其中以RetentionPolicy.RUNTIME最为消耗性能。而ButterKnife使用的则是编译器时期注入,在使用的时候,需要配置classpath ‘com.neenbedankt.gradle.plugins:android-apt:1.8’ , 这个配置说明,在编译的时候,进行注解处理。要对注解进行处理,则需要继承AbstractProcessor , 在boolean process(Set

ButterKnife实现方式

知晓了注解可以在编译的时候进行处理,那么,我们就可以得到注解的字段属性与所在类 , 进而生成注入文件,生成一个注入类的内部类,再进行字段处理 , 编译之后就会合并到注入类中,达到植入新代码段的目的。例如:我们注入@VInjector(R.id.tv_show) TextView tvShow;我们就可以得到tvShow这个变量与R.id.tv_show这个id的值,然后进行模式化处理injectObject.tvShow = injectObject.findViewById(R.id.tv_show); ,再将代码以内部类的心事加入到组件所在的类中 , 完成一次DI(注入) 。

a) 首先创建一个视图注解

b) 创建一个注解处理器,用来得到注解的属性与所属类

c) 解析注解,分离组合Class与属性

d) 组合Class与属性,生成新的Java File

APT生成的Java File , 以及模式代码

使用Javac , 编译时期生成注入类的子类

项目UML图

简要说明:

主要类:

VInjectProcessor —-> 注解处理器 , 需要配置注解处理器

 
  1. resources 
  2.         - META-INF 
  3.               - services 
  4.                     - javax.annotation.processing.Processor  

Processor内容:

 
  1. com.myl.viewinject.apt.VInjectProcessor   # 指定处理器全类名  

VInjectHandler —-> 注解处理类 , 主要进行注入类与注解字段进行解析与封装,将同类的字段使用map集合进行映射。exp: Map

自定义ButterKnife具体实现

因微信字数限制,请点击左下角原文链接查看!~







本文作者:佚名
来源:51CTO

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/455487.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

论文笔记 Aggregated Residual Transformations for Deep Neural Networks

这篇文章构建了一个基本“Block”&#xff0c;并在此“Block”基础上引入了一个新的维度“cardinality”(字母“C”在图、表中表示这一维度)。深度网络的另外两个维度分别为depth&#xff08;层数&#xff09;、width&#xff08;width指一个层的channel的数目&#xff09;。 首…

matlab 归一化_机器学习中如何用Fscore进行特征选择(附Matlab代码)

作者&#xff1a;kervin编辑&#xff1a;阿吉 目前&#xff0c;机器学习在脑科学领域的应用可谓广泛而深入&#xff0c;不论你是做EEG/ERP研究&#xff0c;还是做MRI研究&#xff0c;都会看到机器学习的身影。机器学习最简单或者最常用的一个应用方向是分类&#xff0c;…

IOS安装CocoaPods完整流程

作为一个底层系统大菜鸟,又搞过几年ios来说,安装一个CocoaPods是一件蛋痛的事~ 说懂又懂,说不懂又不懂. 由于安装过程比較复杂,步骤较多,而网上教程又比較零散,并且有一些是扯蛋的,所以本篇文章主要从头到位依据自身安装经历记录每一条终端指令,至于里面的原理和一些概念性的东…

linux 修改时区_教你在Centos8中更改时区

对于许多与系统相关的任务和进程&#xff0c;使用正确的时区是必不可少的。例如&#xff0c;cron守护进程使用系统的时区执行cron作业&#xff0c;日志文件中的时间戳基于同一系统的时区。环 境CentOS 8检查现在的时区timedatectl是一个命令行实用程序&#xff0c;允许您查看和…

vb6在后台将窗体保存到图片_如何将寺库网多个商品图片一键分类保存到一个目录...

寺库网是全球最大的奢侈品网上在线购物平台&#xff0c;那么我们怎样可以从寺库网上一键批量采集到多个宝贝商品图片&#xff0c;并分类保存到电脑呢&#xff1f;今天小编给大家带来一款专业电商图片链接采集软件【载图助手】&#xff0c;它支持平台高达141个&#xff0c;均可支…

浮动与定位

2019独角兽企业重金招聘Python工程师标准>>> 一.浮动:float:一个元素浮动时,其他内容会"环绕"该元素. 浮动元素的外边距不会合并浮动的元素不能超出其包含快的内边界浮动元素彼此会避免重叠浮动元素的顶端不能比之前所有浮动元素或块级元素的顶端更高如果…

驱动级的自动按键_Aqara全自动智能推拉锁D100,体验全自动开门的便捷

大家好&#xff0c;我是梦想是个猪&#xff0c;今天为大家带来的是一篇智能门锁的使用体验。前言家里的这张门陆陆续续的换了好几把智能门锁了&#xff0c;也体验了好几种不同的开锁方式。最开始开发商给安装的是一把指纹和把手分离的那种款式&#xff0c;开锁的时候需要先输入…

深度学习综述

摘要&#xff1a; 深度学习可以完成需要高度抽象特征的人工智能任务&#xff0c;如语音识别、图像识别和检索、自然语言理解等。深层模型是包含多个隐藏层的人工神经网络&#xff0c;多层非线性结构使其具备强大的特征表达能力和对复杂任务建模能力。训练深层模型是长期以来的难…

mac svn工具_Cornerstone 4 for mac(svn管理工具)

Cornerstone 4 for mac是全新版本的svn管理工具&#xff0c;使用cornerstone for mac 特别版建立的版本控制更利于使用&#xff0c;而且cornerstone 4 特别版全面支持Subversion的功能&#xff0c;这里准备了最新版本的cornerstone for mac 特别版&#xff0c;无需激活&#xf…

webgl获取鼠标形状_三模无线搭配对称手型设计,游戏致胜利器,ROG烈刃2无线鼠标...

要想有效地提升游戏体验&#xff0c;我认为除了电脑主机本身的硬件配置要尽可能的硬核之外&#xff0c;玩游戏时所选配的鼠标、键盘等外设的作用也是不可忽视的&#xff0c;所以很多比较注重游戏体验的游戏爱好者都会选择一款自己用着比较顺手的游戏外设装备。我这次入手的华硕…

qmlcanvas绘制3d图形_透视Matplotlib核心功能和工具包 - 绘制3D图形

关联知识MatplotlibPython线图在此&#xff0c;我们将学习如何创建3D线图。 它类似于2D等效折线图&#xff0c;并且2D折线图的许多属性都结转到3D。我们将在相同的轴上绘制凹凸曲线&#xff0c;并从不同角度查看它们&#xff0c;例如平行视图&#xff0c;顶视图以及围绕z轴的旋…

雷军晒3亿估值,意欲“收编”台湾硬件创业者?

1月13日&#xff0c;在台湾的CSMIC 2015移动互联网两岸年会上&#xff0c;雷军以猎豹移动董事长的身份做了《给年轻创业者的两大方向性建议》演讲。 演讲中雷军表示&#xff0c;“在未来5到10年&#xff0c;特别适合台湾年轻创业者做的两个方向&#xff0c;一个是移动互联网&am…

记事本安卓软件代码设计_用轻量级工具 Notepad3 替代 36 岁的微软记事本

一不小心&#xff0c;微软记事本发布 36 年了。虽然微软记事本自 1985 年 Windows 1.0 以来已包含在所有版本的 Microsoft Windows 中&#xff0c;但它太基础了&#xff0c;以至于几乎所有人都要替代它&#xff0c;比如轻量级的 Notepad3&#xff0c;支持代码高亮/折叠、括号匹…

H.264编码技术

H.264基本概况随着HDTV的兴起&#xff0c;H.264这个规范频频出现在我们眼前&#xff0c;HD-DVD和蓝光DVD均计划采用这一标准进行节目制作。而且自2005年下半年以来&#xff0c;无论是NVIDIA还是ATI都把支持H.264硬件解码加速作为自己最值得夸耀的视频技术。H.264到底是何方“神…

macOS自带Python2.7删除之后的悲剧

2019独角兽企业重金招聘Python工程师标准>>> 起因 今天打开xcode想要开始继续写项目的时候悲剧了. Loading a plug-in failed. The plug-in or one of its prerequisite plug-ins may be missing or damaged and may need to be reinstalled. 尝试用终端打开看看报…

EasyUI 表格点击右键添加或刷新 绑定右键菜单

例1 在HTML页面中设置一个隐藏的菜单&#xff08;前提是已经使用封装的Easyui&#xff09; 代码&#xff1a; <div id"contextMenu_jygl" class"easyui-menu" style"width: 80px; display: none;"> <div …

我在全球最大的同性社交平台那点事

本文作者&#xff1a;夏之冰雪&#xff0c;i春秋签约作家 《我在百度网盘上看到上万条车主个人信息,企业、政府高官信息、各种数据库和无穷无尽的盗版》&#xff0c;一时间&#xff0c;这篇文章就火了&#xff0c;火爆程度另百度猝不及防。 其实呢&#xff0c;这事真不能全怪百…

关于java的关键字 transient

我们都知道一个对象只要实现了Serilizable接口&#xff0c;这个对象就可以被序列化&#xff0c;Java的这种序列化模式为开发者提供了很多便利&#xff0c;我们可以不必关系具体序列化的过程&#xff0c;只要这个类实现了Serilizable接口&#xff0c;这个的所有属性和方法都会自…

希捷宣布出货双碟装1TB硬盘 单碟500GB上市

希捷今天宣布&#xff0c;单碟容量高达500GB的Barracuda 7200.12 1TB硬盘已经出货。 现有的1TB硬盘产品一般都是三碟装(此前也有四碟装)&#xff0c;包括希捷自家的ST31000340AS&#xff0c;但希捷一进率先将其升级为双碟装&#xff0c;存储密度也达到了329Gb每平方英寸。这样一…

FFMpeg的output_example.c例子分析

该例子讲了如何输出一个libavformat库所支持格式的媒体文件。 &#xff08;1&#xff09;av_register_all()&#xff0c;初始化libavcodec库&#xff0c;并注册所有的编解码器和格式。 &#xff08;2&#xff09;guess_format()&#xff0c;根据文件名来获取输出文件格式&#…