java.security.KeyException

在centos系统上用openjdk1.7, 用https请求微信api,报错如下

Caused by: java.security.ProviderException: java.security.KeyException
at sun.security.ec.ECKeyPairGenerator.generateKeyPair(ECKeyPairGenerator.java:146)
at java.security.KeyPairGenerator$Delegate.generateKeyPair(KeyPairGenerator.java:704)
at sun.security.ssl.ECDHCrypt.<init>(ECDHCrypt.java:78)
at sun.security.ssl.ClientHandshaker.serverKeyExchange(ClientHandshaker.java:717)
at sun.security.ssl.ClientHandshaker.processMessage(ClientHandshaker.java:278)
at sun.security.ssl.Handshaker.processLoop(Handshaker.java:913)
at sun.security.ssl.Handshaker.process_record(Handshaker.java:849)
at sun.security.ssl.SSLSocketImpl.readRecord(SSLSocketImpl.java:1035)
at sun.security.ssl.SSLSocketImpl.performInitialHandshake(SSLSocketImpl.java:1344)
at sun.security.ssl.SSLSocketImpl.startHandshake(SSLSocketImpl.java:1371)
... 52 more
Caused by: java.security.KeyException
at sun.security.ec.ECKeyPairGenerator.generateECKeyPair(Native Method)
at sun.security.ec.ECKeyPairGenerator.generateKeyPair(ECKeyPairGenerator.java:126)
... 61 more

解决方案

升级nss

sudo yum upgrade nss

使用springboot发送邮件可能存在的异常

使用springboot 发送邮件的时候存在异常

  1. handshake_failure异常

    Mail server connection failed; nested exception is javax.mail.MessagingException: Could not connect to SMTP host: smtp.qq.com, port: 465;
    nested exception is: javax.NET.ssl.SSLHandshakeException: Received fatal alert: handshake_failure. Failed messages: javax.mail. MessagingException: Could not connect to SMTP host: smtp.qq.com, port: 465;
    nested exception is:     javax.net.ssl.SSLHandshakeException: Received fatal alert: handshake_failure
    

    这个问题在jdk1.8中出现,通过替换jdk1.8安装目录/Library/Java/JavaVirtualMachines/jdk1.8.0_101.jdk/Contents/Home/jre/lib/security包路径下的US_export_policy.jarUS_export_policy.jar两个文件来解决。使用jdk1.7版本同样目录下的这两个文件替换。或者到这里下载http://www.oracle.com/technetwork/java/javase/downloads/jce-7-download-432124.html

  2. the trustAnchors parameter must be non-empty

    可能原因是使用的jdk1.7中/Library/Java/JavaVirtualMachines/jdk1.8.0_101.jdk/Contents/Home/jre/lib/security路径下缺少证书,由于我的电脑安装两个jdk版本,我的做法是拷贝1.8版本的/Library/Java/JavaVirtualMachines/jdk1.8.0_101.jdk/Contents/Home/jre/lib/security文件夹到1.7版本,或者到同事的电脑的拷贝。

hdfs伪分布式环境搭建

本地环境是mac

安装虚拟机,设置基本环境

  1. 安装vm虚拟机,并安装centos7系统,拷贝多个节点。变成node1,node2,node3
  2. 设置所有节点的自定义IP,保证在同一个网段
    DNS:172.16.153.2

  3. 关闭所有节点的防火墙

    systemctl stop firewalld
    
  4. 设置免登录

    ssh-keygen -t dsa -P '' -f ~/.ssh/id_dsa && cat ~/.ssh/id_dsa.pub >> ~/.ssh/authorized_keys
    

    把本地的.ssh/id_rsa.pub 拷贝到需要控制的节点的~/.ssh/authorized_keys文件中

  5. 修改本地环境的/etc/hosts

    sudo vim /etc/hosts
    172.16.153.10    node1
    172.16.153.20    node2
    172.16.153.30    node3
    

安装HDSF

假设node1是namenode,node2和node3是datanode,secondarynode在node2上面;

设置node1

解压hadoop-1.2.1到/home/hadoop-1.2,cd到/home/hadoop-1.2目录下;

  1. 设置core-site.xml内容(配置查考docs/core-default.html)

    vim conf/core-site.xml
    
    <configuration>
     <property>
         <name>fs.default.name</name>
         <value>hdfs://node1:9000</value>
     </property>
     <property>
         <name>hadoop.tmp.dir</name>
         <value>/opt/hadoop-1.2</value>
     </property>
    </configuration>
    

    fs.default.name属性设置namenode;
    hadoop.tmp.dir设置临时文件存放位置,默认是在/tmp下面, 这个目录重启之后数据就没有咯。所以需要修改。

  2. 设置hdfs-site.xml(配置查考docs/hdfs-default.html)

    vim conf/hdfs-site.xml
    
    <configuration>
        <property>
             <name>dfs.replication</name>
             <value>2</value>
         </property>
    </configuration>
    

    dfs.replication设置datanode节点个数,默认是3,这里我们是datanode是node2和node3,所以设置为2。

  3. 设置datanode

    vim conf/slaves
    
    node2
    node3
    
  4. 设置secondarynode,这里是设置在node2上

    vim conf/master
    
    node2            
    
  5. 设置 JAVA_HOME变量

    vim conf/hadoop-env.sh
    export JAVA_HOME=/usr/lib/jvm/java-1.8.0-openjdk-1.8.0.102-1.b14.el7_2.x86_64/jre
    
  6. 设置datanode免密码登录
    这里我需要在namenode(node1)统一启动,所以我需要从node1到node2,node3免密码登录

    ssh root@node2
    ssh root@node3
    

    这两个操作都是免密码登录的。

至此,namenode设置完成了。

最后需要拷贝一样的hadoop环境到node2和node3

scp /home/hadoop-1.2 root@node2:/home/
scp /home/hadoop-1.2 root@node3:/home/

注意点:

保证hadoop的包环境完全一致

设置免密码登录

设置JAVA_HOME

关闭防火墙

在本地通过浏览器访问:http://node1:50070就能看到成果了。

centos安装jps命令

hadoop里面使用到jps命令,centos7 自带java环境,但是没有jps命令,需要自行安装:

yum search java-1.8.0
yum install -y java-1.8.0-openjdk-devel.x86_64

安装完成就会有jps命令

mybatis中if里面字符串比较

mybatis中如果使用如下字符串比较会报错:

<choose>
    <when test="module == 'd'">
        topic_module_dynamic b
    </when>
</choose>

改写成这样就ok了:

<choose>
    <when test="module == 'd'.toString()">
        topic_module_dynamic b
    </when>
</choose>

或者

<choose>
    <when test='module == "d"'>
        topic_module_dynamic b
    </when>
</choose>    

mysql时间转换函数

date_add(current_timestamp,interval - 1 month)

date_format(date,’%Y%m%d’);

last_day(‘2007-03-01’);

1 某一天的所在月的第一天:

select date_add(date_add(last_day('2008-02-01'),interval 1 day),interval -1 month);

某一天的所在月的最后一天:

select last_day('2008-02-01');

使用jaxb对字符串操作

对象转xml字符串

/**
 * 
 * @param obj
 * @XmlAccessorType(XmlAccessType.FIELD) @XmlRootElement(name = "xml")
 * @param encoding
 * @return
 */
public static String convertToXml(Object obj, String encoding) {
    String result = null;
    try {
        JAXBContext context = JAXBContext.newInstance(obj.getClass());
        Marshaller marshaller = context.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.setProperty(Marshaller.JAXB_ENCODING, encoding);

        StringWriter writer = new StringWriter();
        marshaller.marshal(obj, writer);
        result = writer.toString();
    } catch (Exception e) {
        e.printStackTrace();
    }

    return result;
}

xml字符串转java对象

public static <T> T converToJavaBean(String xml, Class<T> clazz) {
    try {
        JAXBContext context = JAXBContext.newInstance(clazz);
        Unmarshaller unmarshaller = context.createUnmarshaller();
        AbstractUnmarshallerImpl abstractUnmarshallerImpl = (AbstractUnmarshallerImpl) unmarshaller;
        return (T) abstractUnmarshallerImpl.unmarshal(new StringReader(xml));
    } catch (Exception e) {
        throw new RuntimeException(e);
    }
}    

可能出现的错误:SAXNotRecognizedException: Feature ‘http://javax.xml.XMLConstants/feature/secure-

本地模式运行是好的,服务器模式下运行报错。原因是本地的jdk运行环境是1.7,服务器运行环境是1.8。所以导致此异常。
开发环境里面可以设置eclipse的jre

mysql语句中常见的技巧

update topic_config set value = cast(value as DECIMAL) + 1

使用 cast(value as DECIMAL) 把varchar类型的value值转成int类型然后进行计算

insert into dictionary (value) values ((select * from (select ifnull(max(value),0) from dictionary where type = 2) tmp) + 1)        

insert语句中包含select语句,获取复合条件的最大值然后加1,常见于排序字段

insert into topic_module_subject_category
    (type,text,ptype)
    values
    ( 
    <choose>
        <when test="ptype == 0">
            (select * from (select ifnull(max(type),100000) from topic_module_subject_category where ptype = #{ptype}) tmp) + 10000
        </when>
        <otherwise>
            (select * from (select ifnull(max(type),${ptype}) from topic_module_subject_category where ptype = #{ptype}) tmp) + 1
        </otherwise>
    </choose>
    ,#{text},#{ptype})

ExtJS中组件使用注意点

使用ext组件的时候如果自定义一个公共的组件,最好使用如下方式:

Ext.define('MyComponent',    {
    extend:'Ext.panel.Panel',
    initComponent:function() {
        var me = this;
        Ext.applyIf(me,{
            items:[
                ...
            ]
        });
        me.callParent(arguments);
    }

});

如果使用子面量的形式,比如:

Ext.define('MyComponent',{
    extend:'Ext.panel.Panel',
    items:[
        ...
    ]
});

这种写法会导致组件里面有些store是单例,只会创建一次。则会出现组件的状态共用问题。

maven项目强制不热重启配置

  1. 命令方式

    clean jetty:run -Djetty.port=8081 -Djetty.reload=manual
    
  2. pom.xml中配置

    <plugin>
            <groupId>org.mortbay.jetty</groupId>
            <artifactId>maven-jetty-plugin</artifactId>
            <version>6.1.26</version>
            <configuration>
                <reload>manual</reload>
                <connectors>
                    <connector implementation="org.mortbay.jetty.nio.SelectChannelConnector">
                        <port>8080</port>
                        <maxIdleTime>30000</maxIdleTime>
                    </connector>
                </connectors>
                <scanIntervalSeconds>3</scanIntervalSeconds>
                <webAppSourceDirectory>src/main/webapp</webAppSourceDirectory>
                <contextPath>/</contextPath>
            </configuration>
        </plugin>